+7
-7
.gitignore
+7
-7
.gitignore
···
1
1
node_modules/
2
-
/elixir_blonk/_build
2
+
/_build
3
3
dist/
4
4
.env
5
5
.DS_Store
···
11
11
/_build/
12
12
13
13
# If you run "mix test --cover", coverage assets end up here.
14
-
/elixir_blonk/cover/
14
+
/cover/
15
15
16
16
# The directory Mix downloads your dependencies sources to.
17
-
/elixir_blonk/deps/
17
+
/deps/
18
18
19
19
# Where 3rd-party dependencies like ExDoc output generated docs.
20
-
/elixir_blonk/doc/
20
+
/doc/
21
21
22
22
# Ignore .fetch files in case you like to edit your project deps locally.
23
-
elixir_blonk/.fetch
23
+
.fetch
24
24
25
25
# If the VM crashes, it generates a dump, let's ignore it too.
26
26
erl_crash.dump
···
35
35
elixir_blonk-*.tar
36
36
37
37
# Ignore assets that are produced by build tools.
38
-
elixir_blonk/priv/static/assets/
38
+
priv/static/assets/
39
39
40
40
# Ignore digested assets cache.
41
-
elixir_blonk/priv/static/cache_manifest.json
41
+
priv/static/cache_manifest.json
42
42
43
43
# In case you use Node.js/npm, you want to ignore these.
44
44
npm-debug.log
+192
-87
README.md
+192
-87
README.md
···
1
-
# Blonk
1
+
# Blonk 🎯
2
2
3
-
## What is Blonk?
4
-
ATProto reddit.
5
-
But it will be different, thats just a thing to say to make people place a vibe in their head.
3
+
**A place to find blips on the radar of your web, with a focus on vibes.**
6
4
7
-
Blonk doesn't have subreddits like a forum. It has loose groups of vibes.
5
+
Blonk is a community-driven content discovery platform built on ATProto that enables organic topic communities to form around "vibes" - interest-based feeds where users submit "blips" (content) to get "grooves" (community engagement).
8
6
9
-
Maybe they will just be subscribers, but maybe a vibe isnt just users it identifies but users who label themselves into it or something.
7
+
## 🌟 Core Concepts
10
8
11
-
We will need to build this in a way where it is consumable on its own website.
12
-
It should have some mechanism where it creates its own lexicon or something for the links?
9
+
### The Blonk Ecosystem
13
10
14
-
I dont know, we are going to walk through this together.
11
+
**🎯 Radar** - The frontpage that surfaces trending content across all vibes
12
+
**🌊 Vibes** - Topic-based communities created by community action (#vibe-your_topic)
13
+
**📡 Blips** - Content submissions to vibes that appear on the radar
14
+
**🎵 Grooves** - Community engagement: `looks_good` or `shit_rips`
15
+
**🏷️ Tags** - Universal labels that connect content across vibes
15
16
16
-
## Building It
17
-
I started off with a simple TypeScript project and got cooking.
17
+
### How It Works
18
18
19
-
You can find that in aeb0dc780a395bf6193a8dc90bd4ab4f82f66d90.
19
+
1. **Vibe Creation**: Communities emerge when `#vibe-topic_name` reaches critical mass
20
+
2. **Content Submission**: Users submit blips to vibes with tags for categorization
21
+
3. **Community Engagement**: Others groove on blips, driving popularity
22
+
4. **Radar Discovery**: Trending tagged content surfaces on the frontpage
23
+
5. **Organic Growth**: Popular content attracts more users to related vibes
20
24
21
-
```
22
-
commit aeb0dc780a395bf6193a8dc90bd4ab4f82f66d90 (HEAD -> main)
23
-
Author: Bobby Grayson <53058768+notactuallytreyanastasio@users.noreply.github.com>
24
-
Date: Thu Jun 19 09:53:56 2025 -0400
25
+
### Community Seeding
25
26
26
-
Initial commit with a little structure.
27
+
- **🔥 Hot Posts**: AI monitors the Bluesky firehose for trending content (>5 replies)
28
+
- **Auto-Population**: Trending external content auto-populates the `bsky_hot` vibe
29
+
- **Community Bootstrap**: Seeds engagement to kickstart organic community growth
27
30
28
-
We've added the ATProto SDK and set up a really basic setup.
29
-
It's all Claude generated, but this is a learning exercise to let's take
30
-
a look at it.
31
+
## 🏗️ Architecture
32
+
33
+
### ATProto-Native
34
+
35
+
Blonk is built as a first-class ATProto application with custom record types:
36
+
37
+
- `com.blonk.vibe` - Community topic feeds
38
+
- `com.blonk.blip` - Content submissions
39
+
- `com.blonk.groove` - Community engagement
40
+
- `com.blonk.tag` - Universal content labels
41
+
- `com.blonk.blipTag` - Content categorization associations
42
+
43
+
### Technology Stack
44
+
45
+
- **Backend**: Elixir/Phoenix LiveView
46
+
- **Database**: PostgreSQL with ATProto record sync
47
+
- **Real-time**: WebSocket firehose integration with Bluesky
48
+
- **Authentication**: ATProto app passwords
49
+
- **Deployment**: Docker-ready
31
50
32
-
BlonkAgent - to be renamed, but basically our Bsky client
51
+
## 🚀 Getting Started
33
52
34
-
POST_NSID - our namespace identifier (What is this? We'll come back to that)
53
+
### Prerequisites
35
54
36
-
PostManager - our interface to create or retrieve posts
55
+
- Elixir 1.18+
56
+
- PostgreSQL 14+
57
+
- Bluesky account with app password
37
58
38
-
`index.ts` - a super basic page skeleton
59
+
### Installation
39
60
40
-
With this I guess we can try to get some shit on a page.
61
+
```bash
62
+
# Clone the repository
63
+
git clone https://github.com/your-org/elixir_blonk.git
64
+
cd elixir_blonk
65
+
66
+
# Install dependencies
67
+
mix deps.get
68
+
69
+
# Set up environment
70
+
cp .env.example .env
71
+
# Edit .env with your Bluesky credentials
72
+
73
+
# Set up database
74
+
mix ecto.setup
75
+
76
+
# Start the server
77
+
source .env && mix phx.server
41
78
```
42
79
43
-
I guess now I'm just going to see what Claude has cooked up?
80
+
### Environment Configuration
44
81
45
-
Once we have some shit on a screen I can think a little more.
82
+
```bash
83
+
# .env
84
+
export ATP_SERVICE=https://bsky.social
85
+
export ATP_IDENTIFIER=your-handle.bsky.social
86
+
export ATP_PASSWORD=your-app-password
87
+
```
46
88
47
-
## The Story So Far
48
-
We have a very basic setup that will in fact put something on atproto.
89
+
## 🎮 Usage
49
90
50
-
We made a pull request [here](https://github.com/notactuallytreyanastasio/blonk/pull/1) that got us the basics.
91
+
### Creating Vibes
51
92
52
-
So what next?
93
+
Post `#vibe-topic_name` to create new community vibes:
53
94
54
-
Well, first I am going to drop in React.
95
+
```
96
+
Check out this cool #vibe-blockchain project! #defi #web3
97
+
```
98
+
99
+
Once enough community members use `#vibe-blockchain`, it becomes an official vibe.
100
+
101
+
### Submitting Blips
102
+
103
+
Submit content to vibes with relevant tags:
55
104
56
105
```
57
-
lets just drop in react, we will need it later anyways.
58
-
let's be adults about it. make sure to use the latest, and to do whatever dan abramov would do.
59
-
He's pretty good.
106
+
Title: "New DeFi Protocol Launch"
107
+
Body: "Exciting developments in yield farming..."
108
+
Vibe: blockchain_vibe
109
+
Tags: #defi #yield #ethereum
60
110
```
61
111
62
-
This ought to go pretty far, but we will follow up in another pull request.
112
+
### Community Engagement
113
+
114
+
- **👍 looks_good** - Positive community feedback
115
+
- **💩 shit_rips** - Critical community feedback
116
+
117
+
Grooves drive content visibility and trending algorithms.
118
+
119
+
### Discovery
120
+
121
+
- **Radar**: Trending content across all vibes
122
+
- **Vibe Pages**: Topic-specific content feeds
123
+
- **Tag Pages**: Cross-vibe content by topic
124
+
- **Hot Posts**: AI-curated trending external content
63
125
64
-
[Here] is the pull request.
126
+
## 🏷️ Tag System
65
127
66
-
I am not going to give it a ton of feedback since React isn't really my lane, unless something really jumps out.
128
+
### Universal Tags
67
129
68
-
I had to have a little back and forth, but we got to something running pretty quickly.
130
+
Tags are community-owned labels that enable cross-vibe discovery:
69
131
70
-
## What now?
71
-
Well, right now it only shows blips _from us_ -- we want to see other people's too and let them submit as well.
132
+
- **One Tag Per Name**: Only one `#blockchain` tag exists globally
133
+
- **Community Driven**: Anyone can use any tag
134
+
- **Usage Tracking**: Popular tags surface trending content
135
+
- **Rich Metadata**: Tags support descriptions and attribution
72
136
73
-
How do we go about that?
74
-
Well I am not sure yet.
137
+
### Tag Lifecycle
75
138
76
-
Let's dig into some docs and ask Claude, too.
139
+
1. **First Use**: Creates universal tag record in ATProto
140
+
2. **Association**: Links to blips via BlipTag junction records
141
+
3. **Community Growth**: Usage count drives popularity
142
+
4. **Discovery**: Popular tags surface on radar
77
143
78
-
Another prompt...
144
+
## 🤖 AI Integration
79
145
80
-
```
81
-
Blips need to have a "vibe" they belong to.
82
-
"vibes" are simply a group of people and a feeling.
83
-
Its not a topic like a subreddit or a forum.
84
-
A vibe can be "Sunset Sunglasses Struts" or "doinkin right" or "dork nerd linkage" - we want people to not feel confined to a topic, but have an idea of the type of content that will come up in that circle.
146
+
### Hot Post Detection
85
147
86
-
What do you think a good implementation step here is?
87
-
```
148
+
The HotPostSweeper monitors Bluesky's firehose for trending content:
88
149
89
-
This started off looking pretty sane, and we'll look at it more, but I had a quick piece of feedback for it.
150
+
- **Sampling**: 1 in 10 posts with links
151
+
- **Engagement Check**: Looks for >5 replies after time delay
152
+
- **Auto-Population**: Creates blips in `bsky_hot` vibe
153
+
- **Community Seeding**: Bootstraps engagement for organic growth
90
154
91
-
```
92
-
lets add some constraints.
155
+
## 🎯 Community Philosophy
93
156
94
-
we dont want duplicate vibes to be able to be created.
157
+
### Vibe-Driven Discovery
95
158
96
-
we dont want to allow people to create vibes quite yet.
159
+
Blonk prioritizes **community vibes over algorithmic feeds**:
97
160
98
-
We are going to make a system where instead if enough people skeet a vibe as a #hashtag then we will create one if a certain threshold is hit via the firehose if they match a special form (#vibe-YOUR_VIBE) and make sure vibes must be something like YOUR_VIBE or your_vibe or YOURVIBE but not YOUR VIBE and make sure thats enforced both react client/server/atproto client level
99
-
```
161
+
- Communities form organically around shared interests
162
+
- Content quality emerges through peer grooves
163
+
- Cross-vibe discovery happens through universal tags
164
+
- Trending content reflects genuine community engagement
100
165
101
-
Now, we will see where this really goes.
166
+
### Decentralized Social
102
167
103
-
I kind of really like this idea of creating them by mention velocity.
168
+
Built on ATProto for true decentralization:
104
169
105
-
So, let's see what it has come up with now.
170
+
- **Data Portability**: Your content, your control
171
+
- **Cross-Platform**: Records work across ATProto apps
172
+
- **Community Ownership**: Vibes and tags are community resources
173
+
- **Open Protocol**: Extensible and interoperable
106
174
107
-
`looks at app`
175
+
## 🛠️ Development
108
176
109
-
It got the concept of seeding vibes right.
177
+
### Project Structure
110
178
111
-
There are 6 it seeded things with.
179
+
```
180
+
lib/
181
+
├── elixir_blonk/ # Core business logic
182
+
│ ├── vibes/ # Community topic management
183
+
│ ├── blips/ # Content submission system
184
+
│ ├── grooves/ # Community engagement
185
+
│ ├── tags/ # Universal tag system
186
+
│ ├── blip_tags/ # Tag associations
187
+
│ ├── hot_posts/ # AI content curation
188
+
│ ├── atproto/ # ATProto client & sync
189
+
│ └── firehose/ # Real-time data ingestion
190
+
├── elixir_blonk_web/ # Phoenix web interface
191
+
└── priv/repo/migrations/ # Database schema
192
+
```
112
193
113
-
To create a vibe, 5 people must post with #vibe-SOMETHING-OR_WHATEVER and then it will be found and counted.
194
+
### Key Services
114
195
115
-
Once this happens, it creates the vibe so people can post in it.
196
+
- **SessionManager**: ATProto authentication & session management
197
+
- **Firehose.Consumer**: Real-time Bluesky data ingestion
198
+
- **HotPostSweeper**: AI-driven content curation
199
+
- **ATProto.Client**: Custom record type operations
116
200
117
-
Once a vibe has been filled with blips, you can fluff blips with hell_yeah's or links_good's
201
+
### Testing
118
202
119
-
However, it didn't detect my first post.
203
+
```bash
204
+
# Run tests
205
+
mix test
120
206
207
+
# Run with coverage
208
+
mix test --cover
121
209
```
122
-
I just posted #vibe-test_post and its not being detected.
210
+
211
+
## 📈 Roadmap
123
212
124
-
Are you sure you are monitoring the bluesky firehose for these hashtags and not something else?
213
+
### Phase 1: Community Bootstrap ✅
214
+
- [x] Basic vibe/blip/groove system
215
+
- [x] ATProto integration
216
+
- [x] Hot post AI curation
217
+
- [x] Universal tag system
125
218
126
-
I saw it come along the wire in my other firehose monitor.
127
-
```
219
+
### Phase 2: Enhanced Discovery
220
+
- [ ] Advanced radar algorithms
221
+
- [ ] User following/recommendations
222
+
- [ ] Cross-vibe trending
223
+
- [ ] Mobile app
128
224
129
-
It wasn't detecting my vibes and tried to take some shortcuts.
225
+
### Phase 3: Community Tools
226
+
- [ ] Vibe moderation tools
227
+
- [ ] Community governance
228
+
- [ ] Creator monetization
229
+
- [ ] External integrations
130
230
131
-
So, I had it re-think that approach.
231
+
## 🤝 Contributing
132
232
233
+
We welcome contributions! Please see [CONTRIBUTING.md](CONTRIBUTING.md) for guidelines.
133
234
134
-
```
135
-
its failing to detect emerging vibes and we have no server logs indicating this.
235
+
### Development Setup
136
236
137
-
that is troublesome.
237
+
1. Fork the repository
238
+
2. Create feature branch (`git checkout -b feature/amazing-feature`)
239
+
3. Commit changes (`git commit -m 'Add amazing feature'`)
240
+
4. Push to branch (`git push origin feature/amazing-feature`)
241
+
5. Open Pull Request
138
242
139
-
we need to hash out if this firehose is even working and you are just making willy error handlers that cover important flaws in the system.
243
+
## 📄 License
140
244
141
-
Why is it using a search to find the vibes? We should be consuming the entire firehose!
142
-
```
245
+
This project is licensed under the MIT License - see [LICENSE](LICENSE) for details.
143
246
144
-
And then
247
+
## 🙏 Acknowledgments
145
248
146
-
```
147
-
We should be defining types for all this incoming data!
249
+
- **ATProto Team** - For the decentralized social protocol
250
+
- **Bluesky** - For the firehose API and ecosystem
251
+
- **Phoenix/Elixir** - For the robust web framework
252
+
- **Community** - For making vibes happen
148
253
149
-
This is becoming incredibly hard to reason about and we still arent monitoring the firehose successfully
150
-
```
254
+
---
151
255
152
-
And some manual editing, comments left for it to eat up, let's see where it gets.
256
+
**Ready to find your vibe?** 🌊
153
257
258
+
Start exploring at [blonk.app](https://blonk.app) or run your own instance!
blonk.db
blonk.db
This is a binary file and will not be displayed.
-6
elixir_blonk/.formatter.exs
-6
elixir_blonk/.formatter.exs
-258
elixir_blonk/README.md
-258
elixir_blonk/README.md
···
1
-
# Blonk 🎯
2
-
3
-
**A place to find blips on the radar of your web, with a focus on vibes.**
4
-
5
-
Blonk is a community-driven content discovery platform built on ATProto that enables organic topic communities to form around "vibes" - interest-based feeds where users submit "blips" (content) to get "grooves" (community engagement).
6
-
7
-
## 🌟 Core Concepts
8
-
9
-
### The Blonk Ecosystem
10
-
11
-
**🎯 Radar** - The frontpage that surfaces trending content across all vibes
12
-
**🌊 Vibes** - Topic-based communities created by community action (#vibe-your_topic)
13
-
**📡 Blips** - Content submissions to vibes that appear on the radar
14
-
**🎵 Grooves** - Community engagement: `looks_good` or `shit_rips`
15
-
**🏷️ Tags** - Universal labels that connect content across vibes
16
-
17
-
### How It Works
18
-
19
-
1. **Vibe Creation**: Communities emerge when `#vibe-topic_name` reaches critical mass
20
-
2. **Content Submission**: Users submit blips to vibes with tags for categorization
21
-
3. **Community Engagement**: Others groove on blips, driving popularity
22
-
4. **Radar Discovery**: Trending tagged content surfaces on the frontpage
23
-
5. **Organic Growth**: Popular content attracts more users to related vibes
24
-
25
-
### Community Seeding
26
-
27
-
- **🔥 Hot Posts**: AI monitors the Bluesky firehose for trending content (>5 replies)
28
-
- **Auto-Population**: Trending external content auto-populates the `bsky_hot` vibe
29
-
- **Community Bootstrap**: Seeds engagement to kickstart organic community growth
30
-
31
-
## 🏗️ Architecture
32
-
33
-
### ATProto-Native
34
-
35
-
Blonk is built as a first-class ATProto application with custom record types:
36
-
37
-
- `com.blonk.vibe` - Community topic feeds
38
-
- `com.blonk.blip` - Content submissions
39
-
- `com.blonk.groove` - Community engagement
40
-
- `com.blonk.tag` - Universal content labels
41
-
- `com.blonk.blipTag` - Content categorization associations
42
-
43
-
### Technology Stack
44
-
45
-
- **Backend**: Elixir/Phoenix LiveView
46
-
- **Database**: PostgreSQL with ATProto record sync
47
-
- **Real-time**: WebSocket firehose integration with Bluesky
48
-
- **Authentication**: ATProto app passwords
49
-
- **Deployment**: Docker-ready
50
-
51
-
## 🚀 Getting Started
52
-
53
-
### Prerequisites
54
-
55
-
- Elixir 1.18+
56
-
- PostgreSQL 14+
57
-
- Bluesky account with app password
58
-
59
-
### Installation
60
-
61
-
```bash
62
-
# Clone the repository
63
-
git clone https://github.com/your-org/elixir_blonk.git
64
-
cd elixir_blonk
65
-
66
-
# Install dependencies
67
-
mix deps.get
68
-
69
-
# Set up environment
70
-
cp .env.example .env
71
-
# Edit .env with your Bluesky credentials
72
-
73
-
# Set up database
74
-
mix ecto.setup
75
-
76
-
# Start the server
77
-
source .env && mix phx.server
78
-
```
79
-
80
-
### Environment Configuration
81
-
82
-
```bash
83
-
# .env
84
-
export ATP_SERVICE=https://bsky.social
85
-
export ATP_IDENTIFIER=your-handle.bsky.social
86
-
export ATP_PASSWORD=your-app-password
87
-
```
88
-
89
-
## 🎮 Usage
90
-
91
-
### Creating Vibes
92
-
93
-
Post `#vibe-topic_name` to create new community vibes:
94
-
95
-
```
96
-
Check out this cool #vibe-blockchain project! #defi #web3
97
-
```
98
-
99
-
Once enough community members use `#vibe-blockchain`, it becomes an official vibe.
100
-
101
-
### Submitting Blips
102
-
103
-
Submit content to vibes with relevant tags:
104
-
105
-
```
106
-
Title: "New DeFi Protocol Launch"
107
-
Body: "Exciting developments in yield farming..."
108
-
Vibe: blockchain_vibe
109
-
Tags: #defi #yield #ethereum
110
-
```
111
-
112
-
### Community Engagement
113
-
114
-
- **👍 looks_good** - Positive community feedback
115
-
- **💩 shit_rips** - Critical community feedback
116
-
117
-
Grooves drive content visibility and trending algorithms.
118
-
119
-
### Discovery
120
-
121
-
- **Radar**: Trending content across all vibes
122
-
- **Vibe Pages**: Topic-specific content feeds
123
-
- **Tag Pages**: Cross-vibe content by topic
124
-
- **Hot Posts**: AI-curated trending external content
125
-
126
-
## 🏷️ Tag System
127
-
128
-
### Universal Tags
129
-
130
-
Tags are community-owned labels that enable cross-vibe discovery:
131
-
132
-
- **One Tag Per Name**: Only one `#blockchain` tag exists globally
133
-
- **Community Driven**: Anyone can use any tag
134
-
- **Usage Tracking**: Popular tags surface trending content
135
-
- **Rich Metadata**: Tags support descriptions and attribution
136
-
137
-
### Tag Lifecycle
138
-
139
-
1. **First Use**: Creates universal tag record in ATProto
140
-
2. **Association**: Links to blips via BlipTag junction records
141
-
3. **Community Growth**: Usage count drives popularity
142
-
4. **Discovery**: Popular tags surface on radar
143
-
144
-
## 🤖 AI Integration
145
-
146
-
### Hot Post Detection
147
-
148
-
The HotPostSweeper monitors Bluesky's firehose for trending content:
149
-
150
-
- **Sampling**: 1 in 10 posts with links
151
-
- **Engagement Check**: Looks for >5 replies after time delay
152
-
- **Auto-Population**: Creates blips in `bsky_hot` vibe
153
-
- **Community Seeding**: Bootstraps engagement for organic growth
154
-
155
-
## 🎯 Community Philosophy
156
-
157
-
### Vibe-Driven Discovery
158
-
159
-
Blonk prioritizes **community vibes over algorithmic feeds**:
160
-
161
-
- Communities form organically around shared interests
162
-
- Content quality emerges through peer grooves
163
-
- Cross-vibe discovery happens through universal tags
164
-
- Trending content reflects genuine community engagement
165
-
166
-
### Decentralized Social
167
-
168
-
Built on ATProto for true decentralization:
169
-
170
-
- **Data Portability**: Your content, your control
171
-
- **Cross-Platform**: Records work across ATProto apps
172
-
- **Community Ownership**: Vibes and tags are community resources
173
-
- **Open Protocol**: Extensible and interoperable
174
-
175
-
## 🛠️ Development
176
-
177
-
### Project Structure
178
-
179
-
```
180
-
lib/
181
-
├── elixir_blonk/ # Core business logic
182
-
│ ├── vibes/ # Community topic management
183
-
│ ├── blips/ # Content submission system
184
-
│ ├── grooves/ # Community engagement
185
-
│ ├── tags/ # Universal tag system
186
-
│ ├── blip_tags/ # Tag associations
187
-
│ ├── hot_posts/ # AI content curation
188
-
│ ├── atproto/ # ATProto client & sync
189
-
│ └── firehose/ # Real-time data ingestion
190
-
├── elixir_blonk_web/ # Phoenix web interface
191
-
└── priv/repo/migrations/ # Database schema
192
-
```
193
-
194
-
### Key Services
195
-
196
-
- **SessionManager**: ATProto authentication & session management
197
-
- **Firehose.Consumer**: Real-time Bluesky data ingestion
198
-
- **HotPostSweeper**: AI-driven content curation
199
-
- **ATProto.Client**: Custom record type operations
200
-
201
-
### Testing
202
-
203
-
```bash
204
-
# Run tests
205
-
mix test
206
-
207
-
# Run with coverage
208
-
mix test --cover
209
-
```
210
-
211
-
## 📈 Roadmap
212
-
213
-
### Phase 1: Community Bootstrap ✅
214
-
- [x] Basic vibe/blip/groove system
215
-
- [x] ATProto integration
216
-
- [x] Hot post AI curation
217
-
- [x] Universal tag system
218
-
219
-
### Phase 2: Enhanced Discovery
220
-
- [ ] Advanced radar algorithms
221
-
- [ ] User following/recommendations
222
-
- [ ] Cross-vibe trending
223
-
- [ ] Mobile app
224
-
225
-
### Phase 3: Community Tools
226
-
- [ ] Vibe moderation tools
227
-
- [ ] Community governance
228
-
- [ ] Creator monetization
229
-
- [ ] External integrations
230
-
231
-
## 🤝 Contributing
232
-
233
-
We welcome contributions! Please see [CONTRIBUTING.md](CONTRIBUTING.md) for guidelines.
234
-
235
-
### Development Setup
236
-
237
-
1. Fork the repository
238
-
2. Create feature branch (`git checkout -b feature/amazing-feature`)
239
-
3. Commit changes (`git commit -m 'Add amazing feature'`)
240
-
4. Push to branch (`git push origin feature/amazing-feature`)
241
-
5. Open Pull Request
242
-
243
-
## 📄 License
244
-
245
-
This project is licensed under the MIT License - see [LICENSE](LICENSE) for details.
246
-
247
-
## 🙏 Acknowledgments
248
-
249
-
- **ATProto Team** - For the decentralized social protocol
250
-
- **Bluesky** - For the firehose API and ecosystem
251
-
- **Phoenix/Elixir** - For the robust web framework
252
-
- **Community** - For making vibes happen
253
-
254
-
---
255
-
256
-
**Ready to find your vibe?** 🌊
257
-
258
-
Start exploring at [blonk.app](https://blonk.app) or run your own instance!
elixir_blonk/assets/css/app.css
assets/css/app.css
elixir_blonk/assets/css/app.css
assets/css/app.css
elixir_blonk/assets/js/app.js
assets/js/app.js
elixir_blonk/assets/js/app.js
assets/js/app.js
elixir_blonk/assets/tailwind.config.js
assets/tailwind.config.js
elixir_blonk/assets/tailwind.config.js
assets/tailwind.config.js
elixir_blonk/assets/vendor/topbar.js
assets/vendor/topbar.js
elixir_blonk/assets/vendor/topbar.js
assets/vendor/topbar.js
elixir_blonk/config/config.exs
config/config.exs
elixir_blonk/config/config.exs
config/config.exs
elixir_blonk/config/dev.exs
config/dev.exs
elixir_blonk/config/dev.exs
config/dev.exs
elixir_blonk/config/prod.exs
config/prod.exs
elixir_blonk/config/prod.exs
config/prod.exs
elixir_blonk/config/runtime.exs
config/runtime.exs
elixir_blonk/config/runtime.exs
config/runtime.exs
elixir_blonk/config/test.exs
config/test.exs
elixir_blonk/config/test.exs
config/test.exs
elixir_blonk/lib/elixir_blonk.ex
lib/elixir_blonk.ex
elixir_blonk/lib/elixir_blonk.ex
lib/elixir_blonk.ex
elixir_blonk/lib/elixir_blonk/accounts.ex
lib/elixir_blonk/accounts.ex
elixir_blonk/lib/elixir_blonk/accounts.ex
lib/elixir_blonk/accounts.ex
elixir_blonk/lib/elixir_blonk/accounts/user.ex
lib/elixir_blonk/accounts/user.ex
elixir_blonk/lib/elixir_blonk/accounts/user.ex
lib/elixir_blonk/accounts/user.ex
elixir_blonk/lib/elixir_blonk/accounts/user_notifier.ex
lib/elixir_blonk/accounts/user_notifier.ex
elixir_blonk/lib/elixir_blonk/accounts/user_notifier.ex
lib/elixir_blonk/accounts/user_notifier.ex
elixir_blonk/lib/elixir_blonk/accounts/user_token.ex
lib/elixir_blonk/accounts/user_token.ex
elixir_blonk/lib/elixir_blonk/accounts/user_token.ex
lib/elixir_blonk/accounts/user_token.ex
elixir_blonk/lib/elixir_blonk/application.ex
lib/elixir_blonk/application.ex
elixir_blonk/lib/elixir_blonk/application.ex
lib/elixir_blonk/application.ex
+60
-10
elixir_blonk/lib/elixir_blonk/atproto.ex
lib/elixir_blonk/atproto.ex
+60
-10
elixir_blonk/lib/elixir_blonk/atproto.ex
lib/elixir_blonk/atproto.ex
···
1
1
defmodule ElixirBlonk.ATProto do
2
2
@moduledoc """
3
-
Simple ATProto API client for Blonk operations.
3
+
Simple, direct ATProto API client for Blonk's decentralized social operations.
4
+
5
+
This module provides a clean interface to ATProto services, handling authentication
6
+
and making straightforward HTTP requests using Req. Designed to replace complex
7
+
session management patterns with a simple, reliable approach.
4
8
5
-
This module provides a straightforward interface to ATProto services,
6
-
handling authentication and making direct HTTP requests using Req.
9
+
## Philosophy: Keep It Simple
10
+
11
+
**Why simple?** Because ATProto is just HTTP APIs with bearer tokens:
12
+
- No complex session managers or client wrappers
13
+
- Direct Req calls with proper Authorization headers
14
+
- Fail fast on authentication issues (critical for Blonk)
15
+
- Clean error handling without overengineering
16
+
17
+
## Blonk Integration
18
+
19
+
**Powers Blonk's decentralized architecture** by:
20
+
- Creating custom record types (blips, vibes, tags, grooves)
21
+
- Enabling cross-platform content discovery via ATProto
22
+
- Providing engagement analysis for hot post curation
23
+
- Maintaining data portability and user ownership
7
24
8
-
## Authentication
25
+
## Authentication Strategy
9
26
10
-
Uses app passwords for authentication, storing the access token in
11
-
process state for subsequent requests.
27
+
Uses **app passwords** for secure, long-lived authentication:
28
+
- Authenticate once with user credentials
29
+
- Receive access token for subsequent requests
30
+
- Store token in client struct for reuse
31
+
- System fails fast if authentication fails (no silent degradation)
12
32
13
-
## Usage
33
+
## Custom Record Types
14
34
15
-
# Authenticate once
35
+
Blonk defines several custom NSIDs for community features:
36
+
- `com.blonk.blip` - Content submissions to vibes
37
+
- `com.blonk.tag` - Universal community labels
38
+
- `com.blonk.blipTag` - Content categorization associations
39
+
- `com.blonk.vibe` - Topic-based community feeds (future)
40
+
- `com.blonk.groove` - Community engagement records (future)
41
+
42
+
## Error Handling
43
+
44
+
**Fail fast and clear** approach:
45
+
- Authentication failures crash the system (as intended)
46
+
- API errors return structured `{:error, reason}` tuples
47
+
- HTTP status codes properly mapped to error types
48
+
- No silent failures that could confuse community features
49
+
50
+
## Examples
51
+
52
+
# One-time authentication
16
53
{:ok, client} = ATProto.authenticate()
17
54
18
-
# Make API calls
55
+
# Create Blonk records
56
+
{:ok, %{uri: uri, cid: cid}} = ATProto.create_blip(client, blip)
57
+
{:ok, %{uri: uri, cid: cid}} = ATProto.create_tag(client, tag)
58
+
59
+
# Analyze engagement for hot posts
60
+
{:ok, %{reply_count: count}} = ATProto.get_post_engagement(client, post_uri)
61
+
62
+
# Direct record creation
19
63
{:ok, %{uri: uri, cid: cid}} = ATProto.create_record(client, "com.blonk.blip", record)
20
-
{:ok, post} = ATProto.get_post(client, post_uri)
64
+
65
+
## Performance Characteristics
66
+
67
+
- **Lightweight**: Just HTTP calls with bearer token headers
68
+
- **Concurrent**: Multiple requests can use the same client
69
+
- **Resilient**: Network errors don't break authentication state
70
+
- **Efficient**: No unnecessary abstraction layers or state management
21
71
"""
22
72
23
73
require Logger
elixir_blonk/lib/elixir_blonk/atproto/client.ex
lib/elixir_blonk/atproto/client.ex
elixir_blonk/lib/elixir_blonk/atproto/client.ex
lib/elixir_blonk/atproto/client.ex
elixir_blonk/lib/elixir_blonk/atproto/session_manager.ex
lib/elixir_blonk/atproto/session_manager.ex
elixir_blonk/lib/elixir_blonk/atproto/session_manager.ex
lib/elixir_blonk/atproto/session_manager.ex
-47
elixir_blonk/lib/elixir_blonk/atproto/simple_session.ex
-47
elixir_blonk/lib/elixir_blonk/atproto/simple_session.ex
···
1
-
defmodule ElixirBlonk.ATProto.SimpleSession do
2
-
@moduledoc """
3
-
Simple session manager that maintains a single authenticated ATProto client.
4
-
5
-
This replaces the complex SessionManager with a straightforward approach:
6
-
- Authenticate once on startup
7
-
- Store the client in GenServer state
8
-
- Provide the client for API calls
9
-
- Fail fast if authentication fails
10
-
"""
11
-
12
-
use GenServer
13
-
require Logger
14
-
15
-
def start_link(opts) do
16
-
GenServer.start_link(__MODULE__, opts, name: __MODULE__)
17
-
end
18
-
19
-
@doc """
20
-
Get the authenticated ATProto client.
21
-
"""
22
-
def get_client do
23
-
GenServer.call(__MODULE__, :get_client)
24
-
end
25
-
26
-
# GenServer callbacks
27
-
28
-
@impl true
29
-
def init(_opts) do
30
-
Logger.info("Authenticating with ATProto...")
31
-
32
-
case ElixirBlonk.ATProto.authenticate() do
33
-
{:ok, client} ->
34
-
Logger.info("ATProto authentication successful")
35
-
{:ok, %{client: client}}
36
-
37
-
{:error, reason} ->
38
-
Logger.error("CRITICAL: ATProto authentication failed: #{inspect(reason)}")
39
-
{:stop, {:atproto_auth_failed, reason}}
40
-
end
41
-
end
42
-
43
-
@impl true
44
-
def handle_call(:get_client, _from, %{client: client} = state) do
45
-
{:reply, {:ok, client}, state}
46
-
end
47
-
end
elixir_blonk/lib/elixir_blonk/atproto/sync.ex
lib/elixir_blonk/atproto/sync.ex
elixir_blonk/lib/elixir_blonk/atproto/sync.ex
lib/elixir_blonk/atproto/sync.ex
+74
-1
elixir_blonk/lib/elixir_blonk/blips.ex
lib/elixir_blonk/blips.ex
+74
-1
elixir_blonk/lib/elixir_blonk/blips.ex
lib/elixir_blonk/blips.ex
···
1
1
defmodule ElixirBlonk.Blips do
2
2
@moduledoc """
3
-
The Blips context.
3
+
The Blips context for managing content submissions in the Blonk ecosystem.
4
+
5
+
Blips are the fundamental content unit in Blonk - posts submitted to vibes that appear
6
+
on the radar for community engagement. This context orchestrates blip creation, discovery,
7
+
and the integration with Blonk's universal tag system and ATProto records.
8
+
9
+
## What Are Blips?
10
+
11
+
**Blips are content submissions that create community engagement:**
12
+
- Posts with title, optional body/URL, and tags
13
+
- Submitted to specific vibes for topical organization
14
+
- Appear on the radar for cross-vibe discovery
15
+
- Receive grooves (looks_good/shit_rips) from community
16
+
- Enable organic vibe creation through engagement patterns
17
+
18
+
## Blonk Ecosystem Integration
19
+
20
+
- **Vibes**: Blips are submitted to topic-based communities
21
+
- **Radar**: Popular blips surface on the frontpage across vibes
22
+
- **Tags**: Universal labels enable cross-vibe content discovery
23
+
- **Grooves**: Community engagement drives blip visibility
24
+
- **ATProto**: Each blip becomes a `com.blonk.blip` record for portability
25
+
26
+
## Content Lifecycle
27
+
28
+
1. **Submission**: User creates blip with title, body, tags for a vibe
29
+
2. **Tag Extraction**: System automatically extracts #hashtags from content
30
+
3. **ATProto Creation**: Blip becomes decentralized record with URI/CID
31
+
4. **Community Discovery**: Appears in vibe feeds and radar trending
32
+
5. **Engagement**: Users groove on content, driving popularity
33
+
6. **Cross-Vibe Discovery**: Tags enable finding related content across vibes
34
+
35
+
## Community Engagement Model
36
+
37
+
**Blips are designed to encourage quality content:**
38
+
- Submission to relevant vibes ensures targeted audience
39
+
- Tag system enables broader discovery beyond vibe boundaries
40
+
- Groove system (looks_good/shit_rips) provides clear feedback
41
+
- Popular content surfaces on radar, attracting more engagement
42
+
- External links in blips can become hot posts for community seeding
43
+
44
+
## Tag Integration
45
+
46
+
**Universal tagging system** enhances content discovery:
47
+
- Automatic extraction of #hashtags from blip text
48
+
- Association with universal community-owned tag records
49
+
- Cross-vibe discovery through shared tag vocabulary
50
+
- Tag popularity tracking drives trending algorithms
51
+
52
+
## ATProto Integration
53
+
54
+
**Decentralized content ownership** through ATProto records:
55
+
- Each blip becomes a `com.blonk.blip` ATProto record
56
+
- Content portability across ATProto applications
57
+
- User ownership of content and engagement data
58
+
- Cross-platform discoverability and interaction
59
+
60
+
## Examples
61
+
62
+
# Create a blip for the crypto vibe
63
+
{:ok, blip} = Blips.create_blip(%{
64
+
title: "New DeFi Protocol Launch",
65
+
body: "Exciting developments in yield farming...",
66
+
url: "https://protocol.xyz",
67
+
vibe_uri: crypto_vibe.uri,
68
+
tags: ["defi", "yield", "ethereum"],
69
+
author_did: "did:plc:user123"
70
+
})
71
+
72
+
# Find blips by tag across all vibes
73
+
defi_blips = Blips.list_blips_by_tag("defi")
74
+
75
+
# Search blips by content
76
+
search_results = Blips.search_blips("protocol")
4
77
"""
5
78
6
79
import Ecto.Query, warn: false
elixir_blonk/lib/elixir_blonk/blips/blip.ex
lib/elixir_blonk/blips/blip.ex
elixir_blonk/lib/elixir_blonk/blips/blip.ex
lib/elixir_blonk/blips/blip.ex
elixir_blonk/lib/elixir_blonk/blips/comment.ex
lib/elixir_blonk/blips/comment.ex
elixir_blonk/lib/elixir_blonk/blips/comment.ex
lib/elixir_blonk/blips/comment.ex
+82
elixir_blonk/lib/elixir_blonk/firehose/consumer.ex
lib/elixir_blonk/firehose/consumer.ex
+82
elixir_blonk/lib/elixir_blonk/firehose/consumer.ex
lib/elixir_blonk/firehose/consumer.ex
···
1
1
defmodule ElixirBlonk.Firehose.Consumer do
2
+
@moduledoc """
3
+
Real-time Bluesky firehose consumer that powers Blonk's community discovery and seeding.
4
+
5
+
This WebSocket consumer connects to Bluesky's real-time firehose to capture posts as
6
+
they're created, enabling two critical Blonk functions: organic vibe creation through
7
+
#vibe-name mentions and AI-driven community seeding through hot post detection.
8
+
9
+
## Core Functions
10
+
11
+
**1. Vibe Discovery** - Monitors for `#vibe-name` hashtags to create new communities
12
+
**2. Hot Post Sampling** - Captures 1 in 10 posts with links for trending analysis
13
+
14
+
## Why the Firehose?
15
+
16
+
**Real-time community formation** requires immediate detection of:
17
+
- New vibe mentions that could spawn communities
18
+
- Trending external content that could seed engagement
19
+
- Community activity patterns across the ATProto network
20
+
21
+
## Blonk Integration Strategy
22
+
23
+
- **Vibe Creation**: `#vibe-crypto` mentions accumulate until reaching critical mass
24
+
- **Content Seeding**: Posts with links get sampled for potential trending analysis
25
+
- **Community Bootstrap**: External content helps solve the "empty restaurant" problem
26
+
- **Organic Growth**: Real community formation based on actual user interest
27
+
28
+
## Performance Characteristics
29
+
30
+
**Designed for high-throughput, low-latency processing:**
31
+
- WebSocket connection for real-time data stream
32
+
- Async processing to prevent firehose backpressure
33
+
- Smart sampling (1 in 10) to avoid overwhelming the system
34
+
- Non-blocking operations that don't affect user experience
35
+
36
+
## Data Flow
37
+
38
+
1. **Firehose Stream**: Receives real-time posts from Bluesky network
39
+
2. **Vibe Detection**: Scans for #vibe-name patterns in post text
40
+
3. **Link Sampling**: Identifies posts with external links for hot post analysis
41
+
4. **Async Processing**: Hands off to appropriate handlers without blocking
42
+
5. **Community Impact**: Powers both organic vibe creation and content seeding
43
+
44
+
## Error Handling
45
+
46
+
**Resilient by design** to handle network issues and data anomalies:
47
+
- Automatic reconnection after disconnects
48
+
- Graceful handling of malformed messages
49
+
- Continued operation even if individual posts fail processing
50
+
- Comprehensive logging for debugging community growth patterns
51
+
52
+
## Community Growth Engine
53
+
54
+
The firehose consumer is **Blonk's community growth engine** because it:
55
+
- Detects organic community formation through vibe mentions
56
+
- Seeds new communities with relevant trending content
57
+
- Provides real-time responsiveness to user interests
58
+
- Scales community discovery with network growth
59
+
60
+
## Sampling Strategy
61
+
62
+
**1 in 10 posts with links** balances:
63
+
- **Quality**: Enough samples to catch trending content
64
+
- **Performance**: Doesn't overwhelm the hot post analysis system
65
+
- **Freshness**: Regular sampling ensures recent trends are captured
66
+
- **Diversity**: Broad coverage across different topics and communities
67
+
68
+
## Examples
69
+
70
+
# Vibe mention detection
71
+
"I love this new #vibe-solana ecosystem!"
72
+
→ Records vibe mention for potential community creation
73
+
74
+
# Hot post sampling
75
+
"Check out this amazing new tool: https://cool-tool.com"
76
+
→ 10% chance of being saved for trending analysis
77
+
78
+
# Community seeding result
79
+
→ Hot post with 8 replies becomes blip in bsky_hot vibe
80
+
→ Users groove on it, driving engagement
81
+
→ Topic tags inspire new organic vibes
82
+
"""
83
+
2
84
use WebSockex
3
85
require Logger
4
86
elixir_blonk/lib/elixir_blonk/firehose/supervisor.ex
lib/elixir_blonk/firehose/supervisor.ex
elixir_blonk/lib/elixir_blonk/firehose/supervisor.ex
lib/elixir_blonk/firehose/supervisor.ex
-145
elixir_blonk/lib/elixir_blonk/grooves.ex
-145
elixir_blonk/lib/elixir_blonk/grooves.ex
···
1
-
defmodule ElixirBlonk.Grooves do
2
-
@moduledoc """
3
-
The Grooves context.
4
-
"""
5
-
6
-
import Ecto.Query, warn: false
7
-
alias ElixirBlonk.Repo
8
-
9
-
alias ElixirBlonk.Grooves.Groove
10
-
alias ElixirBlonk.Blips
11
-
12
-
@doc """
13
-
Creates a groove (reaction) for a blip.
14
-
"""
15
-
def create_groove(attrs \\ %{}) do
16
-
# Find the blip by subject_uri if provided
17
-
attrs = if attrs[:subject_uri] do
18
-
case Blips.get_blip_by_uri(attrs[:subject_uri]) do
19
-
nil -> attrs
20
-
blip -> Map.put(attrs, :blip_id, blip.id)
21
-
end
22
-
else
23
-
attrs
24
-
end
25
-
26
-
%Groove{}
27
-
|> Groove.changeset(attrs)
28
-
|> Repo.insert()
29
-
|> case do
30
-
{:ok, groove} ->
31
-
# Update groove counts on the blip
32
-
if groove.blip_id do
33
-
Blips.update_groove_counts(groove.blip_id)
34
-
end
35
-
{:ok, groove}
36
-
error -> error
37
-
end
38
-
end
39
-
40
-
@doc """
41
-
Deletes a groove.
42
-
"""
43
-
def delete_groove(%Groove{} = groove) do
44
-
result = Repo.delete(groove)
45
-
46
-
# Update groove counts on the blip
47
-
if groove.blip_id do
48
-
Blips.update_groove_counts(groove.blip_id)
49
-
end
50
-
51
-
result
52
-
end
53
-
54
-
@doc """
55
-
Gets a groove by author and subject.
56
-
"""
57
-
def get_groove_by_author_and_subject(author_did, subject_uri) do
58
-
Groove
59
-
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
60
-
|> Repo.one()
61
-
end
62
-
63
-
@doc """
64
-
Toggles a groove - creates if doesn't exist, changes type if different, deletes if same.
65
-
"""
66
-
def toggle_groove(author_did, subject_uri, groove_type, attrs \\ %{}) do
67
-
case get_groove_by_author_and_subject(author_did, subject_uri) do
68
-
nil ->
69
-
# Create new groove
70
-
attrs = Map.merge(attrs, %{
71
-
author_did: author_did,
72
-
subject_uri: subject_uri,
73
-
groove_type: groove_type
74
-
})
75
-
create_groove(attrs)
76
-
77
-
%Groove{groove_type: ^groove_type} = groove ->
78
-
# Same type, so remove it
79
-
delete_groove(groove)
80
-
{:ok, nil}
81
-
82
-
groove ->
83
-
# Different type, update it
84
-
groove
85
-
|> Groove.changeset(%{groove_type: groove_type})
86
-
|> Repo.update()
87
-
|> case do
88
-
{:ok, updated_groove} ->
89
-
# Update counts
90
-
if updated_groove.blip_id do
91
-
Blips.update_groove_counts(updated_groove.blip_id)
92
-
end
93
-
{:ok, updated_groove}
94
-
error -> error
95
-
end
96
-
end
97
-
end
98
-
99
-
@doc """
100
-
Lists all grooves for a blip.
101
-
"""
102
-
def list_grooves_for_blip(blip_id) do
103
-
Groove
104
-
|> where([g], g.blip_id == ^blip_id)
105
-
|> Repo.all()
106
-
end
107
-
108
-
@doc """
109
-
Lists all grooves by a specific author.
110
-
"""
111
-
def list_grooves_by_author(author_did) do
112
-
Groove
113
-
|> where([g], g.author_did == ^author_did)
114
-
|> preload(:blip)
115
-
|> Repo.all()
116
-
end
117
-
118
-
@doc """
119
-
Counts grooves by type for a blip.
120
-
"""
121
-
def count_grooves_by_type(blip_id, groove_type) do
122
-
Groove
123
-
|> where([g], g.blip_id == ^blip_id and g.groove_type == ^groove_type)
124
-
|> Repo.aggregate(:count)
125
-
end
126
-
127
-
@doc """
128
-
Checks if an author has grooved a specific blip.
129
-
"""
130
-
def has_grooved?(author_did, subject_uri) do
131
-
Groove
132
-
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
133
-
|> Repo.exists?()
134
-
end
135
-
136
-
@doc """
137
-
Gets the groove type for an author on a specific blip.
138
-
"""
139
-
def get_groove_type(author_did, subject_uri) do
140
-
Groove
141
-
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
142
-
|> select([g], g.groove_type)
143
-
|> Repo.one()
144
-
end
145
-
end
elixir_blonk/lib/elixir_blonk/grooves/groove.ex
lib/elixir_blonk/grooves/groove.ex
elixir_blonk/lib/elixir_blonk/grooves/groove.ex
lib/elixir_blonk/grooves/groove.ex
-134
elixir_blonk/lib/elixir_blonk/hot_post_sweeper.ex
-134
elixir_blonk/lib/elixir_blonk/hot_post_sweeper.ex
···
1
-
defmodule ElixirBlonk.HotPostSweeper do
2
-
@moduledoc """
3
-
GenServer that periodically sweeps hot posts to check for engagement
4
-
and creates blips for trending content.
5
-
"""
6
-
7
-
use GenServer
8
-
require Logger
9
-
10
-
alias ElixirBlonk.{HotPosts, Vibes, Blips}
11
-
12
-
# Check every 10 minutes
13
-
@sweep_interval_ms 10 * 60 * 1000
14
-
15
-
def start_link(opts \\ []) do
16
-
GenServer.start_link(__MODULE__, opts, name: __MODULE__)
17
-
end
18
-
19
-
@impl true
20
-
def init(_opts) do
21
-
Logger.info("Starting HotPostSweeper - checking posts every 10 minutes")
22
-
23
-
# Schedule the first sweep
24
-
schedule_sweep()
25
-
26
-
{:ok, %{}}
27
-
end
28
-
29
-
@impl true
30
-
def handle_info(:sweep, state) do
31
-
Logger.info("Starting hot post sweep")
32
-
33
-
try do
34
-
# Get posts to check (up to 25)
35
-
posts_to_check = HotPosts.get_posts_for_checking(25)
36
-
Logger.info("Found #{length(posts_to_check)} posts to check for replies")
37
-
38
-
# Check each post for replies
39
-
Enum.each(posts_to_check, &check_post_replies/1)
40
-
41
-
# Clean up old posts
42
-
cleanup_count = HotPosts.cleanup_old_posts()
43
-
if cleanup_count > 0 do
44
-
Logger.info("Cleaned up #{cleanup_count} old hot posts")
45
-
end
46
-
47
-
# Create blips for trending posts
48
-
create_blips_for_trending()
49
-
50
-
rescue
51
-
error ->
52
-
Logger.error("Error in hot post sweep: #{inspect(error)}")
53
-
end
54
-
55
-
# Schedule next sweep
56
-
schedule_sweep()
57
-
58
-
{:noreply, state}
59
-
end
60
-
61
-
defp schedule_sweep do
62
-
Process.send_after(self(), :sweep, @sweep_interval_ms)
63
-
end
64
-
65
-
defp check_post_replies(hot_post) do
66
-
Logger.debug("Checking replies for post: #{hot_post.post_uri}")
67
-
68
-
case get_post_reply_count(hot_post.post_uri) do
69
-
{:ok, reply_count} ->
70
-
Logger.debug("Post #{hot_post.post_uri} has #{reply_count} replies")
71
-
HotPosts.update_hot_post_check(hot_post, reply_count)
72
-
73
-
{:error, reason} ->
74
-
Logger.warning("Failed to check replies for #{hot_post.post_uri}: #{inspect(reason)}")
75
-
# Still increment check count even if API failed
76
-
HotPosts.update_hot_post_check(hot_post, hot_post.reply_count)
77
-
end
78
-
end
79
-
80
-
defp get_post_reply_count(post_uri) do
81
-
case ElixirBlonk.ATProto.SimpleSession.get_client() do
82
-
{:ok, client} ->
83
-
ElixirBlonk.ATProto.get_post_engagement(client, post_uri)
84
-
85
-
{:error, reason} ->
86
-
{:error, reason}
87
-
end
88
-
end
89
-
90
-
defp create_blips_for_trending do
91
-
trending_posts = HotPosts.get_trending_posts(5)
92
-
93
-
if length(trending_posts) > 0 do
94
-
Logger.info("Found #{length(trending_posts)} trending posts to convert to blips")
95
-
end
96
-
97
-
Enum.each(trending_posts, &create_blip_from_hot_post/1)
98
-
end
99
-
100
-
defp create_blip_from_hot_post(hot_post) do
101
-
case Vibes.get_vibe_by_name("bsky_hot") do
102
-
nil ->
103
-
Logger.error("bsky_hot vibe not found - cannot create hot blip")
104
-
105
-
bsky_hot_vibe ->
106
-
blip_params = %{
107
-
uri: "at://blonk.app/blip/#{Ecto.UUID.generate()}",
108
-
cid: "bafyrei#{:crypto.strong_rand_bytes(32) |> Base.encode32(case: :lower, padding: false)}",
109
-
author_did: hot_post.author_did,
110
-
title: truncate_text(hot_post.text || "", 100),
111
-
body: hot_post.text || "",
112
-
url: hot_post.external_url || hot_post.post_uri,
113
-
tags: ["trending", "hot", "bsky"],
114
-
vibe_id: bsky_hot_vibe.id,
115
-
vibe_uri: bsky_hot_vibe.uri,
116
-
grooves_looks_good: 0,
117
-
grooves_shit_rips: 0,
118
-
indexed_at: DateTime.utc_now()
119
-
}
120
-
121
-
case Blips.create_blip(blip_params) do
122
-
{:ok, blip} ->
123
-
Logger.info("Created hot blip: #{String.slice(blip.title, 0, 50)}... (#{hot_post.reply_count} replies)")
124
-
HotPosts.mark_as_converted(hot_post)
125
-
126
-
{:error, reason} ->
127
-
Logger.error("Failed to create hot blip: #{inspect(reason)}")
128
-
end
129
-
end
130
-
end
131
-
132
-
defp truncate_text(text, max_length) when byte_size(text) <= max_length, do: text
133
-
defp truncate_text(text, max_length), do: String.slice(text, 0, max_length) <> "..."
134
-
end
-76
elixir_blonk/lib/elixir_blonk/hot_posts.ex
-76
elixir_blonk/lib/elixir_blonk/hot_posts.ex
···
1
-
defmodule ElixirBlonk.HotPosts do
2
-
@moduledoc """
3
-
The HotPosts context for managing potential trending content.
4
-
"""
5
-
6
-
import Ecto.Query, warn: false
7
-
alias ElixirBlonk.Repo
8
-
alias ElixirBlonk.HotPosts.HotPost
9
-
10
-
@doc """
11
-
Creates a hot post record for later processing.
12
-
"""
13
-
def create_hot_post(attrs \\ %{}) do
14
-
%HotPost{}
15
-
|> HotPost.changeset(attrs)
16
-
|> Repo.insert()
17
-
end
18
-
19
-
@doc """
20
-
Gets posts ready for reply checking.
21
-
Returns up to `limit` posts that haven't been checked too many times.
22
-
"""
23
-
def get_posts_for_checking(limit \\ 25) do
24
-
HotPost
25
-
|> where([h], h.check_count < 10)
26
-
|> where([h], h.inserted_at > ago(1, "day"))
27
-
|> order_by([h], asc: h.inserted_at)
28
-
|> limit(^limit)
29
-
|> Repo.all()
30
-
end
31
-
32
-
@doc """
33
-
Updates the check count and reply count for a hot post.
34
-
"""
35
-
def update_hot_post_check(hot_post, reply_count) do
36
-
hot_post
37
-
|> HotPost.update_changeset(%{
38
-
check_count: hot_post.check_count + 1,
39
-
reply_count: reply_count,
40
-
last_checked_at: DateTime.utc_now()
41
-
})
42
-
|> Repo.update()
43
-
end
44
-
45
-
@doc """
46
-
Deletes old or fully processed hot posts.
47
-
"""
48
-
def cleanup_old_posts do
49
-
# Delete posts older than 1 day OR that have been checked 10 times
50
-
{count, _} =
51
-
HotPost
52
-
|> where([h], h.inserted_at < ago(1, "day") or h.check_count >= 10)
53
-
|> Repo.delete_all()
54
-
55
-
count
56
-
end
57
-
58
-
@doc """
59
-
Gets all hot posts with high reply counts that haven't been converted to blips yet.
60
-
"""
61
-
def get_trending_posts(min_replies \\ 5) do
62
-
HotPost
63
-
|> where([h], h.reply_count >= ^min_replies)
64
-
|> where([h], not h.converted_to_blip)
65
-
|> Repo.all()
66
-
end
67
-
68
-
@doc """
69
-
Marks a hot post as converted to a blip.
70
-
"""
71
-
def mark_as_converted(hot_post) do
72
-
hot_post
73
-
|> HotPost.update_changeset(%{converted_to_blip: true})
74
-
|> Repo.update()
75
-
end
76
-
end
-37
elixir_blonk/lib/elixir_blonk/hot_posts/hot_post.ex
-37
elixir_blonk/lib/elixir_blonk/hot_posts/hot_post.ex
···
1
-
defmodule ElixirBlonk.HotPosts.HotPost do
2
-
use Ecto.Schema
3
-
import Ecto.Changeset
4
-
5
-
@primary_key {:id, :binary_id, autogenerate: true}
6
-
@foreign_key_type :binary_id
7
-
schema "hot_posts" do
8
-
field :post_uri, :string
9
-
field :author_did, :string
10
-
field :text, :string
11
-
field :external_url, :string
12
-
field :record_data, :map # Store the full ATProto record
13
-
field :check_count, :integer, default: 0
14
-
field :reply_count, :integer, default: 0
15
-
field :converted_to_blip, :boolean, default: false
16
-
field :last_checked_at, :utc_datetime
17
-
18
-
timestamps(type: :utc_datetime)
19
-
end
20
-
21
-
@doc false
22
-
def changeset(hot_post, attrs) do
23
-
hot_post
24
-
|> cast(attrs, [
25
-
:post_uri, :author_did, :text, :external_url, :record_data,
26
-
:check_count, :reply_count, :converted_to_blip, :last_checked_at
27
-
])
28
-
|> validate_required([:post_uri, :author_did])
29
-
|> unique_constraint(:post_uri)
30
-
end
31
-
32
-
@doc false
33
-
def update_changeset(hot_post, attrs) do
34
-
hot_post
35
-
|> cast(attrs, [:check_count, :reply_count, :converted_to_blip, :last_checked_at])
36
-
end
37
-
end
elixir_blonk/lib/elixir_blonk/mailer.ex
lib/elixir_blonk/mailer.ex
elixir_blonk/lib/elixir_blonk/mailer.ex
lib/elixir_blonk/mailer.ex
elixir_blonk/lib/elixir_blonk/repo.ex
lib/elixir_blonk/repo.ex
elixir_blonk/lib/elixir_blonk/repo.ex
lib/elixir_blonk/repo.ex
-221
elixir_blonk/lib/elixir_blonk/vibes.ex
-221
elixir_blonk/lib/elixir_blonk/vibes.ex
···
1
-
defmodule ElixirBlonk.Vibes do
2
-
@moduledoc """
3
-
The Vibes context.
4
-
"""
5
-
6
-
import Ecto.Query, warn: false
7
-
require Logger
8
-
alias ElixirBlonk.Repo
9
-
10
-
alias ElixirBlonk.Vibes.{Vibe, VibeMember, VibeMention}
11
-
12
-
@doc """
13
-
Returns the list of vibes.
14
-
"""
15
-
def list_vibes do
16
-
Repo.all(Vibe)
17
-
end
18
-
19
-
@doc """
20
-
Returns the list of vibes ordered by pulse score.
21
-
"""
22
-
def list_vibes_by_pulse do
23
-
Vibe
24
-
|> order_by([v], desc: v.pulse_score)
25
-
|> Repo.all()
26
-
end
27
-
28
-
@doc """
29
-
Returns the list of emerging vibes.
30
-
"""
31
-
def list_emerging_vibes do
32
-
Vibe
33
-
|> where([v], v.is_emerging == true)
34
-
|> order_by([v], desc: v.pulse_score)
35
-
|> Repo.all()
36
-
end
37
-
38
-
@doc """
39
-
Gets a single vibe.
40
-
"""
41
-
def get_vibe!(id), do: Repo.get!(Vibe, id)
42
-
43
-
@doc """
44
-
Gets a vibe by URI.
45
-
"""
46
-
def get_vibe_by_uri(uri) do
47
-
Repo.get_by(Vibe, uri: uri)
48
-
end
49
-
50
-
@doc """
51
-
Gets a vibe by name.
52
-
"""
53
-
def get_vibe_by_name(name) do
54
-
Repo.get_by(Vibe, name: name)
55
-
end
56
-
57
-
@doc """
58
-
Creates a vibe.
59
-
"""
60
-
def create_vibe(attrs \\ %{}) do
61
-
# First create in local database
62
-
with {:ok, vibe} <- %Vibe{}
63
-
|> Vibe.changeset(attrs)
64
-
|> Repo.insert() do
65
-
66
-
# Then try to create in ATProto if enabled
67
-
if Application.get_env(:elixir_blonk, :atproto_enabled, true) do
68
-
Task.Supervisor.start_child(ElixirBlonk.TaskSupervisor, fn ->
69
-
create_vibe_in_atproto(vibe)
70
-
end)
71
-
end
72
-
73
-
{:ok, vibe}
74
-
end
75
-
end
76
-
77
-
defp create_vibe_in_atproto(vibe) do
78
-
with {:ok, client} <- ElixirBlonk.ATProto.SessionManager.get_client(),
79
-
{:ok, %{uri: uri, cid: cid}} <- ElixirBlonk.ATProto.Client.create_vibe(client, vibe) do
80
-
81
-
# Update local record with ATProto URI and CID
82
-
update_vibe(vibe, %{uri: uri, cid: cid})
83
-
Logger.info("Created vibe in ATProto: #{uri}")
84
-
else
85
-
{:error, reason} ->
86
-
Logger.error("Failed to create vibe in ATProto: #{inspect(reason)}")
87
-
end
88
-
end
89
-
90
-
@doc """
91
-
Updates a vibe.
92
-
"""
93
-
def update_vibe(%Vibe{} = vibe, attrs) do
94
-
vibe
95
-
|> Vibe.changeset(attrs)
96
-
|> Repo.update()
97
-
end
98
-
99
-
@doc """
100
-
Deletes a vibe.
101
-
"""
102
-
def delete_vibe(%Vibe{} = vibe) do
103
-
Repo.delete(vibe)
104
-
end
105
-
106
-
@doc """
107
-
Joins a user to a vibe.
108
-
"""
109
-
def join_vibe(member_did, vibe_id, attrs \\ %{}) do
110
-
attrs = Map.merge(attrs, %{
111
-
member_did: member_did,
112
-
vibe_id: vibe_id
113
-
})
114
-
115
-
%VibeMember{}
116
-
|> VibeMember.changeset(attrs)
117
-
|> Repo.insert()
118
-
|> case do
119
-
{:ok, member} ->
120
-
update_member_count(vibe_id)
121
-
{:ok, member}
122
-
error -> error
123
-
end
124
-
end
125
-
126
-
@doc """
127
-
Checks if a user is a member of a vibe.
128
-
"""
129
-
def is_member?(member_did, vibe_uri) do
130
-
VibeMember
131
-
|> where([vm], vm.member_did == ^member_did and vm.vibe_uri == ^vibe_uri)
132
-
|> Repo.exists?()
133
-
end
134
-
135
-
@doc """
136
-
Updates the member count for a vibe.
137
-
"""
138
-
def update_member_count(vibe_id) do
139
-
count = VibeMember
140
-
|> where([vm], vm.vibe_id == ^vibe_id)
141
-
|> Repo.aggregate(:count)
142
-
143
-
vibe = Repo.get!(Vibe, vibe_id)
144
-
update_vibe(vibe, %{member_count: count})
145
-
end
146
-
147
-
@doc """
148
-
Records a vibe mention.
149
-
"""
150
-
def record_vibe_mention(attrs) do
151
-
%VibeMention{}
152
-
|> VibeMention.changeset(attrs)
153
-
|> Repo.insert()
154
-
|> case do
155
-
{:ok, mention} ->
156
-
check_and_formalize_vibe(mention.vibe_name)
157
-
{:ok, mention}
158
-
error -> error
159
-
end
160
-
end
161
-
162
-
@doc """
163
-
Checks if a vibe should be formalized based on mention thresholds.
164
-
"""
165
-
def check_and_formalize_vibe(vibe_name) do
166
-
unique_authors = VibeMention
167
-
|> where([vm], vm.vibe_name == ^vibe_name)
168
-
|> distinct([vm], vm.author_did)
169
-
|> Repo.aggregate(:count)
170
-
171
-
total_mentions = VibeMention
172
-
|> where([vm], vm.vibe_name == ^vibe_name)
173
-
|> Repo.aggregate(:count)
174
-
175
-
if unique_authors >= 5 or total_mentions >= 10 do
176
-
unless get_vibe_by_name(vibe_name) do
177
-
case create_vibe(%{
178
-
uri: "at://blonk.app/vibe/#{vibe_name}",
179
-
cid: "bafyrei#{:crypto.strong_rand_bytes(32) |> Base.encode32(case: :lower, padding: false)}",
180
-
creator_did: "did:plc:blonk",
181
-
name: vibe_name,
182
-
mood: vibe_name,
183
-
is_emerging: false
184
-
}) do
185
-
{:ok, vibe} ->
186
-
# Broadcast the emergence
187
-
Phoenix.PubSub.broadcast(
188
-
ElixirBlonk.PubSub,
189
-
"vibes:emerged",
190
-
{:vibe_emerged, vibe}
191
-
)
192
-
{:ok, vibe}
193
-
error -> error
194
-
end
195
-
end
196
-
end
197
-
end
198
-
199
-
@doc """
200
-
Gets vibe mention statistics.
201
-
"""
202
-
def get_vibe_mention_stats do
203
-
VibeMention
204
-
|> group_by([vm], vm.vibe_name)
205
-
|> select([vm], %{
206
-
vibe_name: vm.vibe_name,
207
-
total_mentions: count(vm.id),
208
-
unique_authors: count(fragment("DISTINCT ?", vm.author_did))
209
-
})
210
-
|> Repo.all()
211
-
end
212
-
213
-
@doc """
214
-
Updates pulse scores for all vibes based on recent activity.
215
-
"""
216
-
def update_pulse_scores do
217
-
# This would be called periodically to update vibe pulse scores
218
-
# based on recent blip activity, member count, etc.
219
-
# Implementation depends on specific scoring algorithm
220
-
end
221
-
end
elixir_blonk/lib/elixir_blonk/vibes/vibe.ex
lib/elixir_blonk/vibes/vibe.ex
elixir_blonk/lib/elixir_blonk/vibes/vibe.ex
lib/elixir_blonk/vibes/vibe.ex
elixir_blonk/lib/elixir_blonk/vibes/vibe_member.ex
lib/elixir_blonk/vibes/vibe_member.ex
elixir_blonk/lib/elixir_blonk/vibes/vibe_member.ex
lib/elixir_blonk/vibes/vibe_member.ex
elixir_blonk/lib/elixir_blonk/vibes/vibe_mention.ex
lib/elixir_blonk/vibes/vibe_mention.ex
elixir_blonk/lib/elixir_blonk/vibes/vibe_mention.ex
lib/elixir_blonk/vibes/vibe_mention.ex
elixir_blonk/lib/elixir_blonk_web.ex
lib/elixir_blonk_web.ex
elixir_blonk/lib/elixir_blonk_web.ex
lib/elixir_blonk_web.ex
elixir_blonk/lib/elixir_blonk_web/components/core_components.ex
lib/elixir_blonk_web/components/core_components.ex
elixir_blonk/lib/elixir_blonk_web/components/core_components.ex
lib/elixir_blonk_web/components/core_components.ex
elixir_blonk/lib/elixir_blonk_web/components/layouts.ex
lib/elixir_blonk_web/components/layouts.ex
elixir_blonk/lib/elixir_blonk_web/components/layouts.ex
lib/elixir_blonk_web/components/layouts.ex
elixir_blonk/lib/elixir_blonk_web/components/layouts/app.html.heex
lib/elixir_blonk_web/components/layouts/app.html.heex
elixir_blonk/lib/elixir_blonk_web/components/layouts/app.html.heex
lib/elixir_blonk_web/components/layouts/app.html.heex
elixir_blonk/lib/elixir_blonk_web/components/layouts/root.html.heex
lib/elixir_blonk_web/components/layouts/root.html.heex
elixir_blonk/lib/elixir_blonk_web/components/layouts/root.html.heex
lib/elixir_blonk_web/components/layouts/root.html.heex
elixir_blonk/lib/elixir_blonk_web/components/word_cloud_modal.ex
lib/elixir_blonk_web/components/word_cloud_modal.ex
elixir_blonk/lib/elixir_blonk_web/components/word_cloud_modal.ex
lib/elixir_blonk_web/components/word_cloud_modal.ex
elixir_blonk/lib/elixir_blonk_web/controllers/error_html.ex
lib/elixir_blonk_web/controllers/error_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/error_html.ex
lib/elixir_blonk_web/controllers/error_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/error_json.ex
lib/elixir_blonk_web/controllers/error_json.ex
elixir_blonk/lib/elixir_blonk_web/controllers/error_json.ex
lib/elixir_blonk_web/controllers/error_json.ex
elixir_blonk/lib/elixir_blonk_web/controllers/page_controller.ex
lib/elixir_blonk_web/controllers/page_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/page_controller.ex
lib/elixir_blonk_web/controllers/page_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/page_html.ex
lib/elixir_blonk_web/controllers/page_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/page_html.ex
lib/elixir_blonk_web/controllers/page_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/page_html/home.html.heex
lib/elixir_blonk_web/controllers/page_html/home.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/page_html/home.html.heex
lib/elixir_blonk_web/controllers/page_html/home.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_controller.ex
lib/elixir_blonk_web/controllers/user_confirmation_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_controller.ex
lib/elixir_blonk_web/controllers/user_confirmation_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html.ex
lib/elixir_blonk_web/controllers/user_confirmation_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html.ex
lib/elixir_blonk_web/controllers/user_confirmation_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_confirmation_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_confirmation_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html/new.html.heex
lib/elixir_blonk_web/controllers/user_confirmation_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_confirmation_html/new.html.heex
lib/elixir_blonk_web/controllers/user_confirmation_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_controller.ex
lib/elixir_blonk_web/controllers/user_registration_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_controller.ex
lib/elixir_blonk_web/controllers/user_registration_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_html.ex
lib/elixir_blonk_web/controllers/user_registration_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_html.ex
lib/elixir_blonk_web/controllers/user_registration_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_html/new.html.heex
lib/elixir_blonk_web/controllers/user_registration_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_registration_html/new.html.heex
lib/elixir_blonk_web/controllers/user_registration_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_controller.ex
lib/elixir_blonk_web/controllers/user_reset_password_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_controller.ex
lib/elixir_blonk_web/controllers/user_reset_password_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html.ex
lib/elixir_blonk_web/controllers/user_reset_password_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html.ex
lib/elixir_blonk_web/controllers/user_reset_password_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_reset_password_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_reset_password_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html/new.html.heex
lib/elixir_blonk_web/controllers/user_reset_password_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_reset_password_html/new.html.heex
lib/elixir_blonk_web/controllers/user_reset_password_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_controller.ex
lib/elixir_blonk_web/controllers/user_session_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_controller.ex
lib/elixir_blonk_web/controllers/user_session_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_html.ex
lib/elixir_blonk_web/controllers/user_session_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_html.ex
lib/elixir_blonk_web/controllers/user_session_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_html/new.html.heex
lib/elixir_blonk_web/controllers/user_session_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_session_html/new.html.heex
lib/elixir_blonk_web/controllers/user_session_html/new.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_controller.ex
lib/elixir_blonk_web/controllers/user_settings_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_controller.ex
lib/elixir_blonk_web/controllers/user_settings_controller.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_html.ex
lib/elixir_blonk_web/controllers/user_settings_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_html.ex
lib/elixir_blonk_web/controllers/user_settings_html.ex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_settings_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/controllers/user_settings_html/edit.html.heex
lib/elixir_blonk_web/controllers/user_settings_html/edit.html.heex
elixir_blonk/lib/elixir_blonk_web/endpoint.ex
lib/elixir_blonk_web/endpoint.ex
elixir_blonk/lib/elixir_blonk_web/endpoint.ex
lib/elixir_blonk_web/endpoint.ex
elixir_blonk/lib/elixir_blonk_web/gettext.ex
lib/elixir_blonk_web/gettext.ex
elixir_blonk/lib/elixir_blonk_web/gettext.ex
lib/elixir_blonk_web/gettext.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/form_component.ex
lib/elixir_blonk_web/live/blip_live/form_component.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/form_component.ex
lib/elixir_blonk_web/live/blip_live/form_component.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/index.ex
lib/elixir_blonk_web/live/blip_live/index.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/index.ex
lib/elixir_blonk_web/live/blip_live/index.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/index.html.heex
lib/elixir_blonk_web/live/blip_live/index.html.heex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/index.html.heex
lib/elixir_blonk_web/live/blip_live/index.html.heex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/tag.ex
lib/elixir_blonk_web/live/blip_live/tag.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/tag.ex
lib/elixir_blonk_web/live/blip_live/tag.ex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/tag.html.heex
lib/elixir_blonk_web/live/blip_live/tag.html.heex
elixir_blonk/lib/elixir_blonk_web/live/blip_live/tag.html.heex
lib/elixir_blonk_web/live/blip_live/tag.html.heex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/index.ex
lib/elixir_blonk_web/live/vibe_live/index.ex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/index.ex
lib/elixir_blonk_web/live/vibe_live/index.ex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/index.html.heex
lib/elixir_blonk_web/live/vibe_live/index.html.heex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/index.html.heex
lib/elixir_blonk_web/live/vibe_live/index.html.heex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/show.ex
lib/elixir_blonk_web/live/vibe_live/show.ex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/show.ex
lib/elixir_blonk_web/live/vibe_live/show.ex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/show.html.heex
lib/elixir_blonk_web/live/vibe_live/show.html.heex
elixir_blonk/lib/elixir_blonk_web/live/vibe_live/show.html.heex
lib/elixir_blonk_web/live/vibe_live/show.html.heex
elixir_blonk/lib/elixir_blonk_web/router.ex
lib/elixir_blonk_web/router.ex
elixir_blonk/lib/elixir_blonk_web/router.ex
lib/elixir_blonk_web/router.ex
elixir_blonk/lib/elixir_blonk_web/telemetry.ex
lib/elixir_blonk_web/telemetry.ex
elixir_blonk/lib/elixir_blonk_web/telemetry.ex
lib/elixir_blonk_web/telemetry.ex
elixir_blonk/lib/elixir_blonk_web/user_auth.ex
lib/elixir_blonk_web/user_auth.ex
elixir_blonk/lib/elixir_blonk_web/user_auth.ex
lib/elixir_blonk_web/user_auth.ex
elixir_blonk/mix.exs
mix.exs
elixir_blonk/mix.exs
mix.exs
elixir_blonk/mix.lock
mix.lock
elixir_blonk/mix.lock
mix.lock
elixir_blonk/priv/gettext/en/LC_MESSAGES/errors.po
priv/gettext/en/LC_MESSAGES/errors.po
elixir_blonk/priv/gettext/en/LC_MESSAGES/errors.po
priv/gettext/en/LC_MESSAGES/errors.po
elixir_blonk/priv/gettext/errors.pot
priv/gettext/errors.pot
elixir_blonk/priv/gettext/errors.pot
priv/gettext/errors.pot
elixir_blonk/priv/repo/migrations/.formatter.exs
priv/repo/migrations/.formatter.exs
elixir_blonk/priv/repo/migrations/.formatter.exs
priv/repo/migrations/.formatter.exs
elixir_blonk/priv/repo/migrations/20250619210020_create_vibes.exs
priv/repo/migrations/20250619210020_create_vibes.exs
elixir_blonk/priv/repo/migrations/20250619210020_create_vibes.exs
priv/repo/migrations/20250619210020_create_vibes.exs
elixir_blonk/priv/repo/migrations/20250619210100_create_blips.exs
priv/repo/migrations/20250619210100_create_blips.exs
elixir_blonk/priv/repo/migrations/20250619210100_create_blips.exs
priv/repo/migrations/20250619210100_create_blips.exs
elixir_blonk/priv/repo/migrations/20250619210115_create_grooves.exs
priv/repo/migrations/20250619210115_create_grooves.exs
elixir_blonk/priv/repo/migrations/20250619210115_create_grooves.exs
priv/repo/migrations/20250619210115_create_grooves.exs
elixir_blonk/priv/repo/migrations/20250619210126_create_vibe_members.exs
priv/repo/migrations/20250619210126_create_vibe_members.exs
elixir_blonk/priv/repo/migrations/20250619210126_create_vibe_members.exs
priv/repo/migrations/20250619210126_create_vibe_members.exs
elixir_blonk/priv/repo/migrations/20250619210138_create_vibe_mentions.exs
priv/repo/migrations/20250619210138_create_vibe_mentions.exs
elixir_blonk/priv/repo/migrations/20250619210138_create_vibe_mentions.exs
priv/repo/migrations/20250619210138_create_vibe_mentions.exs
elixir_blonk/priv/repo/migrations/20250619210149_create_comments.exs
priv/repo/migrations/20250619210149_create_comments.exs
elixir_blonk/priv/repo/migrations/20250619210149_create_comments.exs
priv/repo/migrations/20250619210149_create_comments.exs
elixir_blonk/priv/repo/migrations/20250619221940_create_users_auth_tables.exs
priv/repo/migrations/20250619221940_create_users_auth_tables.exs
elixir_blonk/priv/repo/migrations/20250619221940_create_users_auth_tables.exs
priv/repo/migrations/20250619221940_create_users_auth_tables.exs
elixir_blonk/priv/repo/migrations/20250619222140_add_atproto_fields_to_users.exs
priv/repo/migrations/20250619222140_add_atproto_fields_to_users.exs
elixir_blonk/priv/repo/migrations/20250619222140_add_atproto_fields_to_users.exs
priv/repo/migrations/20250619222140_add_atproto_fields_to_users.exs
elixir_blonk/priv/repo/migrations/20250619230142_make_hashed_password_nullable.exs
priv/repo/migrations/20250619230142_make_hashed_password_nullable.exs
elixir_blonk/priv/repo/migrations/20250619230142_make_hashed_password_nullable.exs
priv/repo/migrations/20250619230142_make_hashed_password_nullable.exs
elixir_blonk/priv/repo/migrations/20250620042140_create_hot_posts.exs
priv/repo/migrations/20250620042140_create_hot_posts.exs
elixir_blonk/priv/repo/migrations/20250620042140_create_hot_posts.exs
priv/repo/migrations/20250620042140_create_hot_posts.exs
elixir_blonk/priv/repo/migrations/20250620042756_update_hot_posts_external_url_to_text.exs
priv/repo/migrations/20250620042756_update_hot_posts_external_url_to_text.exs
elixir_blonk/priv/repo/migrations/20250620042756_update_hot_posts_external_url_to_text.exs
priv/repo/migrations/20250620042756_update_hot_posts_external_url_to_text.exs
elixir_blonk/priv/repo/seed_bsky_hot_vibe.exs
priv/repo/seed_bsky_hot_vibe.exs
elixir_blonk/priv/repo/seed_bsky_hot_vibe.exs
priv/repo/seed_bsky_hot_vibe.exs
elixir_blonk/priv/repo/seed_that_guy_vibe.exs
priv/repo/seed_that_guy_vibe.exs
elixir_blonk/priv/repo/seed_that_guy_vibe.exs
priv/repo/seed_that_guy_vibe.exs
elixir_blonk/priv/repo/seeds.exs
priv/repo/seeds.exs
elixir_blonk/priv/repo/seeds.exs
priv/repo/seeds.exs
elixir_blonk/priv/repo/test_bsky_hot_blip.exs
priv/repo/test_bsky_hot_blip.exs
elixir_blonk/priv/repo/test_bsky_hot_blip.exs
priv/repo/test_bsky_hot_blip.exs
elixir_blonk/priv/repo/test_reply_count_api.exs
priv/repo/test_reply_count_api.exs
elixir_blonk/priv/repo/test_reply_count_api.exs
priv/repo/test_reply_count_api.exs
elixir_blonk/priv/static/favicon.ico
priv/static/favicon.ico
elixir_blonk/priv/static/favicon.ico
priv/static/favicon.ico
elixir_blonk/priv/static/images/logo.svg
priv/static/images/logo.svg
elixir_blonk/priv/static/images/logo.svg
priv/static/images/logo.svg
elixir_blonk/priv/static/robots.txt
priv/static/robots.txt
elixir_blonk/priv/static/robots.txt
priv/static/robots.txt
elixir_blonk/test/elixir_blonk/accounts_test.exs
test/elixir_blonk/accounts_test.exs
elixir_blonk/test/elixir_blonk/accounts_test.exs
test/elixir_blonk/accounts_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/error_html_test.exs
test/elixir_blonk_web/controllers/error_html_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/error_html_test.exs
test/elixir_blonk_web/controllers/error_html_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/error_json_test.exs
test/elixir_blonk_web/controllers/error_json_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/error_json_test.exs
test/elixir_blonk_web/controllers/error_json_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/page_controller_test.exs
test/elixir_blonk_web/controllers/page_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/page_controller_test.exs
test/elixir_blonk_web/controllers/page_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_confirmation_controller_test.exs
test/elixir_blonk_web/controllers/user_confirmation_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_confirmation_controller_test.exs
test/elixir_blonk_web/controllers/user_confirmation_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_registration_controller_test.exs
test/elixir_blonk_web/controllers/user_registration_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_registration_controller_test.exs
test/elixir_blonk_web/controllers/user_registration_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_reset_password_controller_test.exs
test/elixir_blonk_web/controllers/user_reset_password_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_reset_password_controller_test.exs
test/elixir_blonk_web/controllers/user_reset_password_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_session_controller_test.exs
test/elixir_blonk_web/controllers/user_session_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_session_controller_test.exs
test/elixir_blonk_web/controllers/user_session_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_settings_controller_test.exs
test/elixir_blonk_web/controllers/user_settings_controller_test.exs
elixir_blonk/test/elixir_blonk_web/controllers/user_settings_controller_test.exs
test/elixir_blonk_web/controllers/user_settings_controller_test.exs
elixir_blonk/test/elixir_blonk_web/user_auth_test.exs
test/elixir_blonk_web/user_auth_test.exs
elixir_blonk/test/elixir_blonk_web/user_auth_test.exs
test/elixir_blonk_web/user_auth_test.exs
elixir_blonk/test/support/conn_case.ex
test/support/conn_case.ex
elixir_blonk/test/support/conn_case.ex
test/support/conn_case.ex
elixir_blonk/test/support/data_case.ex
test/support/data_case.ex
elixir_blonk/test/support/data_case.ex
test/support/data_case.ex
elixir_blonk/test/support/fixtures/accounts_fixtures.ex
test/support/fixtures/accounts_fixtures.ex
elixir_blonk/test/support/fixtures/accounts_fixtures.ex
test/support/fixtures/accounts_fixtures.ex
elixir_blonk/test/test_helper.exs
test/test_helper.exs
elixir_blonk/test/test_helper.exs
test/test_helper.exs
elixir_blonk/test_firehose.exs
test_firehose.exs
elixir_blonk/test_firehose.exs
test_firehose.exs
+100
lib/elixir_blonk/atproto/simple_session.ex
+100
lib/elixir_blonk/atproto/simple_session.ex
···
1
+
defmodule ElixirBlonk.ATProto.SimpleSession do
2
+
@moduledoc """
3
+
Minimal session manager for Blonk's ATProto authentication needs.
4
+
5
+
This GenServer maintains a single authenticated ATProto client for the entire
6
+
application, replacing complex session management with a simple, reliable pattern
7
+
that aligns with Blonk's "fail fast" philosophy.
8
+
9
+
## Why Simple?
10
+
11
+
**Complex session management was causing authentication headaches:**
12
+
- Over-engineered refresh token logic
13
+
- Unnecessary client wrapper abstractions
14
+
- Silent failures that broke community features
15
+
- Debugging nightmares with nested state management
16
+
17
+
## New Approach: One Client, Clear Failures
18
+
19
+
- **Authenticate once** on startup with app password
20
+
- **Store client** in GenServer for all requests to use
21
+
- **Fail immediately** if authentication fails (no retries)
22
+
- **Crash the system** rather than silently degrade
23
+
24
+
## Blonk Integration
25
+
26
+
**Critical for community features** that depend on ATProto:
27
+
- HotPostSweeper needs authenticated calls to check reply counts
28
+
- Blip creation requires valid sessions to store records
29
+
- Tag system depends on reliable ATProto record creation
30
+
- Firehose processing must authenticate to analyze engagement
31
+
32
+
## Failure Philosophy
33
+
34
+
**Better to crash than confuse users:**
35
+
- Authentication failure = system shutdown (as intended)
36
+
- No degraded mode where some features silently break
37
+
- Clear error messages in logs for debugging
38
+
- Forces ops teams to fix auth issues immediately
39
+
40
+
## Usage Pattern
41
+
42
+
# System startup
43
+
{:ok, client} = SimpleSession.get_client()
44
+
45
+
# All services use the same authenticated client
46
+
ATProto.create_blip(client, blip)
47
+
ATProto.get_post_engagement(client, post_uri)
48
+
ATProto.create_tag(client, tag)
49
+
50
+
## Performance Benefits
51
+
52
+
- **No per-request auth overhead** - client reused across all calls
53
+
- **No session refresh logic** - app passwords are long-lived
54
+
- **No complex state synchronization** - single source of truth
55
+
- **Predictable memory usage** - one client struct for entire app
56
+
57
+
## Error Recovery
58
+
59
+
**Intentionally minimal** - authentication should "just work":
60
+
- No automatic retries (app passwords rarely fail)
61
+
- No graceful degradation (would confuse community features)
62
+
- System restart required for auth failures (clear resolution path)
63
+
"""
64
+
65
+
use GenServer
66
+
require Logger
67
+
68
+
def start_link(opts) do
69
+
GenServer.start_link(__MODULE__, opts, name: __MODULE__)
70
+
end
71
+
72
+
@doc """
73
+
Get the authenticated ATProto client.
74
+
"""
75
+
def get_client do
76
+
GenServer.call(__MODULE__, :get_client)
77
+
end
78
+
79
+
# GenServer callbacks
80
+
81
+
@impl true
82
+
def init(_opts) do
83
+
Logger.info("Authenticating with ATProto...")
84
+
85
+
case ElixirBlonk.ATProto.authenticate() do
86
+
{:ok, client} ->
87
+
Logger.info("ATProto authentication successful")
88
+
{:ok, %{client: client}}
89
+
90
+
{:error, reason} ->
91
+
Logger.error("CRITICAL: ATProto authentication failed: #{inspect(reason)}")
92
+
{:stop, {:atproto_auth_failed, reason}}
93
+
end
94
+
end
95
+
96
+
@impl true
97
+
def handle_call(:get_client, _from, %{client: client} = state) do
98
+
{:reply, {:ok, client}, state}
99
+
end
100
+
end
+228
lib/elixir_blonk/grooves.ex
+228
lib/elixir_blonk/grooves.ex
···
1
+
defmodule ElixirBlonk.Grooves do
2
+
@moduledoc """
3
+
The Grooves context for managing community engagement in the Blonk ecosystem.
4
+
5
+
Grooves are Blonk's community feedback mechanism - the way users express their
6
+
reaction to blips through "looks_good" (positive) or "shit_rips" (critical)
7
+
responses. This context orchestrates the engagement that drives content visibility
8
+
and community curation.
9
+
10
+
## What Are Grooves?
11
+
12
+
**Grooves are binary community feedback on blips:**
13
+
- **looks_good** (👍) - Positive community endorsement
14
+
- **shit_rips** (💩) - Critical community feedback
15
+
- Each user can groove once per blip with either reaction
16
+
- Groove counts drive content visibility and trending algorithms
17
+
- Community-driven curation without complex scoring systems
18
+
19
+
## Philosophy: Simple, Clear Feedback
20
+
21
+
**Why binary grooves instead of complex voting?**
22
+
- Clear, unambiguous community sentiment
23
+
- Prevents gaming through vote manipulation
24
+
- Encourages authentic engagement over optimization
25
+
- Simple UI that promotes quick, honest reactions
26
+
- Community consensus emerges naturally through patterns
27
+
28
+
## Blonk Ecosystem Integration
29
+
30
+
- **Blips**: Content receives grooves from community members
31
+
- **Vibes**: Groove activity indicates vibe health and engagement
32
+
- **Radar**: Popular (well-grooved) content surfaces on frontpage
33
+
- **Tags**: Grooves on tagged content influence tag popularity
34
+
- **Community**: Groove patterns reveal quality content and active members
35
+
36
+
## Community Engagement Mechanics
37
+
38
+
**Grooves drive organic content curation:**
39
+
- High "looks_good" count signals quality content worth surfacing
40
+
- "shit_rips" provides critical feedback for content improvement
41
+
- Groove ratios help identify controversial vs consensus content
42
+
- Activity patterns reveal engaged community members
43
+
- Aggregated data drives radar trending algorithms
44
+
45
+
## Content Visibility Impact
46
+
47
+
**Grooves determine what the community sees:**
48
+
1. **Vibe Ordering**: Well-grooved blips rise in vibe feeds
49
+
2. **Radar Prominence**: Popular content appears on frontpage
50
+
3. **Tag Trending**: Grooved tagged content influences tag popularity
51
+
4. **Community Health**: Active grooving indicates vibrant community
52
+
5. **Quality Signal**: Consistent groove patterns identify good content
53
+
54
+
## Anti-Gaming Design
55
+
56
+
**Simple system resists manipulation:**
57
+
- One groove per user per blip (no vote stacking)
58
+
- Binary choice prevents complex optimization strategies
59
+
- Community patterns harder to fake than individual metrics
60
+
- Real engagement required - no anonymous or bulk actions
61
+
- ATProto attribution provides accountability
62
+
63
+
## Social Dynamics
64
+
65
+
**Grooves create healthy community interaction:**
66
+
- Positive reinforcement for quality contributions
67
+
- Critical feedback mechanism for improvement
68
+
- Community consensus building through collective action
69
+
- Recognition for active, thoughtful community members
70
+
- Natural moderation through peer feedback
71
+
72
+
## Examples
73
+
74
+
# User grooves positively on a blip
75
+
{:ok, groove} = Grooves.toggle_groove(
76
+
"did:plc:user123",
77
+
"at://did:plc:author/com.blonk.blip/rkey",
78
+
"looks_good"
79
+
)
80
+
81
+
# Check community sentiment on content
82
+
%{looks_good: 42, shit_rips: 3} = Grooves.get_groove_counts(blip_id)
83
+
84
+
# Find most grooved content in vibe
85
+
trending_blips = Blips.list_blips_by_vibe(vibe_uri)
86
+
|> Enum.sort_by(&(&1.grooves_looks_good), :desc)
87
+
"""
88
+
89
+
import Ecto.Query, warn: false
90
+
alias ElixirBlonk.Repo
91
+
92
+
alias ElixirBlonk.Grooves.Groove
93
+
alias ElixirBlonk.Blips
94
+
95
+
@doc """
96
+
Creates a groove (reaction) for a blip.
97
+
"""
98
+
def create_groove(attrs \\ %{}) do
99
+
# Find the blip by subject_uri if provided
100
+
attrs = if attrs[:subject_uri] do
101
+
case Blips.get_blip_by_uri(attrs[:subject_uri]) do
102
+
nil -> attrs
103
+
blip -> Map.put(attrs, :blip_id, blip.id)
104
+
end
105
+
else
106
+
attrs
107
+
end
108
+
109
+
%Groove{}
110
+
|> Groove.changeset(attrs)
111
+
|> Repo.insert()
112
+
|> case do
113
+
{:ok, groove} ->
114
+
# Update groove counts on the blip
115
+
if groove.blip_id do
116
+
Blips.update_groove_counts(groove.blip_id)
117
+
end
118
+
{:ok, groove}
119
+
error -> error
120
+
end
121
+
end
122
+
123
+
@doc """
124
+
Deletes a groove.
125
+
"""
126
+
def delete_groove(%Groove{} = groove) do
127
+
result = Repo.delete(groove)
128
+
129
+
# Update groove counts on the blip
130
+
if groove.blip_id do
131
+
Blips.update_groove_counts(groove.blip_id)
132
+
end
133
+
134
+
result
135
+
end
136
+
137
+
@doc """
138
+
Gets a groove by author and subject.
139
+
"""
140
+
def get_groove_by_author_and_subject(author_did, subject_uri) do
141
+
Groove
142
+
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
143
+
|> Repo.one()
144
+
end
145
+
146
+
@doc """
147
+
Toggles a groove - creates if doesn't exist, changes type if different, deletes if same.
148
+
"""
149
+
def toggle_groove(author_did, subject_uri, groove_type, attrs \\ %{}) do
150
+
case get_groove_by_author_and_subject(author_did, subject_uri) do
151
+
nil ->
152
+
# Create new groove
153
+
attrs = Map.merge(attrs, %{
154
+
author_did: author_did,
155
+
subject_uri: subject_uri,
156
+
groove_type: groove_type
157
+
})
158
+
create_groove(attrs)
159
+
160
+
%Groove{groove_type: ^groove_type} = groove ->
161
+
# Same type, so remove it
162
+
delete_groove(groove)
163
+
{:ok, nil}
164
+
165
+
groove ->
166
+
# Different type, update it
167
+
groove
168
+
|> Groove.changeset(%{groove_type: groove_type})
169
+
|> Repo.update()
170
+
|> case do
171
+
{:ok, updated_groove} ->
172
+
# Update counts
173
+
if updated_groove.blip_id do
174
+
Blips.update_groove_counts(updated_groove.blip_id)
175
+
end
176
+
{:ok, updated_groove}
177
+
error -> error
178
+
end
179
+
end
180
+
end
181
+
182
+
@doc """
183
+
Lists all grooves for a blip.
184
+
"""
185
+
def list_grooves_for_blip(blip_id) do
186
+
Groove
187
+
|> where([g], g.blip_id == ^blip_id)
188
+
|> Repo.all()
189
+
end
190
+
191
+
@doc """
192
+
Lists all grooves by a specific author.
193
+
"""
194
+
def list_grooves_by_author(author_did) do
195
+
Groove
196
+
|> where([g], g.author_did == ^author_did)
197
+
|> preload(:blip)
198
+
|> Repo.all()
199
+
end
200
+
201
+
@doc """
202
+
Counts grooves by type for a blip.
203
+
"""
204
+
def count_grooves_by_type(blip_id, groove_type) do
205
+
Groove
206
+
|> where([g], g.blip_id == ^blip_id and g.groove_type == ^groove_type)
207
+
|> Repo.aggregate(:count)
208
+
end
209
+
210
+
@doc """
211
+
Checks if an author has grooved a specific blip.
212
+
"""
213
+
def has_grooved?(author_did, subject_uri) do
214
+
Groove
215
+
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
216
+
|> Repo.exists?()
217
+
end
218
+
219
+
@doc """
220
+
Gets the groove type for an author on a specific blip.
221
+
"""
222
+
def get_groove_type(author_did, subject_uri) do
223
+
Groove
224
+
|> where([g], g.author_did == ^author_did and g.subject_uri == ^subject_uri)
225
+
|> select([g], g.groove_type)
226
+
|> Repo.one()
227
+
end
228
+
end
+197
lib/elixir_blonk/hot_post_sweeper.ex
+197
lib/elixir_blonk/hot_post_sweeper.ex
···
1
+
defmodule ElixirBlonk.HotPostSweeper do
2
+
@moduledoc """
3
+
AI-powered content curation service that transforms trending Bluesky posts into Blonk blips.
4
+
5
+
The HotPostSweeper is the heart of Blonk's community bootstrap strategy, running every
6
+
10 minutes to analyze engagement on posts captured from the firehose and converting
7
+
highly-engaged content into blips that seed community activity.
8
+
9
+
## Core Mission
10
+
11
+
**Solve the cold start problem** by ensuring there's always engaging content on the radar:
12
+
- Continuously analyze posts captured from Bluesky firehose
13
+
- Check reply counts to gauge community interest
14
+
- Convert trending content into blips for the bsky_hot vibe
15
+
- Maintain system performance through intelligent cleanup
16
+
17
+
## Why Every 10 Minutes?
18
+
19
+
- **Fresh Content**: Recent posts need time to accumulate replies
20
+
- **System Performance**: Avoid overwhelming ATProto APIs with constant requests
21
+
- **Quality Control**: Allows natural filtering - truly engaging content rises
22
+
- **Community Timing**: Balances freshness with engagement validation
23
+
24
+
## Integration with Blonk Ecosystem
25
+
26
+
- **Firehose Consumer**: Receives posts to analyze from real-time capture
27
+
- **ATProto API**: Checks reply counts via authenticated Bluesky calls
28
+
- **bsky_hot Vibe**: Creates blips for trending content in this community space
29
+
- **Radar**: Newly created blips surface on the frontpage for community grooves
30
+
- **HotPosts Context**: Manages the lifecycle of potential trending content
31
+
32
+
## AI Curation Logic
33
+
34
+
1. **Batch Processing**: Analyzes up to 25 posts per sweep for efficiency
35
+
2. **Engagement Threshold**: Posts with ≥5 replies qualify as "hot"
36
+
3. **Retry Logic**: Failed API calls don't block other posts from processing
37
+
4. **Smart Cleanup**: Removes posts >1 day old or checked >10 times
38
+
5. **Conversion Tracking**: Prevents duplicate blips from same hot post
39
+
40
+
## Community Impact
41
+
42
+
The sweeper creates a **virtuous cycle of engagement**:
43
+
- Quality external content attracts users to Blonk
44
+
- Users groove on hot blips, increasing visibility
45
+
- Popular topics inspire organic vibe creation
46
+
- Growing community activity attracts more users
47
+
48
+
## Performance Characteristics
49
+
50
+
- **Non-blocking**: Runs in background without affecting user experience
51
+
- **Error Recovery**: Individual post failures don't crash the system
52
+
- **Rate Limited**: Respects ATProto API limits through batching
53
+
- **Self-cleaning**: Automatically maintains database efficiency
54
+
55
+
## Examples
56
+
57
+
# Sweeper finds a trending crypto post
58
+
hot_post = %HotPost{
59
+
text: "New DeFi protocol just launched...",
60
+
external_url: "https://protocol.xyz",
61
+
reply_count: 12 # Above threshold!
62
+
}
63
+
64
+
# Creates blip in bsky_hot vibe
65
+
# → Surfaces on radar
66
+
# → Users groove on it
67
+
# → Drives more engagement
68
+
"""
69
+
70
+
use GenServer
71
+
require Logger
72
+
73
+
alias ElixirBlonk.{HotPosts, Vibes, Blips}
74
+
75
+
# Check every 10 minutes
76
+
@sweep_interval_ms 10 * 60 * 1000
77
+
78
+
def start_link(opts \\ []) do
79
+
GenServer.start_link(__MODULE__, opts, name: __MODULE__)
80
+
end
81
+
82
+
@impl true
83
+
def init(_opts) do
84
+
Logger.info("Starting HotPostSweeper - checking posts every 10 minutes")
85
+
86
+
# Schedule the first sweep
87
+
schedule_sweep()
88
+
89
+
{:ok, %{}}
90
+
end
91
+
92
+
@impl true
93
+
def handle_info(:sweep, state) do
94
+
Logger.info("Starting hot post sweep")
95
+
96
+
try do
97
+
# Get posts to check (up to 25)
98
+
posts_to_check = HotPosts.get_posts_for_checking(25)
99
+
Logger.info("Found #{length(posts_to_check)} posts to check for replies")
100
+
101
+
# Check each post for replies
102
+
Enum.each(posts_to_check, &check_post_replies/1)
103
+
104
+
# Clean up old posts
105
+
cleanup_count = HotPosts.cleanup_old_posts()
106
+
if cleanup_count > 0 do
107
+
Logger.info("Cleaned up #{cleanup_count} old hot posts")
108
+
end
109
+
110
+
# Create blips for trending posts
111
+
create_blips_for_trending()
112
+
113
+
rescue
114
+
error ->
115
+
Logger.error("Error in hot post sweep: #{inspect(error)}")
116
+
end
117
+
118
+
# Schedule next sweep
119
+
schedule_sweep()
120
+
121
+
{:noreply, state}
122
+
end
123
+
124
+
defp schedule_sweep do
125
+
Process.send_after(self(), :sweep, @sweep_interval_ms)
126
+
end
127
+
128
+
defp check_post_replies(hot_post) do
129
+
Logger.debug("Checking replies for post: #{hot_post.post_uri}")
130
+
131
+
case get_post_reply_count(hot_post.post_uri) do
132
+
{:ok, reply_count} ->
133
+
Logger.debug("Post #{hot_post.post_uri} has #{reply_count} replies")
134
+
HotPosts.update_hot_post_check(hot_post, reply_count)
135
+
136
+
{:error, reason} ->
137
+
Logger.warning("Failed to check replies for #{hot_post.post_uri}: #{inspect(reason)}")
138
+
# Still increment check count even if API failed
139
+
HotPosts.update_hot_post_check(hot_post, hot_post.reply_count)
140
+
end
141
+
end
142
+
143
+
defp get_post_reply_count(post_uri) do
144
+
case ElixirBlonk.ATProto.SimpleSession.get_client() do
145
+
{:ok, client} ->
146
+
ElixirBlonk.ATProto.get_post_engagement(client, post_uri)
147
+
148
+
{:error, reason} ->
149
+
{:error, reason}
150
+
end
151
+
end
152
+
153
+
defp create_blips_for_trending do
154
+
trending_posts = HotPosts.get_trending_posts(5)
155
+
156
+
if length(trending_posts) > 0 do
157
+
Logger.info("Found #{length(trending_posts)} trending posts to convert to blips")
158
+
end
159
+
160
+
Enum.each(trending_posts, &create_blip_from_hot_post/1)
161
+
end
162
+
163
+
defp create_blip_from_hot_post(hot_post) do
164
+
case Vibes.get_vibe_by_name("bsky_hot") do
165
+
nil ->
166
+
Logger.error("bsky_hot vibe not found - cannot create hot blip")
167
+
168
+
bsky_hot_vibe ->
169
+
blip_params = %{
170
+
uri: "at://blonk.app/blip/#{Ecto.UUID.generate()}",
171
+
cid: "bafyrei#{:crypto.strong_rand_bytes(32) |> Base.encode32(case: :lower, padding: false)}",
172
+
author_did: hot_post.author_did,
173
+
title: truncate_text(hot_post.text || "", 100),
174
+
body: hot_post.text || "",
175
+
url: hot_post.external_url || hot_post.post_uri,
176
+
tags: ["trending", "hot", "bsky"],
177
+
vibe_id: bsky_hot_vibe.id,
178
+
vibe_uri: bsky_hot_vibe.uri,
179
+
grooves_looks_good: 0,
180
+
grooves_shit_rips: 0,
181
+
indexed_at: DateTime.utc_now()
182
+
}
183
+
184
+
case Blips.create_blip(blip_params) do
185
+
{:ok, blip} ->
186
+
Logger.info("Created hot blip: #{String.slice(blip.title, 0, 50)}... (#{hot_post.reply_count} replies)")
187
+
HotPosts.mark_as_converted(hot_post)
188
+
189
+
{:error, reason} ->
190
+
Logger.error("Failed to create hot blip: #{inspect(reason)}")
191
+
end
192
+
end
193
+
end
194
+
195
+
defp truncate_text(text, max_length) when byte_size(text) <= max_length, do: text
196
+
defp truncate_text(text, max_length), do: String.slice(text, 0, max_length) <> "..."
197
+
end
+130
lib/elixir_blonk/hot_posts.ex
+130
lib/elixir_blonk/hot_posts.ex
···
1
+
defmodule ElixirBlonk.HotPosts do
2
+
@moduledoc """
3
+
The HotPosts context for managing AI-driven content curation in the Blonk ecosystem.
4
+
5
+
This context orchestrates the discovery and analysis of trending content from the
6
+
Bluesky firehose, providing the community bootstrap mechanism that seeds engagement
7
+
and attracts users to the platform.
8
+
9
+
## Core Purpose in Blonk
10
+
11
+
**HotPosts solve the cold start problem** for community platforms by:
12
+
- Automatically discovering trending external content
13
+
- Analyzing community engagement through reply counts
14
+
- Converting popular content into blips for community grooves
15
+
- Seeding the radar with quality content to drive initial engagement
16
+
17
+
## Integration with Blonk Ecosystem
18
+
19
+
- **Firehose**: Captures posts with links from Bluesky's real-time feed
20
+
- **bsky_hot Vibe**: Auto-populated community space for trending content
21
+
- **Radar**: Hot blips surface on the frontpage across all vibes
22
+
- **Grooves**: Community engagement on converted content drives visibility
23
+
- **AI Curation**: Algorithmic filtering ensures quality over quantity
24
+
25
+
## Community Bootstrap Strategy
26
+
27
+
1. **Content Discovery**: Monitor Bluesky firehose for posts with external links
28
+
2. **Smart Sampling**: Take 1 in 10 posts to avoid overwhelming the system
29
+
3. **Time-Delayed Analysis**: Wait for posts to accumulate replies naturally
30
+
4. **Engagement Threshold**: Convert posts with ≥5 replies to community blips
31
+
5. **Automatic Cleanup**: Remove old/processed posts to maintain performance
32
+
33
+
## Why This Matters
34
+
35
+
Without hot posts, Blonk would face the **empty restaurant problem**:
36
+
- No content → no users → no engagement → no growth
37
+
- Hot posts create the initial activity that attracts real community
38
+
- Quality external content gives users something to groove on immediately
39
+
- Trending topics seed organic vibe creation and community formation
40
+
41
+
## Lifecycle Management
42
+
43
+
**Capture Phase**: Firehose consumer saves promising posts
44
+
**Analysis Phase**: HotPostSweeper checks engagement metrics
45
+
**Conversion Phase**: Popular posts become blips in bsky_hot vibe
46
+
**Cleanup Phase**: Old/processed posts are automatically removed
47
+
48
+
## Examples
49
+
50
+
# Find posts ready for engagement analysis
51
+
posts_to_check = HotPosts.get_posts_for_checking(25)
52
+
53
+
# Convert highly-engaged posts to community blips
54
+
trending_posts = HotPosts.get_trending_posts(5)
55
+
56
+
# Cleanup old posts to maintain performance
57
+
cleanup_count = HotPosts.cleanup_old_posts()
58
+
"""
59
+
60
+
import Ecto.Query, warn: false
61
+
alias ElixirBlonk.Repo
62
+
alias ElixirBlonk.HotPosts.HotPost
63
+
64
+
@doc """
65
+
Creates a hot post record for later processing.
66
+
"""
67
+
def create_hot_post(attrs \\ %{}) do
68
+
%HotPost{}
69
+
|> HotPost.changeset(attrs)
70
+
|> Repo.insert()
71
+
end
72
+
73
+
@doc """
74
+
Gets posts ready for reply checking.
75
+
Returns up to `limit` posts that haven't been checked too many times.
76
+
"""
77
+
def get_posts_for_checking(limit \\ 25) do
78
+
HotPost
79
+
|> where([h], h.check_count < 10)
80
+
|> where([h], h.inserted_at > ago(1, "day"))
81
+
|> order_by([h], asc: h.inserted_at)
82
+
|> limit(^limit)
83
+
|> Repo.all()
84
+
end
85
+
86
+
@doc """
87
+
Updates the check count and reply count for a hot post.
88
+
"""
89
+
def update_hot_post_check(hot_post, reply_count) do
90
+
hot_post
91
+
|> HotPost.update_changeset(%{
92
+
check_count: hot_post.check_count + 1,
93
+
reply_count: reply_count,
94
+
last_checked_at: DateTime.utc_now()
95
+
})
96
+
|> Repo.update()
97
+
end
98
+
99
+
@doc """
100
+
Deletes old or fully processed hot posts.
101
+
"""
102
+
def cleanup_old_posts do
103
+
# Delete posts older than 1 day OR that have been checked 10 times
104
+
{count, _} =
105
+
HotPost
106
+
|> where([h], h.inserted_at < ago(1, "day") or h.check_count >= 10)
107
+
|> Repo.delete_all()
108
+
109
+
count
110
+
end
111
+
112
+
@doc """
113
+
Gets all hot posts with high reply counts that haven't been converted to blips yet.
114
+
"""
115
+
def get_trending_posts(min_replies \\ 5) do
116
+
HotPost
117
+
|> where([h], h.reply_count >= ^min_replies)
118
+
|> where([h], not h.converted_to_blip)
119
+
|> Repo.all()
120
+
end
121
+
122
+
@doc """
123
+
Marks a hot post as converted to a blip.
124
+
"""
125
+
def mark_as_converted(hot_post) do
126
+
hot_post
127
+
|> HotPost.update_changeset(%{converted_to_blip: true})
128
+
|> Repo.update()
129
+
end
130
+
end
+93
lib/elixir_blonk/hot_posts/hot_post.ex
+93
lib/elixir_blonk/hot_posts/hot_post.ex
···
1
+
defmodule ElixirBlonk.HotPosts.HotPost do
2
+
@moduledoc """
3
+
Represents a potential trending post from the Bluesky firehose awaiting engagement analysis.
4
+
5
+
HotPost records are created by the firehose consumer when it samples posts with external
6
+
links. These posts are then analyzed by the HotPostSweeper to determine if they have
7
+
enough community engagement (replies) to become blips in the bsky_hot vibe.
8
+
9
+
## Blonk Integration
10
+
11
+
**Hot Posts seed community engagement** by:
12
+
- Monitoring Bluesky firehose for content with links
13
+
- Sampling 1 in 10 posts to avoid overwhelming the system
14
+
- Time-delayed engagement checking (posts need time to get replies)
15
+
- Auto-converting trending content into blips for community grooves
16
+
17
+
## Lifecycle in Community Bootstrap
18
+
19
+
1. **Capture**: Firehose consumer saves posts with links from Bluesky
20
+
2. **Patience**: Posts wait for time delay to allow replies to accumulate
21
+
3. **Analysis**: HotPostSweeper checks reply counts via ATProto API
22
+
4. **Conversion**: Posts with ≥5 replies become blips in bsky_hot vibe
23
+
5. **Cleanup**: Old or fully-processed posts are removed automatically
24
+
25
+
## Why This Matters for Blonk
26
+
27
+
The hot posts system **bootstraps community engagement** by:
28
+
- Seeding the radar with trending external content
29
+
- Providing initial blips for users to groove on
30
+
- Creating conversation starters across vibes
31
+
- Attracting users through quality content discovery
32
+
33
+
## Schema Fields
34
+
35
+
- `post_uri` - ATProto URI of the original Bluesky post
36
+
- `author_did` - DID of the original post author
37
+
- `text` - Post content/body text
38
+
- `external_url` - The external link that made this post interesting
39
+
- `record_data` - Full ATProto record for reference
40
+
- `check_count` - How many times we've checked this post for replies
41
+
- `reply_count` - Current number of replies from last check
42
+
- `converted_to_blip` - Whether this became a blip in bsky_hot vibe
43
+
- `last_checked_at` - When we last analyzed engagement
44
+
45
+
## Examples
46
+
47
+
# A hot post awaiting analysis
48
+
%HotPost{
49
+
post_uri: "at://did:plc:user/app.bsky.feed.post/rkey",
50
+
text: "Check out this amazing new protocol...",
51
+
external_url: "https://protocol.xyz/announcement",
52
+
check_count: 2,
53
+
reply_count: 7, # Above threshold!
54
+
converted_to_blip: false # Ready for conversion
55
+
}
56
+
"""
57
+
58
+
use Ecto.Schema
59
+
import Ecto.Changeset
60
+
61
+
@primary_key {:id, :binary_id, autogenerate: true}
62
+
@foreign_key_type :binary_id
63
+
schema "hot_posts" do
64
+
field :post_uri, :string
65
+
field :author_did, :string
66
+
field :text, :string
67
+
field :external_url, :string
68
+
field :record_data, :map # Store the full ATProto record
69
+
field :check_count, :integer, default: 0
70
+
field :reply_count, :integer, default: 0
71
+
field :converted_to_blip, :boolean, default: false
72
+
field :last_checked_at, :utc_datetime
73
+
74
+
timestamps(type: :utc_datetime)
75
+
end
76
+
77
+
@doc false
78
+
def changeset(hot_post, attrs) do
79
+
hot_post
80
+
|> cast(attrs, [
81
+
:post_uri, :author_did, :text, :external_url, :record_data,
82
+
:check_count, :reply_count, :converted_to_blip, :last_checked_at
83
+
])
84
+
|> validate_required([:post_uri, :author_did])
85
+
|> unique_constraint(:post_uri)
86
+
end
87
+
88
+
@doc false
89
+
def update_changeset(hot_post, attrs) do
90
+
hot_post
91
+
|> cast(attrs, [:check_count, :reply_count, :converted_to_blip, :last_checked_at])
92
+
end
93
+
end
+307
lib/elixir_blonk/vibes.ex
+307
lib/elixir_blonk/vibes.ex
···
1
+
defmodule ElixirBlonk.Vibes do
2
+
@moduledoc """
3
+
The Vibes context for managing topic-based communities in the Blonk ecosystem.
4
+
5
+
Vibes are the heart of Blonk's community organization - interest-based feeds where
6
+
users submit blips and engage through grooves. This context manages the organic
7
+
creation, discovery, and growth of community spaces through grassroots engagement.
8
+
9
+
## What Are Vibes?
10
+
11
+
**Vibes are community-driven topic feeds:**
12
+
- Interest-based communities (e.g., crypto_vibe, art_vibe, tech_vibe)
13
+
- Created organically through #vibe-name mentions reaching critical mass
14
+
- Contain blips relevant to the community's focus
15
+
- Enable targeted audience engagement and content discovery
16
+
- Form the foundation for radar trending and cross-vibe connections
17
+
18
+
## Organic Community Creation
19
+
20
+
**Vibes emerge naturally from community interest:**
21
+
1. **Mention Phase**: Users post content with #vibe-name hashtags
22
+
2. **Accumulation**: System tracks mentions across the firehose
23
+
3. **Critical Mass**: Once threshold is reached, vibe officially emerges
24
+
4. **Community Growth**: Members join, submit blips, and engage through grooves
25
+
5. **Radar Integration**: Popular vibe content surfaces on the frontpage
26
+
27
+
## Blonk Ecosystem Integration
28
+
29
+
- **Blips**: Content submissions that give vibes their substance
30
+
- **Grooves**: Community engagement that drives vibe activity
31
+
- **Tags**: Universal labels that connect content across vibes
32
+
- **Radar**: Popular vibe content surfaces on the frontpage
33
+
- **Hot Posts**: AI-curated content seeds engagement in new vibes
34
+
35
+
## Community Philosophy
36
+
37
+
**Vibes prioritize authentic community formation:**
38
+
- No top-down vibe creation - communities must emerge organically
39
+
- Interest-based rather than algorithm-driven organization
40
+
- Quality content rises through peer grooves, not engagement manipulation
41
+
- Cross-vibe discovery through universal tags promotes healthy growth
42
+
- Real community engagement over vanity metrics
43
+
44
+
## Vibe Lifecycle
45
+
46
+
1. **Grassroots Mentions**: Users naturally reference #vibe-topics in posts
47
+
2. **Threshold Detection**: Firehose consumer tracks mention accumulation
48
+
3. **Emergence**: Vibe officially created when community interest is proven
49
+
4. **Content Submission**: Users submit relevant blips to the new vibe
50
+
5. **Community Engagement**: Members groove on content, driving activity
51
+
6. **Radar Visibility**: Popular content attracts new members
52
+
53
+
## Membership and Engagement
54
+
55
+
**Flexible community participation:**
56
+
- Users can join multiple vibes based on interests
57
+
- Member counts visible for community size indication
58
+
- Activity levels drive vibe prominence on radar
59
+
- Tag frequency analysis reveals community interests
60
+
- Cross-vibe connections through shared tags and members
61
+
62
+
## Discovery Mechanisms
63
+
64
+
**Multiple pathways for vibe discovery:**
65
+
- Emerging vibes list shows communities gaining momentum
66
+
- Tag-based discovery reveals related vibes
67
+
- Radar trending surfaces popular vibe content
68
+
- Member activity patterns suggest relevant communities
69
+
70
+
## Examples
71
+
72
+
# Track a potential new vibe
73
+
Vibes.record_vibe_mention(%{
74
+
vibe_name: "defi",
75
+
author_did: "did:plc:user123",
76
+
post_uri: "at://did:plc:user123/app.bsky.feed.post/rkey",
77
+
mentioned_at: DateTime.utc_now()
78
+
})
79
+
80
+
# Check if vibe has reached emergence threshold
81
+
case Vibes.check_vibe_emergence("defi") do
82
+
{:emerging, vibe} ->
83
+
# New community has formed!
84
+
{:not_ready, count} ->
85
+
# Still accumulating mentions: #{count}
86
+
end
87
+
88
+
# Get vibe content for radar
89
+
popular_blips = Blips.list_blips_by_vibe(crypto_vibe.uri)
90
+
"""
91
+
92
+
import Ecto.Query, warn: false
93
+
require Logger
94
+
alias ElixirBlonk.Repo
95
+
96
+
alias ElixirBlonk.Vibes.{Vibe, VibeMember, VibeMention}
97
+
98
+
@doc """
99
+
Returns the list of vibes.
100
+
"""
101
+
def list_vibes do
102
+
Repo.all(Vibe)
103
+
end
104
+
105
+
@doc """
106
+
Returns the list of vibes ordered by pulse score.
107
+
"""
108
+
def list_vibes_by_pulse do
109
+
Vibe
110
+
|> order_by([v], desc: v.pulse_score)
111
+
|> Repo.all()
112
+
end
113
+
114
+
@doc """
115
+
Returns the list of emerging vibes.
116
+
"""
117
+
def list_emerging_vibes do
118
+
Vibe
119
+
|> where([v], v.is_emerging == true)
120
+
|> order_by([v], desc: v.pulse_score)
121
+
|> Repo.all()
122
+
end
123
+
124
+
@doc """
125
+
Gets a single vibe.
126
+
"""
127
+
def get_vibe!(id), do: Repo.get!(Vibe, id)
128
+
129
+
@doc """
130
+
Gets a vibe by URI.
131
+
"""
132
+
def get_vibe_by_uri(uri) do
133
+
Repo.get_by(Vibe, uri: uri)
134
+
end
135
+
136
+
@doc """
137
+
Gets a vibe by name.
138
+
"""
139
+
def get_vibe_by_name(name) do
140
+
Repo.get_by(Vibe, name: name)
141
+
end
142
+
143
+
@doc """
144
+
Creates a vibe.
145
+
"""
146
+
def create_vibe(attrs \\ %{}) do
147
+
# First create in local database
148
+
with {:ok, vibe} <- %Vibe{}
149
+
|> Vibe.changeset(attrs)
150
+
|> Repo.insert() do
151
+
152
+
# Then try to create in ATProto if enabled
153
+
if Application.get_env(:elixir_blonk, :atproto_enabled, true) do
154
+
Task.Supervisor.start_child(ElixirBlonk.TaskSupervisor, fn ->
155
+
create_vibe_in_atproto(vibe)
156
+
end)
157
+
end
158
+
159
+
{:ok, vibe}
160
+
end
161
+
end
162
+
163
+
defp create_vibe_in_atproto(vibe) do
164
+
with {:ok, client} <- ElixirBlonk.ATProto.SessionManager.get_client(),
165
+
{:ok, %{uri: uri, cid: cid}} <- ElixirBlonk.ATProto.Client.create_vibe(client, vibe) do
166
+
167
+
# Update local record with ATProto URI and CID
168
+
update_vibe(vibe, %{uri: uri, cid: cid})
169
+
Logger.info("Created vibe in ATProto: #{uri}")
170
+
else
171
+
{:error, reason} ->
172
+
Logger.error("Failed to create vibe in ATProto: #{inspect(reason)}")
173
+
end
174
+
end
175
+
176
+
@doc """
177
+
Updates a vibe.
178
+
"""
179
+
def update_vibe(%Vibe{} = vibe, attrs) do
180
+
vibe
181
+
|> Vibe.changeset(attrs)
182
+
|> Repo.update()
183
+
end
184
+
185
+
@doc """
186
+
Deletes a vibe.
187
+
"""
188
+
def delete_vibe(%Vibe{} = vibe) do
189
+
Repo.delete(vibe)
190
+
end
191
+
192
+
@doc """
193
+
Joins a user to a vibe.
194
+
"""
195
+
def join_vibe(member_did, vibe_id, attrs \\ %{}) do
196
+
attrs = Map.merge(attrs, %{
197
+
member_did: member_did,
198
+
vibe_id: vibe_id
199
+
})
200
+
201
+
%VibeMember{}
202
+
|> VibeMember.changeset(attrs)
203
+
|> Repo.insert()
204
+
|> case do
205
+
{:ok, member} ->
206
+
update_member_count(vibe_id)
207
+
{:ok, member}
208
+
error -> error
209
+
end
210
+
end
211
+
212
+
@doc """
213
+
Checks if a user is a member of a vibe.
214
+
"""
215
+
def is_member?(member_did, vibe_uri) do
216
+
VibeMember
217
+
|> where([vm], vm.member_did == ^member_did and vm.vibe_uri == ^vibe_uri)
218
+
|> Repo.exists?()
219
+
end
220
+
221
+
@doc """
222
+
Updates the member count for a vibe.
223
+
"""
224
+
def update_member_count(vibe_id) do
225
+
count = VibeMember
226
+
|> where([vm], vm.vibe_id == ^vibe_id)
227
+
|> Repo.aggregate(:count)
228
+
229
+
vibe = Repo.get!(Vibe, vibe_id)
230
+
update_vibe(vibe, %{member_count: count})
231
+
end
232
+
233
+
@doc """
234
+
Records a vibe mention.
235
+
"""
236
+
def record_vibe_mention(attrs) do
237
+
%VibeMention{}
238
+
|> VibeMention.changeset(attrs)
239
+
|> Repo.insert()
240
+
|> case do
241
+
{:ok, mention} ->
242
+
check_and_formalize_vibe(mention.vibe_name)
243
+
{:ok, mention}
244
+
error -> error
245
+
end
246
+
end
247
+
248
+
@doc """
249
+
Checks if a vibe should be formalized based on mention thresholds.
250
+
"""
251
+
def check_and_formalize_vibe(vibe_name) do
252
+
unique_authors = VibeMention
253
+
|> where([vm], vm.vibe_name == ^vibe_name)
254
+
|> distinct([vm], vm.author_did)
255
+
|> Repo.aggregate(:count)
256
+
257
+
total_mentions = VibeMention
258
+
|> where([vm], vm.vibe_name == ^vibe_name)
259
+
|> Repo.aggregate(:count)
260
+
261
+
if unique_authors >= 5 or total_mentions >= 10 do
262
+
unless get_vibe_by_name(vibe_name) do
263
+
case create_vibe(%{
264
+
uri: "at://blonk.app/vibe/#{vibe_name}",
265
+
cid: "bafyrei#{:crypto.strong_rand_bytes(32) |> Base.encode32(case: :lower, padding: false)}",
266
+
creator_did: "did:plc:blonk",
267
+
name: vibe_name,
268
+
mood: vibe_name,
269
+
is_emerging: false
270
+
}) do
271
+
{:ok, vibe} ->
272
+
# Broadcast the emergence
273
+
Phoenix.PubSub.broadcast(
274
+
ElixirBlonk.PubSub,
275
+
"vibes:emerged",
276
+
{:vibe_emerged, vibe}
277
+
)
278
+
{:ok, vibe}
279
+
error -> error
280
+
end
281
+
end
282
+
end
283
+
end
284
+
285
+
@doc """
286
+
Gets vibe mention statistics.
287
+
"""
288
+
def get_vibe_mention_stats do
289
+
VibeMention
290
+
|> group_by([vm], vm.vibe_name)
291
+
|> select([vm], %{
292
+
vibe_name: vm.vibe_name,
293
+
total_mentions: count(vm.id),
294
+
unique_authors: count(fragment("DISTINCT ?", vm.author_did))
295
+
})
296
+
|> Repo.all()
297
+
end
298
+
299
+
@doc """
300
+
Updates pulse scores for all vibes based on recent activity.
301
+
"""
302
+
def update_pulse_scores do
303
+
# This would be called periodically to update vibe pulse scores
304
+
# based on recent blip activity, member count, etc.
305
+
# Implementation depends on specific scoring algorithm
306
+
end
307
+
end
+2370
priv/static/assets/app.css
+2370
priv/static/assets/app.css
···
1
+
/*
2
+
! tailwindcss v3.4.3 | MIT License | https://tailwindcss.com
3
+
*/
4
+
5
+
/*
6
+
1. Prevent padding and border from affecting element width. (https://github.com/mozdevs/cssremedy/issues/4)
7
+
2. Allow adding a border to an element by just adding a border-width. (https://github.com/tailwindcss/tailwindcss/pull/116)
8
+
*/
9
+
10
+
*,
11
+
::before,
12
+
::after {
13
+
box-sizing: border-box;
14
+
/* 1 */
15
+
border-width: 0;
16
+
/* 2 */
17
+
border-style: solid;
18
+
/* 2 */
19
+
border-color: #e5e7eb;
20
+
/* 2 */
21
+
}
22
+
23
+
::before,
24
+
::after {
25
+
--tw-content: '';
26
+
}
27
+
28
+
/*
29
+
1. Use a consistent sensible line-height in all browsers.
30
+
2. Prevent adjustments of font size after orientation changes in iOS.
31
+
3. Use a more readable tab size.
32
+
4. Use the user's configured `sans` font-family by default.
33
+
5. Use the user's configured `sans` font-feature-settings by default.
34
+
6. Use the user's configured `sans` font-variation-settings by default.
35
+
7. Disable tap highlights on iOS
36
+
*/
37
+
38
+
html,
39
+
:host {
40
+
line-height: 1.5;
41
+
/* 1 */
42
+
-webkit-text-size-adjust: 100%;
43
+
/* 2 */
44
+
-moz-tab-size: 4;
45
+
/* 3 */
46
+
-o-tab-size: 4;
47
+
tab-size: 4;
48
+
/* 3 */
49
+
font-family: ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
50
+
/* 4 */
51
+
font-feature-settings: normal;
52
+
/* 5 */
53
+
font-variation-settings: normal;
54
+
/* 6 */
55
+
-webkit-tap-highlight-color: transparent;
56
+
/* 7 */
57
+
}
58
+
59
+
/*
60
+
1. Remove the margin in all browsers.
61
+
2. Inherit line-height from `html` so users can set them as a class directly on the `html` element.
62
+
*/
63
+
64
+
body {
65
+
margin: 0;
66
+
/* 1 */
67
+
line-height: inherit;
68
+
/* 2 */
69
+
}
70
+
71
+
/*
72
+
1. Add the correct height in Firefox.
73
+
2. Correct the inheritance of border color in Firefox. (https://bugzilla.mozilla.org/show_bug.cgi?id=190655)
74
+
3. Ensure horizontal rules are visible by default.
75
+
*/
76
+
77
+
hr {
78
+
height: 0;
79
+
/* 1 */
80
+
color: inherit;
81
+
/* 2 */
82
+
border-top-width: 1px;
83
+
/* 3 */
84
+
}
85
+
86
+
/*
87
+
Add the correct text decoration in Chrome, Edge, and Safari.
88
+
*/
89
+
90
+
abbr:where([title]) {
91
+
-webkit-text-decoration: underline dotted;
92
+
text-decoration: underline dotted;
93
+
}
94
+
95
+
/*
96
+
Remove the default font size and weight for headings.
97
+
*/
98
+
99
+
h1,
100
+
h2,
101
+
h3,
102
+
h4,
103
+
h5,
104
+
h6 {
105
+
font-size: inherit;
106
+
font-weight: inherit;
107
+
}
108
+
109
+
/*
110
+
Reset links to optimize for opt-in styling instead of opt-out.
111
+
*/
112
+
113
+
a {
114
+
color: inherit;
115
+
text-decoration: inherit;
116
+
}
117
+
118
+
/*
119
+
Add the correct font weight in Edge and Safari.
120
+
*/
121
+
122
+
b,
123
+
strong {
124
+
font-weight: bolder;
125
+
}
126
+
127
+
/*
128
+
1. Use the user's configured `mono` font-family by default.
129
+
2. Use the user's configured `mono` font-feature-settings by default.
130
+
3. Use the user's configured `mono` font-variation-settings by default.
131
+
4. Correct the odd `em` font sizing in all browsers.
132
+
*/
133
+
134
+
code,
135
+
kbd,
136
+
samp,
137
+
pre {
138
+
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
139
+
/* 1 */
140
+
font-feature-settings: normal;
141
+
/* 2 */
142
+
font-variation-settings: normal;
143
+
/* 3 */
144
+
font-size: 1em;
145
+
/* 4 */
146
+
}
147
+
148
+
/*
149
+
Add the correct font size in all browsers.
150
+
*/
151
+
152
+
small {
153
+
font-size: 80%;
154
+
}
155
+
156
+
/*
157
+
Prevent `sub` and `sup` elements from affecting the line height in all browsers.
158
+
*/
159
+
160
+
sub,
161
+
sup {
162
+
font-size: 75%;
163
+
line-height: 0;
164
+
position: relative;
165
+
vertical-align: baseline;
166
+
}
167
+
168
+
sub {
169
+
bottom: -0.25em;
170
+
}
171
+
172
+
sup {
173
+
top: -0.5em;
174
+
}
175
+
176
+
/*
177
+
1. Remove text indentation from table contents in Chrome and Safari. (https://bugs.chromium.org/p/chromium/issues/detail?id=999088, https://bugs.webkit.org/show_bug.cgi?id=201297)
178
+
2. Correct table border color inheritance in all Chrome and Safari. (https://bugs.chromium.org/p/chromium/issues/detail?id=935729, https://bugs.webkit.org/show_bug.cgi?id=195016)
179
+
3. Remove gaps between table borders by default.
180
+
*/
181
+
182
+
table {
183
+
text-indent: 0;
184
+
/* 1 */
185
+
border-color: inherit;
186
+
/* 2 */
187
+
border-collapse: collapse;
188
+
/* 3 */
189
+
}
190
+
191
+
/*
192
+
1. Change the font styles in all browsers.
193
+
2. Remove the margin in Firefox and Safari.
194
+
3. Remove default padding in all browsers.
195
+
*/
196
+
197
+
button,
198
+
input,
199
+
optgroup,
200
+
select,
201
+
textarea {
202
+
font-family: inherit;
203
+
/* 1 */
204
+
font-feature-settings: inherit;
205
+
/* 1 */
206
+
font-variation-settings: inherit;
207
+
/* 1 */
208
+
font-size: 100%;
209
+
/* 1 */
210
+
font-weight: inherit;
211
+
/* 1 */
212
+
line-height: inherit;
213
+
/* 1 */
214
+
letter-spacing: inherit;
215
+
/* 1 */
216
+
color: inherit;
217
+
/* 1 */
218
+
margin: 0;
219
+
/* 2 */
220
+
padding: 0;
221
+
/* 3 */
222
+
}
223
+
224
+
/*
225
+
Remove the inheritance of text transform in Edge and Firefox.
226
+
*/
227
+
228
+
button,
229
+
select {
230
+
text-transform: none;
231
+
}
232
+
233
+
/*
234
+
1. Correct the inability to style clickable types in iOS and Safari.
235
+
2. Remove default button styles.
236
+
*/
237
+
238
+
button,
239
+
input:where([type='button']),
240
+
input:where([type='reset']),
241
+
input:where([type='submit']) {
242
+
-webkit-appearance: button;
243
+
/* 1 */
244
+
background-color: transparent;
245
+
/* 2 */
246
+
background-image: none;
247
+
/* 2 */
248
+
}
249
+
250
+
/*
251
+
Use the modern Firefox focus style for all focusable elements.
252
+
*/
253
+
254
+
:-moz-focusring {
255
+
outline: auto;
256
+
}
257
+
258
+
/*
259
+
Remove the additional `:invalid` styles in Firefox. (https://github.com/mozilla/gecko-dev/blob/2f9eacd9d3d995c937b4251a5557d95d494c9be1/layout/style/res/forms.css#L728-L737)
260
+
*/
261
+
262
+
:-moz-ui-invalid {
263
+
box-shadow: none;
264
+
}
265
+
266
+
/*
267
+
Add the correct vertical alignment in Chrome and Firefox.
268
+
*/
269
+
270
+
progress {
271
+
vertical-align: baseline;
272
+
}
273
+
274
+
/*
275
+
Correct the cursor style of increment and decrement buttons in Safari.
276
+
*/
277
+
278
+
::-webkit-inner-spin-button,
279
+
::-webkit-outer-spin-button {
280
+
height: auto;
281
+
}
282
+
283
+
/*
284
+
1. Correct the odd appearance in Chrome and Safari.
285
+
2. Correct the outline style in Safari.
286
+
*/
287
+
288
+
[type='search'] {
289
+
-webkit-appearance: textfield;
290
+
/* 1 */
291
+
outline-offset: -2px;
292
+
/* 2 */
293
+
}
294
+
295
+
/*
296
+
Remove the inner padding in Chrome and Safari on macOS.
297
+
*/
298
+
299
+
::-webkit-search-decoration {
300
+
-webkit-appearance: none;
301
+
}
302
+
303
+
/*
304
+
1. Correct the inability to style clickable types in iOS and Safari.
305
+
2. Change font properties to `inherit` in Safari.
306
+
*/
307
+
308
+
::-webkit-file-upload-button {
309
+
-webkit-appearance: button;
310
+
/* 1 */
311
+
font: inherit;
312
+
/* 2 */
313
+
}
314
+
315
+
/*
316
+
Add the correct display in Chrome and Safari.
317
+
*/
318
+
319
+
summary {
320
+
display: list-item;
321
+
}
322
+
323
+
/*
324
+
Removes the default spacing and border for appropriate elements.
325
+
*/
326
+
327
+
blockquote,
328
+
dl,
329
+
dd,
330
+
h1,
331
+
h2,
332
+
h3,
333
+
h4,
334
+
h5,
335
+
h6,
336
+
hr,
337
+
figure,
338
+
p,
339
+
pre {
340
+
margin: 0;
341
+
}
342
+
343
+
fieldset {
344
+
margin: 0;
345
+
padding: 0;
346
+
}
347
+
348
+
legend {
349
+
padding: 0;
350
+
}
351
+
352
+
ol,
353
+
ul,
354
+
menu {
355
+
list-style: none;
356
+
margin: 0;
357
+
padding: 0;
358
+
}
359
+
360
+
/*
361
+
Reset default styling for dialogs.
362
+
*/
363
+
364
+
dialog {
365
+
padding: 0;
366
+
}
367
+
368
+
/*
369
+
Prevent resizing textareas horizontally by default.
370
+
*/
371
+
372
+
textarea {
373
+
resize: vertical;
374
+
}
375
+
376
+
/*
377
+
1. Reset the default placeholder opacity in Firefox. (https://github.com/tailwindlabs/tailwindcss/issues/3300)
378
+
2. Set the default placeholder color to the user's configured gray 400 color.
379
+
*/
380
+
381
+
input::-moz-placeholder, textarea::-moz-placeholder {
382
+
opacity: 1;
383
+
/* 1 */
384
+
color: #9ca3af;
385
+
/* 2 */
386
+
}
387
+
388
+
input::placeholder,
389
+
textarea::placeholder {
390
+
opacity: 1;
391
+
/* 1 */
392
+
color: #9ca3af;
393
+
/* 2 */
394
+
}
395
+
396
+
/*
397
+
Set the default cursor for buttons.
398
+
*/
399
+
400
+
button,
401
+
[role="button"] {
402
+
cursor: pointer;
403
+
}
404
+
405
+
/*
406
+
Make sure disabled buttons don't get the pointer cursor.
407
+
*/
408
+
409
+
:disabled {
410
+
cursor: default;
411
+
}
412
+
413
+
/*
414
+
1. Make replaced elements `display: block` by default. (https://github.com/mozdevs/cssremedy/issues/14)
415
+
2. Add `vertical-align: middle` to align replaced elements more sensibly by default. (https://github.com/jensimmons/cssremedy/issues/14#issuecomment-634934210)
416
+
This can trigger a poorly considered lint error in some tools but is included by design.
417
+
*/
418
+
419
+
img,
420
+
svg,
421
+
video,
422
+
canvas,
423
+
audio,
424
+
iframe,
425
+
embed,
426
+
object {
427
+
display: block;
428
+
/* 1 */
429
+
vertical-align: middle;
430
+
/* 2 */
431
+
}
432
+
433
+
/*
434
+
Constrain images and videos to the parent width and preserve their intrinsic aspect ratio. (https://github.com/mozdevs/cssremedy/issues/14)
435
+
*/
436
+
437
+
img,
438
+
video {
439
+
max-width: 100%;
440
+
height: auto;
441
+
}
442
+
443
+
/* Make elements with the HTML hidden attribute stay hidden by default */
444
+
445
+
[hidden] {
446
+
display: none;
447
+
}
448
+
449
+
[type='text'],[type='email'],[type='url'],[type='password'],[type='number'],[type='date'],[type='datetime-local'],[type='month'],[type='search'],[type='tel'],[type='time'],[type='week'],[multiple],textarea,select {
450
+
-webkit-appearance: none;
451
+
-moz-appearance: none;
452
+
appearance: none;
453
+
background-color: #fff;
454
+
border-color: #6b7280;
455
+
border-width: 1px;
456
+
border-radius: 0px;
457
+
padding-top: 0.5rem;
458
+
padding-right: 0.75rem;
459
+
padding-bottom: 0.5rem;
460
+
padding-left: 0.75rem;
461
+
font-size: 1rem;
462
+
line-height: 1.5rem;
463
+
--tw-shadow: 0 0 #0000;
464
+
}
465
+
466
+
[type='text']:focus, [type='email']:focus, [type='url']:focus, [type='password']:focus, [type='number']:focus, [type='date']:focus, [type='datetime-local']:focus, [type='month']:focus, [type='search']:focus, [type='tel']:focus, [type='time']:focus, [type='week']:focus, [multiple]:focus, textarea:focus, select:focus {
467
+
outline: 2px solid transparent;
468
+
outline-offset: 2px;
469
+
--tw-ring-inset: var(--tw-empty,/*!*/ /*!*/);
470
+
--tw-ring-offset-width: 0px;
471
+
--tw-ring-offset-color: #fff;
472
+
--tw-ring-color: #2563eb;
473
+
--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
474
+
--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);
475
+
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
476
+
border-color: #2563eb;
477
+
}
478
+
479
+
input::-moz-placeholder, textarea::-moz-placeholder {
480
+
color: #6b7280;
481
+
opacity: 1;
482
+
}
483
+
484
+
input::placeholder,textarea::placeholder {
485
+
color: #6b7280;
486
+
opacity: 1;
487
+
}
488
+
489
+
::-webkit-datetime-edit-fields-wrapper {
490
+
padding: 0;
491
+
}
492
+
493
+
::-webkit-date-and-time-value {
494
+
min-height: 1.5em;
495
+
}
496
+
497
+
::-webkit-datetime-edit,::-webkit-datetime-edit-year-field,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute-field,::-webkit-datetime-edit-second-field,::-webkit-datetime-edit-millisecond-field,::-webkit-datetime-edit-meridiem-field {
498
+
padding-top: 0;
499
+
padding-bottom: 0;
500
+
}
501
+
502
+
select {
503
+
background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3e%3cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='1.5' d='M6 8l4 4 4-4'/%3e%3c/svg%3e");
504
+
background-position: right 0.5rem center;
505
+
background-repeat: no-repeat;
506
+
background-size: 1.5em 1.5em;
507
+
padding-right: 2.5rem;
508
+
-webkit-print-color-adjust: exact;
509
+
print-color-adjust: exact;
510
+
}
511
+
512
+
[multiple] {
513
+
background-image: initial;
514
+
background-position: initial;
515
+
background-repeat: unset;
516
+
background-size: initial;
517
+
padding-right: 0.75rem;
518
+
-webkit-print-color-adjust: unset;
519
+
print-color-adjust: unset;
520
+
}
521
+
522
+
[type='checkbox'],[type='radio'] {
523
+
-webkit-appearance: none;
524
+
-moz-appearance: none;
525
+
appearance: none;
526
+
padding: 0;
527
+
-webkit-print-color-adjust: exact;
528
+
print-color-adjust: exact;
529
+
display: inline-block;
530
+
vertical-align: middle;
531
+
background-origin: border-box;
532
+
-webkit-user-select: none;
533
+
-moz-user-select: none;
534
+
user-select: none;
535
+
flex-shrink: 0;
536
+
height: 1rem;
537
+
width: 1rem;
538
+
color: #2563eb;
539
+
background-color: #fff;
540
+
border-color: #6b7280;
541
+
border-width: 1px;
542
+
--tw-shadow: 0 0 #0000;
543
+
}
544
+
545
+
[type='checkbox'] {
546
+
border-radius: 0px;
547
+
}
548
+
549
+
[type='radio'] {
550
+
border-radius: 100%;
551
+
}
552
+
553
+
[type='checkbox']:focus,[type='radio']:focus {
554
+
outline: 2px solid transparent;
555
+
outline-offset: 2px;
556
+
--tw-ring-inset: var(--tw-empty,/*!*/ /*!*/);
557
+
--tw-ring-offset-width: 2px;
558
+
--tw-ring-offset-color: #fff;
559
+
--tw-ring-color: #2563eb;
560
+
--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
561
+
--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);
562
+
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
563
+
}
564
+
565
+
[type='checkbox']:checked,[type='radio']:checked {
566
+
border-color: transparent;
567
+
background-color: currentColor;
568
+
background-size: 100% 100%;
569
+
background-position: center;
570
+
background-repeat: no-repeat;
571
+
}
572
+
573
+
[type='checkbox']:checked {
574
+
background-image: url("data:image/svg+xml,%3csvg viewBox='0 0 16 16' fill='white' xmlns='http://www.w3.org/2000/svg'%3e%3cpath d='M12.207 4.793a1 1 0 010 1.414l-5 5a1 1 0 01-1.414 0l-2-2a1 1 0 011.414-1.414L6.5 9.086l4.293-4.293a1 1 0 011.414 0z'/%3e%3c/svg%3e");
575
+
}
576
+
577
+
[type='radio']:checked {
578
+
background-image: url("data:image/svg+xml,%3csvg viewBox='0 0 16 16' fill='white' xmlns='http://www.w3.org/2000/svg'%3e%3ccircle cx='8' cy='8' r='3'/%3e%3c/svg%3e");
579
+
}
580
+
581
+
[type='checkbox']:checked:hover,[type='checkbox']:checked:focus,[type='radio']:checked:hover,[type='radio']:checked:focus {
582
+
border-color: transparent;
583
+
background-color: currentColor;
584
+
}
585
+
586
+
[type='checkbox']:indeterminate {
587
+
background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 16 16'%3e%3cpath stroke='white' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M4 8h8'/%3e%3c/svg%3e");
588
+
border-color: transparent;
589
+
background-color: currentColor;
590
+
background-size: 100% 100%;
591
+
background-position: center;
592
+
background-repeat: no-repeat;
593
+
}
594
+
595
+
[type='checkbox']:indeterminate:hover,[type='checkbox']:indeterminate:focus {
596
+
border-color: transparent;
597
+
background-color: currentColor;
598
+
}
599
+
600
+
[type='file'] {
601
+
background: unset;
602
+
border-color: inherit;
603
+
border-width: 0;
604
+
border-radius: 0;
605
+
padding: 0;
606
+
font-size: unset;
607
+
line-height: inherit;
608
+
}
609
+
610
+
[type='file']:focus {
611
+
outline: 1px solid ButtonText;
612
+
outline: 1px auto -webkit-focus-ring-color;
613
+
}
614
+
615
+
*, ::before, ::after {
616
+
--tw-border-spacing-x: 0;
617
+
--tw-border-spacing-y: 0;
618
+
--tw-translate-x: 0;
619
+
--tw-translate-y: 0;
620
+
--tw-rotate: 0;
621
+
--tw-skew-x: 0;
622
+
--tw-skew-y: 0;
623
+
--tw-scale-x: 1;
624
+
--tw-scale-y: 1;
625
+
--tw-pan-x: ;
626
+
--tw-pan-y: ;
627
+
--tw-pinch-zoom: ;
628
+
--tw-scroll-snap-strictness: proximity;
629
+
--tw-gradient-from-position: ;
630
+
--tw-gradient-via-position: ;
631
+
--tw-gradient-to-position: ;
632
+
--tw-ordinal: ;
633
+
--tw-slashed-zero: ;
634
+
--tw-numeric-figure: ;
635
+
--tw-numeric-spacing: ;
636
+
--tw-numeric-fraction: ;
637
+
--tw-ring-inset: ;
638
+
--tw-ring-offset-width: 0px;
639
+
--tw-ring-offset-color: #fff;
640
+
--tw-ring-color: rgb(59 130 246 / 0.5);
641
+
--tw-ring-offset-shadow: 0 0 #0000;
642
+
--tw-ring-shadow: 0 0 #0000;
643
+
--tw-shadow: 0 0 #0000;
644
+
--tw-shadow-colored: 0 0 #0000;
645
+
--tw-blur: ;
646
+
--tw-brightness: ;
647
+
--tw-contrast: ;
648
+
--tw-grayscale: ;
649
+
--tw-hue-rotate: ;
650
+
--tw-invert: ;
651
+
--tw-saturate: ;
652
+
--tw-sepia: ;
653
+
--tw-drop-shadow: ;
654
+
--tw-backdrop-blur: ;
655
+
--tw-backdrop-brightness: ;
656
+
--tw-backdrop-contrast: ;
657
+
--tw-backdrop-grayscale: ;
658
+
--tw-backdrop-hue-rotate: ;
659
+
--tw-backdrop-invert: ;
660
+
--tw-backdrop-opacity: ;
661
+
--tw-backdrop-saturate: ;
662
+
--tw-backdrop-sepia: ;
663
+
--tw-contain-size: ;
664
+
--tw-contain-layout: ;
665
+
--tw-contain-paint: ;
666
+
--tw-contain-style: ;
667
+
}
668
+
669
+
::backdrop {
670
+
--tw-border-spacing-x: 0;
671
+
--tw-border-spacing-y: 0;
672
+
--tw-translate-x: 0;
673
+
--tw-translate-y: 0;
674
+
--tw-rotate: 0;
675
+
--tw-skew-x: 0;
676
+
--tw-skew-y: 0;
677
+
--tw-scale-x: 1;
678
+
--tw-scale-y: 1;
679
+
--tw-pan-x: ;
680
+
--tw-pan-y: ;
681
+
--tw-pinch-zoom: ;
682
+
--tw-scroll-snap-strictness: proximity;
683
+
--tw-gradient-from-position: ;
684
+
--tw-gradient-via-position: ;
685
+
--tw-gradient-to-position: ;
686
+
--tw-ordinal: ;
687
+
--tw-slashed-zero: ;
688
+
--tw-numeric-figure: ;
689
+
--tw-numeric-spacing: ;
690
+
--tw-numeric-fraction: ;
691
+
--tw-ring-inset: ;
692
+
--tw-ring-offset-width: 0px;
693
+
--tw-ring-offset-color: #fff;
694
+
--tw-ring-color: rgb(59 130 246 / 0.5);
695
+
--tw-ring-offset-shadow: 0 0 #0000;
696
+
--tw-ring-shadow: 0 0 #0000;
697
+
--tw-shadow: 0 0 #0000;
698
+
--tw-shadow-colored: 0 0 #0000;
699
+
--tw-blur: ;
700
+
--tw-brightness: ;
701
+
--tw-contrast: ;
702
+
--tw-grayscale: ;
703
+
--tw-hue-rotate: ;
704
+
--tw-invert: ;
705
+
--tw-saturate: ;
706
+
--tw-sepia: ;
707
+
--tw-drop-shadow: ;
708
+
--tw-backdrop-blur: ;
709
+
--tw-backdrop-brightness: ;
710
+
--tw-backdrop-contrast: ;
711
+
--tw-backdrop-grayscale: ;
712
+
--tw-backdrop-hue-rotate: ;
713
+
--tw-backdrop-invert: ;
714
+
--tw-backdrop-opacity: ;
715
+
--tw-backdrop-saturate: ;
716
+
--tw-backdrop-sepia: ;
717
+
--tw-contain-size: ;
718
+
--tw-contain-layout: ;
719
+
--tw-contain-paint: ;
720
+
--tw-contain-style: ;
721
+
}
722
+
723
+
.container {
724
+
width: 100%;
725
+
}
726
+
727
+
@media (min-width: 640px) {
728
+
.container {
729
+
max-width: 640px;
730
+
}
731
+
}
732
+
733
+
@media (min-width: 768px) {
734
+
.container {
735
+
max-width: 768px;
736
+
}
737
+
}
738
+
739
+
@media (min-width: 1024px) {
740
+
.container {
741
+
max-width: 1024px;
742
+
}
743
+
}
744
+
745
+
@media (min-width: 1280px) {
746
+
.container {
747
+
max-width: 1280px;
748
+
}
749
+
}
750
+
751
+
@media (min-width: 1536px) {
752
+
.container {
753
+
max-width: 1536px;
754
+
}
755
+
}
756
+
757
+
.hero-arrow-left-solid {
758
+
--hero-arrow-left-solid: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="currentColor" aria-hidden="true" data-slot="icon"> <path fill-rule="evenodd" d="M11.03 3.97a.75.75 0 0 1 0 1.06l-6.22 6.22H21a.75.75 0 0 1 0 1.5H4.81l6.22 6.22a.75.75 0 1 1-1.06 1.06l-7.5-7.5a.75.75 0 0 1 0-1.06l7.5-7.5a.75.75 0 0 1 1.06 0Z" clip-rule="evenodd"/></svg>');
759
+
-webkit-mask: var(--hero-arrow-left-solid);
760
+
mask: var(--hero-arrow-left-solid);
761
+
-webkit-mask-repeat: no-repeat;
762
+
mask-repeat: no-repeat;
763
+
background-color: currentColor;
764
+
vertical-align: middle;
765
+
display: inline-block;
766
+
width: 1.5rem;
767
+
height: 1.5rem;
768
+
}
769
+
770
+
.hero-arrow-path {
771
+
--hero-arrow-path: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" aria-hidden="true" data-slot="icon"> <path stroke-linecap="round" stroke-linejoin="round" d="M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0 3.181 3.183a8.25 8.25 0 0 0 13.803-3.7M4.031 9.865a8.25 8.25 0 0 1 13.803-3.7l3.181 3.182m0-4.991v4.99"/></svg>');
772
+
-webkit-mask: var(--hero-arrow-path);
773
+
mask: var(--hero-arrow-path);
774
+
-webkit-mask-repeat: no-repeat;
775
+
mask-repeat: no-repeat;
776
+
background-color: currentColor;
777
+
vertical-align: middle;
778
+
display: inline-block;
779
+
width: 1.5rem;
780
+
height: 1.5rem;
781
+
}
782
+
783
+
.hero-exclamation-circle-mini {
784
+
--hero-exclamation-circle-mini: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true" data-slot="icon"> <path fill-rule="evenodd" d="M18 10a8 8 0 1 1-16 0 8 8 0 0 1 16 0Zm-8-5a.75.75 0 0 1 .75.75v4.5a.75.75 0 0 1-1.5 0v-4.5A.75.75 0 0 1 10 5Zm0 10a1 1 0 1 0 0-2 1 1 0 0 0 0 2Z" clip-rule="evenodd"/></svg>');
785
+
-webkit-mask: var(--hero-exclamation-circle-mini);
786
+
mask: var(--hero-exclamation-circle-mini);
787
+
-webkit-mask-repeat: no-repeat;
788
+
mask-repeat: no-repeat;
789
+
background-color: currentColor;
790
+
vertical-align: middle;
791
+
display: inline-block;
792
+
width: 1.25rem;
793
+
height: 1.25rem;
794
+
}
795
+
796
+
.hero-information-circle-mini {
797
+
--hero-information-circle-mini: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor" aria-hidden="true" data-slot="icon"> <path fill-rule="evenodd" d="M18 10a8 8 0 1 1-16 0 8 8 0 0 1 16 0Zm-7-4a1 1 0 1 1-2 0 1 1 0 0 1 2 0ZM9 9a.75.75 0 0 0 0 1.5h.253a.25.25 0 0 1 .244.304l-.459 2.066A1.75 1.75 0 0 0 10.747 15H11a.75.75 0 0 0 0-1.5h-.253a.25.25 0 0 1-.244-.304l.459-2.066A1.75 1.75 0 0 0 9.253 9H9Z" clip-rule="evenodd"/></svg>');
798
+
-webkit-mask: var(--hero-information-circle-mini);
799
+
mask: var(--hero-information-circle-mini);
800
+
-webkit-mask-repeat: no-repeat;
801
+
mask-repeat: no-repeat;
802
+
background-color: currentColor;
803
+
vertical-align: middle;
804
+
display: inline-block;
805
+
width: 1.25rem;
806
+
height: 1.25rem;
807
+
}
808
+
809
+
.hero-tag {
810
+
--hero-tag: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" aria-hidden="true" data-slot="icon"> <path stroke-linecap="round" stroke-linejoin="round" d="M9.568 3H5.25A2.25 2.25 0 0 0 3 5.25v4.318c0 .597.237 1.17.659 1.591l9.581 9.581c.699.699 1.78.872 2.607.33a18.095 18.095 0 0 0 5.223-5.223c.542-.827.369-1.908-.33-2.607L11.16 3.66A2.25 2.25 0 0 0 9.568 3Z"/> <path stroke-linecap="round" stroke-linejoin="round" d="M6 6h.008v.008H6V6Z"/></svg>');
811
+
-webkit-mask: var(--hero-tag);
812
+
mask: var(--hero-tag);
813
+
-webkit-mask-repeat: no-repeat;
814
+
mask-repeat: no-repeat;
815
+
background-color: currentColor;
816
+
vertical-align: middle;
817
+
display: inline-block;
818
+
width: 1.5rem;
819
+
height: 1.5rem;
820
+
}
821
+
822
+
.hero-x-mark {
823
+
--hero-x-mark: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" aria-hidden="true" data-slot="icon"> <path stroke-linecap="round" stroke-linejoin="round" d="M6 18 18 6M6 6l12 12"/></svg>');
824
+
-webkit-mask: var(--hero-x-mark);
825
+
mask: var(--hero-x-mark);
826
+
-webkit-mask-repeat: no-repeat;
827
+
mask-repeat: no-repeat;
828
+
background-color: currentColor;
829
+
vertical-align: middle;
830
+
display: inline-block;
831
+
width: 1.5rem;
832
+
height: 1.5rem;
833
+
}
834
+
835
+
.hero-x-mark-solid {
836
+
--hero-x-mark-solid: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="currentColor" aria-hidden="true" data-slot="icon"> <path fill-rule="evenodd" d="M5.47 5.47a.75.75 0 0 1 1.06 0L12 10.94l5.47-5.47a.75.75 0 1 1 1.06 1.06L13.06 12l5.47 5.47a.75.75 0 1 1-1.06 1.06L12 13.06l-5.47 5.47a.75.75 0 0 1-1.06-1.06L10.94 12 5.47 6.53a.75.75 0 0 1 0-1.06Z" clip-rule="evenodd"/></svg>');
837
+
-webkit-mask: var(--hero-x-mark-solid);
838
+
mask: var(--hero-x-mark-solid);
839
+
-webkit-mask-repeat: no-repeat;
840
+
mask-repeat: no-repeat;
841
+
background-color: currentColor;
842
+
vertical-align: middle;
843
+
display: inline-block;
844
+
width: 1.5rem;
845
+
height: 1.5rem;
846
+
}
847
+
848
+
.sr-only {
849
+
position: absolute;
850
+
width: 1px;
851
+
height: 1px;
852
+
padding: 0;
853
+
margin: -1px;
854
+
overflow: hidden;
855
+
clip: rect(0, 0, 0, 0);
856
+
white-space: nowrap;
857
+
border-width: 0;
858
+
}
859
+
860
+
.static {
861
+
position: static;
862
+
}
863
+
864
+
.fixed {
865
+
position: fixed;
866
+
}
867
+
868
+
.absolute {
869
+
position: absolute;
870
+
}
871
+
872
+
.relative {
873
+
position: relative;
874
+
}
875
+
876
+
.inset-0 {
877
+
inset: 0px;
878
+
}
879
+
880
+
.-inset-y-px {
881
+
top: -1px;
882
+
bottom: -1px;
883
+
}
884
+
885
+
.inset-y-0 {
886
+
top: 0px;
887
+
bottom: 0px;
888
+
}
889
+
890
+
.-left-4 {
891
+
left: -1rem;
892
+
}
893
+
894
+
.-right-4 {
895
+
right: -1rem;
896
+
}
897
+
898
+
.left-0 {
899
+
left: 0px;
900
+
}
901
+
902
+
.left-\[40rem\] {
903
+
left: 40rem;
904
+
}
905
+
906
+
.right-0 {
907
+
right: 0px;
908
+
}
909
+
910
+
.right-1 {
911
+
right: 0.25rem;
912
+
}
913
+
914
+
.right-2 {
915
+
right: 0.5rem;
916
+
}
917
+
918
+
.right-5 {
919
+
right: 1.25rem;
920
+
}
921
+
922
+
.top-1 {
923
+
top: 0.25rem;
924
+
}
925
+
926
+
.top-2 {
927
+
top: 0.5rem;
928
+
}
929
+
930
+
.top-20 {
931
+
top: 5rem;
932
+
}
933
+
934
+
.top-6 {
935
+
top: 1.5rem;
936
+
}
937
+
938
+
.z-0 {
939
+
z-index: 0;
940
+
}
941
+
942
+
.z-10 {
943
+
z-index: 10;
944
+
}
945
+
946
+
.z-50 {
947
+
z-index: 50;
948
+
}
949
+
950
+
.-m-3 {
951
+
margin: -0.75rem;
952
+
}
953
+
954
+
.-mx-2 {
955
+
margin-left: -0.5rem;
956
+
margin-right: -0.5rem;
957
+
}
958
+
959
+
.-my-0 {
960
+
margin-top: -0px;
961
+
margin-bottom: -0px;
962
+
}
963
+
964
+
.-my-0\.5 {
965
+
margin-top: -0.125rem;
966
+
margin-bottom: -0.125rem;
967
+
}
968
+
969
+
.-my-4 {
970
+
margin-top: -1rem;
971
+
margin-bottom: -1rem;
972
+
}
973
+
974
+
.mx-auto {
975
+
margin-left: auto;
976
+
margin-right: auto;
977
+
}
978
+
979
+
.mb-2 {
980
+
margin-bottom: 0.5rem;
981
+
}
982
+
983
+
.mb-3 {
984
+
margin-bottom: 0.75rem;
985
+
}
986
+
987
+
.mb-4 {
988
+
margin-bottom: 1rem;
989
+
}
990
+
991
+
.mb-6 {
992
+
margin-bottom: 1.5rem;
993
+
}
994
+
995
+
.mb-8 {
996
+
margin-bottom: 2rem;
997
+
}
998
+
999
+
.ml-1 {
1000
+
margin-left: 0.25rem;
1001
+
}
1002
+
1003
+
.ml-2 {
1004
+
margin-left: 0.5rem;
1005
+
}
1006
+
1007
+
.ml-3 {
1008
+
margin-left: 0.75rem;
1009
+
}
1010
+
1011
+
.ml-4 {
1012
+
margin-left: 1rem;
1013
+
}
1014
+
1015
+
.mr-2 {
1016
+
margin-right: 0.5rem;
1017
+
}
1018
+
1019
+
.mt-0 {
1020
+
margin-top: 0px;
1021
+
}
1022
+
1023
+
.mt-0\.5 {
1024
+
margin-top: 0.125rem;
1025
+
}
1026
+
1027
+
.mt-1 {
1028
+
margin-top: 0.25rem;
1029
+
}
1030
+
1031
+
.mt-10 {
1032
+
margin-top: 2.5rem;
1033
+
}
1034
+
1035
+
.mt-11 {
1036
+
margin-top: 2.75rem;
1037
+
}
1038
+
1039
+
.mt-14 {
1040
+
margin-top: 3.5rem;
1041
+
}
1042
+
1043
+
.mt-16 {
1044
+
margin-top: 4rem;
1045
+
}
1046
+
1047
+
.mt-2 {
1048
+
margin-top: 0.5rem;
1049
+
}
1050
+
1051
+
.mt-3 {
1052
+
margin-top: 0.75rem;
1053
+
}
1054
+
1055
+
.mt-4 {
1056
+
margin-top: 1rem;
1057
+
}
1058
+
1059
+
.mt-8 {
1060
+
margin-top: 2rem;
1061
+
}
1062
+
1063
+
.block {
1064
+
display: block;
1065
+
}
1066
+
1067
+
.inline-block {
1068
+
display: inline-block;
1069
+
}
1070
+
1071
+
.flex {
1072
+
display: flex;
1073
+
}
1074
+
1075
+
.inline-flex {
1076
+
display: inline-flex;
1077
+
}
1078
+
1079
+
.table {
1080
+
display: table;
1081
+
}
1082
+
1083
+
.grid {
1084
+
display: grid;
1085
+
}
1086
+
1087
+
.contents {
1088
+
display: contents;
1089
+
}
1090
+
1091
+
.hidden {
1092
+
display: none;
1093
+
}
1094
+
1095
+
.h-12 {
1096
+
height: 3rem;
1097
+
}
1098
+
1099
+
.h-16 {
1100
+
height: 4rem;
1101
+
}
1102
+
1103
+
.h-2 {
1104
+
height: 0.5rem;
1105
+
}
1106
+
1107
+
.h-3 {
1108
+
height: 0.75rem;
1109
+
}
1110
+
1111
+
.h-4 {
1112
+
height: 1rem;
1113
+
}
1114
+
1115
+
.h-5 {
1116
+
height: 1.25rem;
1117
+
}
1118
+
1119
+
.h-6 {
1120
+
height: 1.5rem;
1121
+
}
1122
+
1123
+
.h-full {
1124
+
height: 100%;
1125
+
}
1126
+
1127
+
.min-h-\[200px\] {
1128
+
min-height: 200px;
1129
+
}
1130
+
1131
+
.min-h-\[6rem\] {
1132
+
min-height: 6rem;
1133
+
}
1134
+
1135
+
.min-h-full {
1136
+
min-height: 100%;
1137
+
}
1138
+
1139
+
.w-1\/4 {
1140
+
width: 25%;
1141
+
}
1142
+
1143
+
.w-11\/12 {
1144
+
width: 91.666667%;
1145
+
}
1146
+
1147
+
.w-12 {
1148
+
width: 3rem;
1149
+
}
1150
+
1151
+
.w-14 {
1152
+
width: 3.5rem;
1153
+
}
1154
+
1155
+
.w-3 {
1156
+
width: 0.75rem;
1157
+
}
1158
+
1159
+
.w-32 {
1160
+
width: 8rem;
1161
+
}
1162
+
1163
+
.w-4 {
1164
+
width: 1rem;
1165
+
}
1166
+
1167
+
.w-5 {
1168
+
width: 1.25rem;
1169
+
}
1170
+
1171
+
.w-6 {
1172
+
width: 1.5rem;
1173
+
}
1174
+
1175
+
.w-80 {
1176
+
width: 20rem;
1177
+
}
1178
+
1179
+
.w-\[40rem\] {
1180
+
width: 40rem;
1181
+
}
1182
+
1183
+
.w-full {
1184
+
width: 100%;
1185
+
}
1186
+
1187
+
.min-w-\[3rem\] {
1188
+
min-width: 3rem;
1189
+
}
1190
+
1191
+
.max-w-2xl {
1192
+
max-width: 42rem;
1193
+
}
1194
+
1195
+
.max-w-3xl {
1196
+
max-width: 48rem;
1197
+
}
1198
+
1199
+
.max-w-4xl {
1200
+
max-width: 56rem;
1201
+
}
1202
+
1203
+
.max-w-6xl {
1204
+
max-width: 72rem;
1205
+
}
1206
+
1207
+
.max-w-7xl {
1208
+
max-width: 80rem;
1209
+
}
1210
+
1211
+
.max-w-md {
1212
+
max-width: 28rem;
1213
+
}
1214
+
1215
+
.max-w-sm {
1216
+
max-width: 24rem;
1217
+
}
1218
+
1219
+
.max-w-xl {
1220
+
max-width: 36rem;
1221
+
}
1222
+
1223
+
.flex-1 {
1224
+
flex: 1 1 0%;
1225
+
}
1226
+
1227
+
.flex-none {
1228
+
flex: none;
1229
+
}
1230
+
1231
+
.translate-y-0 {
1232
+
--tw-translate-y: 0px;
1233
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
1234
+
}
1235
+
1236
+
.translate-y-4 {
1237
+
--tw-translate-y: 1rem;
1238
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
1239
+
}
1240
+
1241
+
.transform {
1242
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
1243
+
}
1244
+
1245
+
@keyframes spin {
1246
+
to {
1247
+
transform: rotate(360deg);
1248
+
}
1249
+
}
1250
+
1251
+
.animate-spin {
1252
+
animation: spin 1s linear infinite;
1253
+
}
1254
+
1255
+
.cursor-pointer {
1256
+
cursor: pointer;
1257
+
}
1258
+
1259
+
.resize-none {
1260
+
resize: none;
1261
+
}
1262
+
1263
+
.grid-cols-1 {
1264
+
grid-template-columns: repeat(1, minmax(0, 1fr));
1265
+
}
1266
+
1267
+
.grid-cols-2 {
1268
+
grid-template-columns: repeat(2, minmax(0, 1fr));
1269
+
}
1270
+
1271
+
.flex-col {
1272
+
flex-direction: column;
1273
+
}
1274
+
1275
+
.flex-wrap {
1276
+
flex-wrap: wrap;
1277
+
}
1278
+
1279
+
.items-start {
1280
+
align-items: flex-start;
1281
+
}
1282
+
1283
+
.items-center {
1284
+
align-items: center;
1285
+
}
1286
+
1287
+
.justify-end {
1288
+
justify-content: flex-end;
1289
+
}
1290
+
1291
+
.justify-center {
1292
+
justify-content: center;
1293
+
}
1294
+
1295
+
.justify-between {
1296
+
justify-content: space-between;
1297
+
}
1298
+
1299
+
.gap-1 {
1300
+
gap: 0.25rem;
1301
+
}
1302
+
1303
+
.gap-1\.5 {
1304
+
gap: 0.375rem;
1305
+
}
1306
+
1307
+
.gap-2 {
1308
+
gap: 0.5rem;
1309
+
}
1310
+
1311
+
.gap-3 {
1312
+
gap: 0.75rem;
1313
+
}
1314
+
1315
+
.gap-4 {
1316
+
gap: 1rem;
1317
+
}
1318
+
1319
+
.gap-6 {
1320
+
gap: 1.5rem;
1321
+
}
1322
+
1323
+
.gap-x-6 {
1324
+
-moz-column-gap: 1.5rem;
1325
+
column-gap: 1.5rem;
1326
+
}
1327
+
1328
+
.gap-y-4 {
1329
+
row-gap: 1rem;
1330
+
}
1331
+
1332
+
.space-x-1 > :not([hidden]) ~ :not([hidden]) {
1333
+
--tw-space-x-reverse: 0;
1334
+
margin-right: calc(0.25rem * var(--tw-space-x-reverse));
1335
+
margin-left: calc(0.25rem * calc(1 - var(--tw-space-x-reverse)));
1336
+
}
1337
+
1338
+
.space-x-3 > :not([hidden]) ~ :not([hidden]) {
1339
+
--tw-space-x-reverse: 0;
1340
+
margin-right: calc(0.75rem * var(--tw-space-x-reverse));
1341
+
margin-left: calc(0.75rem * calc(1 - var(--tw-space-x-reverse)));
1342
+
}
1343
+
1344
+
.space-x-4 > :not([hidden]) ~ :not([hidden]) {
1345
+
--tw-space-x-reverse: 0;
1346
+
margin-right: calc(1rem * var(--tw-space-x-reverse));
1347
+
margin-left: calc(1rem * calc(1 - var(--tw-space-x-reverse)));
1348
+
}
1349
+
1350
+
.space-x-6 > :not([hidden]) ~ :not([hidden]) {
1351
+
--tw-space-x-reverse: 0;
1352
+
margin-right: calc(1.5rem * var(--tw-space-x-reverse));
1353
+
margin-left: calc(1.5rem * calc(1 - var(--tw-space-x-reverse)));
1354
+
}
1355
+
1356
+
.space-x-8 > :not([hidden]) ~ :not([hidden]) {
1357
+
--tw-space-x-reverse: 0;
1358
+
margin-right: calc(2rem * var(--tw-space-x-reverse));
1359
+
margin-left: calc(2rem * calc(1 - var(--tw-space-x-reverse)));
1360
+
}
1361
+
1362
+
.space-y-12 > :not([hidden]) ~ :not([hidden]) {
1363
+
--tw-space-y-reverse: 0;
1364
+
margin-top: calc(3rem * calc(1 - var(--tw-space-y-reverse)));
1365
+
margin-bottom: calc(3rem * var(--tw-space-y-reverse));
1366
+
}
1367
+
1368
+
.space-y-4 > :not([hidden]) ~ :not([hidden]) {
1369
+
--tw-space-y-reverse: 0;
1370
+
margin-top: calc(1rem * calc(1 - var(--tw-space-y-reverse)));
1371
+
margin-bottom: calc(1rem * var(--tw-space-y-reverse));
1372
+
}
1373
+
1374
+
.space-y-6 > :not([hidden]) ~ :not([hidden]) {
1375
+
--tw-space-y-reverse: 0;
1376
+
margin-top: calc(1.5rem * calc(1 - var(--tw-space-y-reverse)));
1377
+
margin-bottom: calc(1.5rem * var(--tw-space-y-reverse));
1378
+
}
1379
+
1380
+
.space-y-8 > :not([hidden]) ~ :not([hidden]) {
1381
+
--tw-space-y-reverse: 0;
1382
+
margin-top: calc(2rem * calc(1 - var(--tw-space-y-reverse)));
1383
+
margin-bottom: calc(2rem * var(--tw-space-y-reverse));
1384
+
}
1385
+
1386
+
.divide-y > :not([hidden]) ~ :not([hidden]) {
1387
+
--tw-divide-y-reverse: 0;
1388
+
border-top-width: calc(1px * calc(1 - var(--tw-divide-y-reverse)));
1389
+
border-bottom-width: calc(1px * var(--tw-divide-y-reverse));
1390
+
}
1391
+
1392
+
.divide-zinc-100 > :not([hidden]) ~ :not([hidden]) {
1393
+
--tw-divide-opacity: 1;
1394
+
border-color: rgb(244 244 245 / var(--tw-divide-opacity));
1395
+
}
1396
+
1397
+
.overflow-hidden {
1398
+
overflow: hidden;
1399
+
}
1400
+
1401
+
.overflow-y-auto {
1402
+
overflow-y: auto;
1403
+
}
1404
+
1405
+
.whitespace-nowrap {
1406
+
white-space: nowrap;
1407
+
}
1408
+
1409
+
.text-balance {
1410
+
text-wrap: balance;
1411
+
}
1412
+
1413
+
.rounded {
1414
+
border-radius: 0.25rem;
1415
+
}
1416
+
1417
+
.rounded-2xl {
1418
+
border-radius: 1rem;
1419
+
}
1420
+
1421
+
.rounded-full {
1422
+
border-radius: 9999px;
1423
+
}
1424
+
1425
+
.rounded-lg {
1426
+
border-radius: 0.5rem;
1427
+
}
1428
+
1429
+
.rounded-md {
1430
+
border-radius: 0.375rem;
1431
+
}
1432
+
1433
+
.border {
1434
+
border-width: 1px;
1435
+
}
1436
+
1437
+
.border-2 {
1438
+
border-width: 2px;
1439
+
}
1440
+
1441
+
.border-b {
1442
+
border-bottom-width: 1px;
1443
+
}
1444
+
1445
+
.border-t {
1446
+
border-top-width: 1px;
1447
+
}
1448
+
1449
+
.border-black {
1450
+
--tw-border-opacity: 1;
1451
+
border-color: rgb(0 0 0 / var(--tw-border-opacity));
1452
+
}
1453
+
1454
+
.border-blue-200 {
1455
+
--tw-border-opacity: 1;
1456
+
border-color: rgb(191 219 254 / var(--tw-border-opacity));
1457
+
}
1458
+
1459
+
.border-blue-400 {
1460
+
--tw-border-opacity: 1;
1461
+
border-color: rgb(96 165 250 / var(--tw-border-opacity));
1462
+
}
1463
+
1464
+
.border-gray-200 {
1465
+
--tw-border-opacity: 1;
1466
+
border-color: rgb(229 231 235 / var(--tw-border-opacity));
1467
+
}
1468
+
1469
+
.border-gray-300 {
1470
+
--tw-border-opacity: 1;
1471
+
border-color: rgb(209 213 219 / var(--tw-border-opacity));
1472
+
}
1473
+
1474
+
.border-green-400 {
1475
+
--tw-border-opacity: 1;
1476
+
border-color: rgb(74 222 128 / var(--tw-border-opacity));
1477
+
}
1478
+
1479
+
.border-orange-400 {
1480
+
--tw-border-opacity: 1;
1481
+
border-color: rgb(251 146 60 / var(--tw-border-opacity));
1482
+
}
1483
+
1484
+
.border-pink-400 {
1485
+
--tw-border-opacity: 1;
1486
+
border-color: rgb(244 114 182 / var(--tw-border-opacity));
1487
+
}
1488
+
1489
+
.border-red-200 {
1490
+
--tw-border-opacity: 1;
1491
+
border-color: rgb(254 202 202 / var(--tw-border-opacity));
1492
+
}
1493
+
1494
+
.border-rose-400 {
1495
+
--tw-border-opacity: 1;
1496
+
border-color: rgb(251 113 133 / var(--tw-border-opacity));
1497
+
}
1498
+
1499
+
.border-yellow-400 {
1500
+
--tw-border-opacity: 1;
1501
+
border-color: rgb(250 204 21 / var(--tw-border-opacity));
1502
+
}
1503
+
1504
+
.border-zinc-200 {
1505
+
--tw-border-opacity: 1;
1506
+
border-color: rgb(228 228 231 / var(--tw-border-opacity));
1507
+
}
1508
+
1509
+
.border-zinc-300 {
1510
+
--tw-border-opacity: 1;
1511
+
border-color: rgb(212 212 216 / var(--tw-border-opacity));
1512
+
}
1513
+
1514
+
.bg-black {
1515
+
--tw-bg-opacity: 1;
1516
+
background-color: rgb(0 0 0 / var(--tw-bg-opacity));
1517
+
}
1518
+
1519
+
.bg-blue-100 {
1520
+
--tw-bg-opacity: 1;
1521
+
background-color: rgb(219 234 254 / var(--tw-bg-opacity));
1522
+
}
1523
+
1524
+
.bg-blue-50 {
1525
+
--tw-bg-opacity: 1;
1526
+
background-color: rgb(239 246 255 / var(--tw-bg-opacity));
1527
+
}
1528
+
1529
+
.bg-blue-600 {
1530
+
--tw-bg-opacity: 1;
1531
+
background-color: rgb(37 99 235 / var(--tw-bg-opacity));
1532
+
}
1533
+
1534
+
.bg-brand\/5 {
1535
+
background-color: rgb(253 79 0 / 0.05);
1536
+
}
1537
+
1538
+
.bg-emerald-50 {
1539
+
--tw-bg-opacity: 1;
1540
+
background-color: rgb(236 253 245 / var(--tw-bg-opacity));
1541
+
}
1542
+
1543
+
.bg-gray-100 {
1544
+
--tw-bg-opacity: 1;
1545
+
background-color: rgb(243 244 246 / var(--tw-bg-opacity));
1546
+
}
1547
+
1548
+
.bg-gray-200 {
1549
+
--tw-bg-opacity: 1;
1550
+
background-color: rgb(229 231 235 / var(--tw-bg-opacity));
1551
+
}
1552
+
1553
+
.bg-gray-300 {
1554
+
--tw-bg-opacity: 1;
1555
+
background-color: rgb(209 213 219 / var(--tw-bg-opacity));
1556
+
}
1557
+
1558
+
.bg-gray-50 {
1559
+
--tw-bg-opacity: 1;
1560
+
background-color: rgb(249 250 251 / var(--tw-bg-opacity));
1561
+
}
1562
+
1563
+
.bg-gray-600 {
1564
+
--tw-bg-opacity: 1;
1565
+
background-color: rgb(75 85 99 / var(--tw-bg-opacity));
1566
+
}
1567
+
1568
+
.bg-red-50 {
1569
+
--tw-bg-opacity: 1;
1570
+
background-color: rgb(254 242 242 / var(--tw-bg-opacity));
1571
+
}
1572
+
1573
+
.bg-rose-50 {
1574
+
--tw-bg-opacity: 1;
1575
+
background-color: rgb(255 241 242 / var(--tw-bg-opacity));
1576
+
}
1577
+
1578
+
.bg-white {
1579
+
--tw-bg-opacity: 1;
1580
+
background-color: rgb(255 255 255 / var(--tw-bg-opacity));
1581
+
}
1582
+
1583
+
.bg-zinc-50 {
1584
+
--tw-bg-opacity: 1;
1585
+
background-color: rgb(250 250 250 / var(--tw-bg-opacity));
1586
+
}
1587
+
1588
+
.bg-zinc-50\/90 {
1589
+
background-color: rgb(250 250 250 / 0.9);
1590
+
}
1591
+
1592
+
.bg-zinc-900 {
1593
+
--tw-bg-opacity: 1;
1594
+
background-color: rgb(24 24 27 / var(--tw-bg-opacity));
1595
+
}
1596
+
1597
+
.bg-opacity-50 {
1598
+
--tw-bg-opacity: 0.5;
1599
+
}
1600
+
1601
+
.fill-cyan-900 {
1602
+
fill: #164e63;
1603
+
}
1604
+
1605
+
.fill-rose-900 {
1606
+
fill: #881337;
1607
+
}
1608
+
1609
+
.fill-zinc-400 {
1610
+
fill: #a1a1aa;
1611
+
}
1612
+
1613
+
.p-0 {
1614
+
padding: 0px;
1615
+
}
1616
+
1617
+
.p-14 {
1618
+
padding: 3.5rem;
1619
+
}
1620
+
1621
+
.p-2 {
1622
+
padding: 0.5rem;
1623
+
}
1624
+
1625
+
.p-3 {
1626
+
padding: 0.75rem;
1627
+
}
1628
+
1629
+
.p-4 {
1630
+
padding: 1rem;
1631
+
}
1632
+
1633
+
.p-5 {
1634
+
padding: 1.25rem;
1635
+
}
1636
+
1637
+
.p-6 {
1638
+
padding: 1.5rem;
1639
+
}
1640
+
1641
+
.p-8 {
1642
+
padding: 2rem;
1643
+
}
1644
+
1645
+
.px-1 {
1646
+
padding-left: 0.25rem;
1647
+
padding-right: 0.25rem;
1648
+
}
1649
+
1650
+
.px-2 {
1651
+
padding-left: 0.5rem;
1652
+
padding-right: 0.5rem;
1653
+
}
1654
+
1655
+
.px-3 {
1656
+
padding-left: 0.75rem;
1657
+
padding-right: 0.75rem;
1658
+
}
1659
+
1660
+
.px-4 {
1661
+
padding-left: 1rem;
1662
+
padding-right: 1rem;
1663
+
}
1664
+
1665
+
.px-6 {
1666
+
padding-left: 1.5rem;
1667
+
padding-right: 1.5rem;
1668
+
}
1669
+
1670
+
.px-8 {
1671
+
padding-left: 2rem;
1672
+
padding-right: 2rem;
1673
+
}
1674
+
1675
+
.py-0 {
1676
+
padding-top: 0px;
1677
+
padding-bottom: 0px;
1678
+
}
1679
+
1680
+
.py-0\.5 {
1681
+
padding-top: 0.125rem;
1682
+
padding-bottom: 0.125rem;
1683
+
}
1684
+
1685
+
.py-1 {
1686
+
padding-top: 0.25rem;
1687
+
padding-bottom: 0.25rem;
1688
+
}
1689
+
1690
+
.py-10 {
1691
+
padding-top: 2.5rem;
1692
+
padding-bottom: 2.5rem;
1693
+
}
1694
+
1695
+
.py-12 {
1696
+
padding-top: 3rem;
1697
+
padding-bottom: 3rem;
1698
+
}
1699
+
1700
+
.py-2 {
1701
+
padding-top: 0.5rem;
1702
+
padding-bottom: 0.5rem;
1703
+
}
1704
+
1705
+
.py-3 {
1706
+
padding-top: 0.75rem;
1707
+
padding-bottom: 0.75rem;
1708
+
}
1709
+
1710
+
.py-4 {
1711
+
padding-top: 1rem;
1712
+
padding-bottom: 1rem;
1713
+
}
1714
+
1715
+
.py-8 {
1716
+
padding-top: 2rem;
1717
+
padding-bottom: 2rem;
1718
+
}
1719
+
1720
+
.pb-4 {
1721
+
padding-bottom: 1rem;
1722
+
}
1723
+
1724
+
.pr-6 {
1725
+
padding-right: 1.5rem;
1726
+
}
1727
+
1728
+
.pt-4 {
1729
+
padding-top: 1rem;
1730
+
}
1731
+
1732
+
.text-left {
1733
+
text-align: left;
1734
+
}
1735
+
1736
+
.text-center {
1737
+
text-align: center;
1738
+
}
1739
+
1740
+
.text-right {
1741
+
text-align: right;
1742
+
}
1743
+
1744
+
.font-mono {
1745
+
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
1746
+
}
1747
+
1748
+
.text-2xl {
1749
+
font-size: 1.5rem;
1750
+
line-height: 2rem;
1751
+
}
1752
+
1753
+
.text-3xl {
1754
+
font-size: 1.875rem;
1755
+
line-height: 2.25rem;
1756
+
}
1757
+
1758
+
.text-4xl {
1759
+
font-size: 2.25rem;
1760
+
line-height: 2.5rem;
1761
+
}
1762
+
1763
+
.text-\[0\.8125rem\] {
1764
+
font-size: 0.8125rem;
1765
+
}
1766
+
1767
+
.text-\[2rem\] {
1768
+
font-size: 2rem;
1769
+
}
1770
+
1771
+
.text-base {
1772
+
font-size: 1rem;
1773
+
line-height: 1.5rem;
1774
+
}
1775
+
1776
+
.text-lg {
1777
+
font-size: 1.125rem;
1778
+
line-height: 1.75rem;
1779
+
}
1780
+
1781
+
.text-sm {
1782
+
font-size: 0.875rem;
1783
+
line-height: 1.25rem;
1784
+
}
1785
+
1786
+
.text-xl {
1787
+
font-size: 1.25rem;
1788
+
line-height: 1.75rem;
1789
+
}
1790
+
1791
+
.text-xs {
1792
+
font-size: 0.75rem;
1793
+
line-height: 1rem;
1794
+
}
1795
+
1796
+
.font-bold {
1797
+
font-weight: 700;
1798
+
}
1799
+
1800
+
.font-medium {
1801
+
font-weight: 500;
1802
+
}
1803
+
1804
+
.font-normal {
1805
+
font-weight: 400;
1806
+
}
1807
+
1808
+
.font-semibold {
1809
+
font-weight: 600;
1810
+
}
1811
+
1812
+
.leading-10 {
1813
+
line-height: 2.5rem;
1814
+
}
1815
+
1816
+
.leading-5 {
1817
+
line-height: 1.25rem;
1818
+
}
1819
+
1820
+
.leading-6 {
1821
+
line-height: 1.5rem;
1822
+
}
1823
+
1824
+
.leading-7 {
1825
+
line-height: 1.75rem;
1826
+
}
1827
+
1828
+
.leading-8 {
1829
+
line-height: 2rem;
1830
+
}
1831
+
1832
+
.leading-relaxed {
1833
+
line-height: 1.625;
1834
+
}
1835
+
1836
+
.tracking-tighter {
1837
+
letter-spacing: -0.05em;
1838
+
}
1839
+
1840
+
.text-blue-400 {
1841
+
--tw-text-opacity: 1;
1842
+
color: rgb(96 165 250 / var(--tw-text-opacity));
1843
+
}
1844
+
1845
+
.text-blue-500 {
1846
+
--tw-text-opacity: 1;
1847
+
color: rgb(59 130 246 / var(--tw-text-opacity));
1848
+
}
1849
+
1850
+
.text-blue-600 {
1851
+
--tw-text-opacity: 1;
1852
+
color: rgb(37 99 235 / var(--tw-text-opacity));
1853
+
}
1854
+
1855
+
.text-blue-700 {
1856
+
--tw-text-opacity: 1;
1857
+
color: rgb(29 78 216 / var(--tw-text-opacity));
1858
+
}
1859
+
1860
+
.text-blue-800 {
1861
+
--tw-text-opacity: 1;
1862
+
color: rgb(30 64 175 / var(--tw-text-opacity));
1863
+
}
1864
+
1865
+
.text-brand {
1866
+
--tw-text-opacity: 1;
1867
+
color: rgb(253 79 0 / var(--tw-text-opacity));
1868
+
}
1869
+
1870
+
.text-emerald-800 {
1871
+
--tw-text-opacity: 1;
1872
+
color: rgb(6 95 70 / var(--tw-text-opacity));
1873
+
}
1874
+
1875
+
.text-gray-400 {
1876
+
--tw-text-opacity: 1;
1877
+
color: rgb(156 163 175 / var(--tw-text-opacity));
1878
+
}
1879
+
1880
+
.text-gray-500 {
1881
+
--tw-text-opacity: 1;
1882
+
color: rgb(107 114 128 / var(--tw-text-opacity));
1883
+
}
1884
+
1885
+
.text-gray-600 {
1886
+
--tw-text-opacity: 1;
1887
+
color: rgb(75 85 99 / var(--tw-text-opacity));
1888
+
}
1889
+
1890
+
.text-gray-700 {
1891
+
--tw-text-opacity: 1;
1892
+
color: rgb(55 65 81 / var(--tw-text-opacity));
1893
+
}
1894
+
1895
+
.text-gray-900 {
1896
+
--tw-text-opacity: 1;
1897
+
color: rgb(17 24 39 / var(--tw-text-opacity));
1898
+
}
1899
+
1900
+
.text-red-700 {
1901
+
--tw-text-opacity: 1;
1902
+
color: rgb(185 28 28 / var(--tw-text-opacity));
1903
+
}
1904
+
1905
+
.text-rose-600 {
1906
+
--tw-text-opacity: 1;
1907
+
color: rgb(225 29 72 / var(--tw-text-opacity));
1908
+
}
1909
+
1910
+
.text-rose-900 {
1911
+
--tw-text-opacity: 1;
1912
+
color: rgb(136 19 55 / var(--tw-text-opacity));
1913
+
}
1914
+
1915
+
.text-white {
1916
+
--tw-text-opacity: 1;
1917
+
color: rgb(255 255 255 / var(--tw-text-opacity));
1918
+
}
1919
+
1920
+
.text-zinc-500 {
1921
+
--tw-text-opacity: 1;
1922
+
color: rgb(113 113 122 / var(--tw-text-opacity));
1923
+
}
1924
+
1925
+
.text-zinc-600 {
1926
+
--tw-text-opacity: 1;
1927
+
color: rgb(82 82 91 / var(--tw-text-opacity));
1928
+
}
1929
+
1930
+
.text-zinc-700 {
1931
+
--tw-text-opacity: 1;
1932
+
color: rgb(63 63 70 / var(--tw-text-opacity));
1933
+
}
1934
+
1935
+
.text-zinc-800 {
1936
+
--tw-text-opacity: 1;
1937
+
color: rgb(39 39 42 / var(--tw-text-opacity));
1938
+
}
1939
+
1940
+
.text-zinc-900 {
1941
+
--tw-text-opacity: 1;
1942
+
color: rgb(24 24 27 / var(--tw-text-opacity));
1943
+
}
1944
+
1945
+
.opacity-0 {
1946
+
opacity: 0;
1947
+
}
1948
+
1949
+
.opacity-100 {
1950
+
opacity: 1;
1951
+
}
1952
+
1953
+
.opacity-20 {
1954
+
opacity: 0.2;
1955
+
}
1956
+
1957
+
.opacity-40 {
1958
+
opacity: 0.4;
1959
+
}
1960
+
1961
+
.shadow-lg {
1962
+
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
1963
+
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
1964
+
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
1965
+
}
1966
+
1967
+
.shadow-md {
1968
+
--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / 0.1), 0 2px 4px -2px rgb(0 0 0 / 0.1);
1969
+
--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);
1970
+
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
1971
+
}
1972
+
1973
+
.shadow-sm {
1974
+
--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / 0.05);
1975
+
--tw-shadow-colored: 0 1px 2px 0 var(--tw-shadow-color);
1976
+
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
1977
+
}
1978
+
1979
+
.shadow-zinc-700\/10 {
1980
+
--tw-shadow-color: rgb(63 63 70 / 0.1);
1981
+
--tw-shadow: var(--tw-shadow-colored);
1982
+
}
1983
+
1984
+
.outline {
1985
+
outline-style: solid;
1986
+
}
1987
+
1988
+
.ring-1 {
1989
+
--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
1990
+
--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);
1991
+
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
1992
+
}
1993
+
1994
+
.ring-emerald-500 {
1995
+
--tw-ring-opacity: 1;
1996
+
--tw-ring-color: rgb(16 185 129 / var(--tw-ring-opacity));
1997
+
}
1998
+
1999
+
.ring-rose-500 {
2000
+
--tw-ring-opacity: 1;
2001
+
--tw-ring-color: rgb(244 63 94 / var(--tw-ring-opacity));
2002
+
}
2003
+
2004
+
.ring-zinc-700\/10 {
2005
+
--tw-ring-color: rgb(63 63 70 / 0.1);
2006
+
}
2007
+
2008
+
.transition {
2009
+
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, -webkit-backdrop-filter;
2010
+
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter;
2011
+
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter, -webkit-backdrop-filter;
2012
+
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
2013
+
transition-duration: 150ms;
2014
+
}
2015
+
2016
+
.transition-all {
2017
+
transition-property: all;
2018
+
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
2019
+
transition-duration: 150ms;
2020
+
}
2021
+
2022
+
.transition-colors {
2023
+
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke;
2024
+
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
2025
+
transition-duration: 150ms;
2026
+
}
2027
+
2028
+
.transition-opacity {
2029
+
transition-property: opacity;
2030
+
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
2031
+
transition-duration: 150ms;
2032
+
}
2033
+
2034
+
.duration-200 {
2035
+
transition-duration: 200ms;
2036
+
}
2037
+
2038
+
.duration-300 {
2039
+
transition-duration: 300ms;
2040
+
}
2041
+
2042
+
.ease-in {
2043
+
transition-timing-function: cubic-bezier(0.4, 0, 1, 1);
2044
+
}
2045
+
2046
+
.ease-out {
2047
+
transition-timing-function: cubic-bezier(0, 0, 0.2, 1);
2048
+
}
2049
+
2050
+
.\[scrollbar-gutter\:stable\] {
2051
+
scrollbar-gutter: stable;
2052
+
}
2053
+
2054
+
/* This file is for your main application CSS */
2055
+
2056
+
.last\:border-b-0:last-child {
2057
+
border-bottom-width: 0px;
2058
+
}
2059
+
2060
+
.hover\:cursor-pointer:hover {
2061
+
cursor: pointer;
2062
+
}
2063
+
2064
+
.hover\:border-black:hover {
2065
+
--tw-border-opacity: 1;
2066
+
border-color: rgb(0 0 0 / var(--tw-border-opacity));
2067
+
}
2068
+
2069
+
.hover\:bg-blue-200:hover {
2070
+
--tw-bg-opacity: 1;
2071
+
background-color: rgb(191 219 254 / var(--tw-bg-opacity));
2072
+
}
2073
+
2074
+
.hover\:bg-blue-700:hover {
2075
+
--tw-bg-opacity: 1;
2076
+
background-color: rgb(29 78 216 / var(--tw-bg-opacity));
2077
+
}
2078
+
2079
+
.hover\:bg-gray-100:hover {
2080
+
--tw-bg-opacity: 1;
2081
+
background-color: rgb(243 244 246 / var(--tw-bg-opacity));
2082
+
}
2083
+
2084
+
.hover\:bg-gray-400:hover {
2085
+
--tw-bg-opacity: 1;
2086
+
background-color: rgb(156 163 175 / var(--tw-bg-opacity));
2087
+
}
2088
+
2089
+
.hover\:bg-gray-50:hover {
2090
+
--tw-bg-opacity: 1;
2091
+
background-color: rgb(249 250 251 / var(--tw-bg-opacity));
2092
+
}
2093
+
2094
+
.hover\:bg-gray-800:hover {
2095
+
--tw-bg-opacity: 1;
2096
+
background-color: rgb(31 41 55 / var(--tw-bg-opacity));
2097
+
}
2098
+
2099
+
.hover\:bg-zinc-50:hover {
2100
+
--tw-bg-opacity: 1;
2101
+
background-color: rgb(250 250 250 / var(--tw-bg-opacity));
2102
+
}
2103
+
2104
+
.hover\:bg-zinc-700:hover {
2105
+
--tw-bg-opacity: 1;
2106
+
background-color: rgb(63 63 70 / var(--tw-bg-opacity));
2107
+
}
2108
+
2109
+
.hover\:text-blue-200:hover {
2110
+
--tw-text-opacity: 1;
2111
+
color: rgb(191 219 254 / var(--tw-text-opacity));
2112
+
}
2113
+
2114
+
.hover\:text-blue-700:hover {
2115
+
--tw-text-opacity: 1;
2116
+
color: rgb(29 78 216 / var(--tw-text-opacity));
2117
+
}
2118
+
2119
+
.hover\:text-gray-600:hover {
2120
+
--tw-text-opacity: 1;
2121
+
color: rgb(75 85 99 / var(--tw-text-opacity));
2122
+
}
2123
+
2124
+
.hover\:text-zinc-700:hover {
2125
+
--tw-text-opacity: 1;
2126
+
color: rgb(63 63 70 / var(--tw-text-opacity));
2127
+
}
2128
+
2129
+
.hover\:text-zinc-900:hover {
2130
+
--tw-text-opacity: 1;
2131
+
color: rgb(24 24 27 / var(--tw-text-opacity));
2132
+
}
2133
+
2134
+
.hover\:underline:hover {
2135
+
text-decoration-line: underline;
2136
+
}
2137
+
2138
+
.hover\:opacity-40:hover {
2139
+
opacity: 0.4;
2140
+
}
2141
+
2142
+
.hover\:shadow-lg:hover {
2143
+
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
2144
+
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
2145
+
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
2146
+
}
2147
+
2148
+
.focus\:border-rose-400:focus {
2149
+
--tw-border-opacity: 1;
2150
+
border-color: rgb(251 113 133 / var(--tw-border-opacity));
2151
+
}
2152
+
2153
+
.focus\:border-transparent:focus {
2154
+
border-color: transparent;
2155
+
}
2156
+
2157
+
.focus\:border-zinc-400:focus {
2158
+
--tw-border-opacity: 1;
2159
+
border-color: rgb(161 161 170 / var(--tw-border-opacity));
2160
+
}
2161
+
2162
+
.focus\:ring-0:focus {
2163
+
--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
2164
+
--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color);
2165
+
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
2166
+
}
2167
+
2168
+
.focus\:ring-2:focus {
2169
+
--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
2170
+
--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);
2171
+
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
2172
+
}
2173
+
2174
+
.focus\:ring-blue-500:focus {
2175
+
--tw-ring-opacity: 1;
2176
+
--tw-ring-color: rgb(59 130 246 / var(--tw-ring-opacity));
2177
+
}
2178
+
2179
+
.active\:text-white\/80:active {
2180
+
color: rgb(255 255 255 / 0.8);
2181
+
}
2182
+
2183
+
.group:hover .group-hover\:bg-zinc-100 {
2184
+
--tw-bg-opacity: 1;
2185
+
background-color: rgb(244 244 245 / var(--tw-bg-opacity));
2186
+
}
2187
+
2188
+
.group:hover .group-hover\:bg-zinc-50 {
2189
+
--tw-bg-opacity: 1;
2190
+
background-color: rgb(250 250 250 / var(--tw-bg-opacity));
2191
+
}
2192
+
2193
+
.group:hover .group-hover\:fill-zinc-600 {
2194
+
fill: #52525b;
2195
+
}
2196
+
2197
+
.group:hover .group-hover\:opacity-70 {
2198
+
opacity: 0.7;
2199
+
}
2200
+
2201
+
.phx-submit-loading.phx-submit-loading\:opacity-75 {
2202
+
opacity: 0.75;
2203
+
}
2204
+
2205
+
.phx-submit-loading .phx-submit-loading\:opacity-75 {
2206
+
opacity: 0.75;
2207
+
}
2208
+
2209
+
@media (min-width: 640px) {
2210
+
.sm\:w-96 {
2211
+
width: 24rem;
2212
+
}
2213
+
2214
+
.sm\:w-auto {
2215
+
width: auto;
2216
+
}
2217
+
2218
+
.sm\:w-full {
2219
+
width: 100%;
2220
+
}
2221
+
2222
+
.sm\:translate-y-0 {
2223
+
--tw-translate-y: 0px;
2224
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
2225
+
}
2226
+
2227
+
.sm\:scale-100 {
2228
+
--tw-scale-x: 1;
2229
+
--tw-scale-y: 1;
2230
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
2231
+
}
2232
+
2233
+
.sm\:scale-95 {
2234
+
--tw-scale-x: .95;
2235
+
--tw-scale-y: .95;
2236
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
2237
+
}
2238
+
2239
+
.sm\:grid-cols-2 {
2240
+
grid-template-columns: repeat(2, minmax(0, 1fr));
2241
+
}
2242
+
2243
+
.sm\:grid-cols-3 {
2244
+
grid-template-columns: repeat(3, minmax(0, 1fr));
2245
+
}
2246
+
2247
+
.sm\:flex-col {
2248
+
flex-direction: column;
2249
+
}
2250
+
2251
+
.sm\:gap-8 {
2252
+
gap: 2rem;
2253
+
}
2254
+
2255
+
.sm\:overflow-visible {
2256
+
overflow: visible;
2257
+
}
2258
+
2259
+
.sm\:rounded-l-xl {
2260
+
border-top-left-radius: 0.75rem;
2261
+
border-bottom-left-radius: 0.75rem;
2262
+
}
2263
+
2264
+
.sm\:rounded-r-xl {
2265
+
border-top-right-radius: 0.75rem;
2266
+
border-bottom-right-radius: 0.75rem;
2267
+
}
2268
+
2269
+
.sm\:p-6 {
2270
+
padding: 1.5rem;
2271
+
}
2272
+
2273
+
.sm\:px-0 {
2274
+
padding-left: 0px;
2275
+
padding-right: 0px;
2276
+
}
2277
+
2278
+
.sm\:px-6 {
2279
+
padding-left: 1.5rem;
2280
+
padding-right: 1.5rem;
2281
+
}
2282
+
2283
+
.sm\:py-28 {
2284
+
padding-top: 7rem;
2285
+
padding-bottom: 7rem;
2286
+
}
2287
+
2288
+
.sm\:py-6 {
2289
+
padding-top: 1.5rem;
2290
+
padding-bottom: 1.5rem;
2291
+
}
2292
+
2293
+
.sm\:text-sm {
2294
+
font-size: 0.875rem;
2295
+
line-height: 1.25rem;
2296
+
}
2297
+
2298
+
.sm\:leading-6 {
2299
+
line-height: 1.5rem;
2300
+
}
2301
+
2302
+
.group:hover .sm\:group-hover\:scale-105 {
2303
+
--tw-scale-x: 1.05;
2304
+
--tw-scale-y: 1.05;
2305
+
transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));
2306
+
}
2307
+
}
2308
+
2309
+
@media (min-width: 768px) {
2310
+
.md\:w-3\/4 {
2311
+
width: 75%;
2312
+
}
2313
+
2314
+
.md\:grid-cols-2 {
2315
+
grid-template-columns: repeat(2, minmax(0, 1fr));
2316
+
}
2317
+
2318
+
.md\:grid-cols-3 {
2319
+
grid-template-columns: repeat(3, minmax(0, 1fr));
2320
+
}
2321
+
}
2322
+
2323
+
@media (min-width: 1024px) {
2324
+
.lg\:mx-0 {
2325
+
margin-left: 0px;
2326
+
margin-right: 0px;
2327
+
}
2328
+
2329
+
.lg\:block {
2330
+
display: block;
2331
+
}
2332
+
2333
+
.lg\:w-1\/2 {
2334
+
width: 50%;
2335
+
}
2336
+
2337
+
.lg\:grid-cols-3 {
2338
+
grid-template-columns: repeat(3, minmax(0, 1fr));
2339
+
}
2340
+
2341
+
.lg\:px-8 {
2342
+
padding-left: 2rem;
2343
+
padding-right: 2rem;
2344
+
}
2345
+
2346
+
.lg\:py-8 {
2347
+
padding-top: 2rem;
2348
+
padding-bottom: 2rem;
2349
+
}
2350
+
}
2351
+
2352
+
@media (min-width: 1280px) {
2353
+
.xl\:left-\[50rem\] {
2354
+
left: 50rem;
2355
+
}
2356
+
2357
+
.xl\:grid-cols-4 {
2358
+
grid-template-columns: repeat(4, minmax(0, 1fr));
2359
+
}
2360
+
2361
+
.xl\:px-28 {
2362
+
padding-left: 7rem;
2363
+
padding-right: 7rem;
2364
+
}
2365
+
2366
+
.xl\:py-32 {
2367
+
padding-top: 8rem;
2368
+
padding-bottom: 8rem;
2369
+
}
2370
+
}
+7282
priv/static/assets/app.js
+7282
priv/static/assets/app.js
···
1
+
(() => {
2
+
var __create = Object.create;
3
+
var __defProp = Object.defineProperty;
4
+
var __defProps = Object.defineProperties;
5
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
6
+
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
7
+
var __getOwnPropNames = Object.getOwnPropertyNames;
8
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
9
+
var __getProtoOf = Object.getPrototypeOf;
10
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
11
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
12
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
13
+
var __spreadValues = (a, b) => {
14
+
for (var prop in b || (b = {}))
15
+
if (__hasOwnProp.call(b, prop))
16
+
__defNormalProp(a, prop, b[prop]);
17
+
if (__getOwnPropSymbols)
18
+
for (var prop of __getOwnPropSymbols(b)) {
19
+
if (__propIsEnum.call(b, prop))
20
+
__defNormalProp(a, prop, b[prop]);
21
+
}
22
+
return a;
23
+
};
24
+
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
25
+
var __commonJS = (cb, mod) => function __require() {
26
+
return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
27
+
};
28
+
var __copyProps = (to, from, except, desc) => {
29
+
if (from && typeof from === "object" || typeof from === "function") {
30
+
for (let key of __getOwnPropNames(from))
31
+
if (!__hasOwnProp.call(to, key) && key !== except)
32
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
33
+
}
34
+
return to;
35
+
};
36
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
37
+
// If the importer is in node compatibility mode or this is not an ESM
38
+
// file that has been converted to a CommonJS file using a Babel-
39
+
// compatible transform (i.e. "__esModule" has not been set), then set
40
+
// "default" to the CommonJS "module.exports" for node compatibility.
41
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
42
+
mod
43
+
));
44
+
45
+
// vendor/topbar.js
46
+
var require_topbar = __commonJS({
47
+
"vendor/topbar.js"(exports, module) {
48
+
(function(window2, document2) {
49
+
"use strict";
50
+
(function() {
51
+
var lastTime = 0;
52
+
var vendors = ["ms", "moz", "webkit", "o"];
53
+
for (var x = 0; x < vendors.length && !window2.requestAnimationFrame; ++x) {
54
+
window2.requestAnimationFrame = window2[vendors[x] + "RequestAnimationFrame"];
55
+
window2.cancelAnimationFrame = window2[vendors[x] + "CancelAnimationFrame"] || window2[vendors[x] + "CancelRequestAnimationFrame"];
56
+
}
57
+
if (!window2.requestAnimationFrame)
58
+
window2.requestAnimationFrame = function(callback, element) {
59
+
var currTime = (/* @__PURE__ */ new Date()).getTime();
60
+
var timeToCall = Math.max(0, 16 - (currTime - lastTime));
61
+
var id = window2.setTimeout(function() {
62
+
callback(currTime + timeToCall);
63
+
}, timeToCall);
64
+
lastTime = currTime + timeToCall;
65
+
return id;
66
+
};
67
+
if (!window2.cancelAnimationFrame)
68
+
window2.cancelAnimationFrame = function(id) {
69
+
clearTimeout(id);
70
+
};
71
+
})();
72
+
var canvas, currentProgress, showing, progressTimerId = null, fadeTimerId = null, delayTimerId = null, addEvent = function(elem, type, handler) {
73
+
if (elem.addEventListener)
74
+
elem.addEventListener(type, handler, false);
75
+
else if (elem.attachEvent)
76
+
elem.attachEvent("on" + type, handler);
77
+
else
78
+
elem["on" + type] = handler;
79
+
}, options = {
80
+
autoRun: true,
81
+
barThickness: 3,
82
+
barColors: {
83
+
0: "rgba(26, 188, 156, .9)",
84
+
".25": "rgba(52, 152, 219, .9)",
85
+
".50": "rgba(241, 196, 15, .9)",
86
+
".75": "rgba(230, 126, 34, .9)",
87
+
"1.0": "rgba(211, 84, 0, .9)"
88
+
},
89
+
shadowBlur: 10,
90
+
shadowColor: "rgba(0, 0, 0, .6)",
91
+
className: null
92
+
}, repaint = function() {
93
+
canvas.width = window2.innerWidth;
94
+
canvas.height = options.barThickness * 5;
95
+
var ctx = canvas.getContext("2d");
96
+
ctx.shadowBlur = options.shadowBlur;
97
+
ctx.shadowColor = options.shadowColor;
98
+
var lineGradient = ctx.createLinearGradient(0, 0, canvas.width, 0);
99
+
for (var stop in options.barColors)
100
+
lineGradient.addColorStop(stop, options.barColors[stop]);
101
+
ctx.lineWidth = options.barThickness;
102
+
ctx.beginPath();
103
+
ctx.moveTo(0, options.barThickness / 2);
104
+
ctx.lineTo(
105
+
Math.ceil(currentProgress * canvas.width),
106
+
options.barThickness / 2
107
+
);
108
+
ctx.strokeStyle = lineGradient;
109
+
ctx.stroke();
110
+
}, createCanvas = function() {
111
+
canvas = document2.createElement("canvas");
112
+
var style = canvas.style;
113
+
style.position = "fixed";
114
+
style.top = style.left = style.right = style.margin = style.padding = 0;
115
+
style.zIndex = 100001;
116
+
style.display = "none";
117
+
if (options.className)
118
+
canvas.classList.add(options.className);
119
+
document2.body.appendChild(canvas);
120
+
addEvent(window2, "resize", repaint);
121
+
}, topbar2 = {
122
+
config: function(opts) {
123
+
for (var key in opts)
124
+
if (options.hasOwnProperty(key))
125
+
options[key] = opts[key];
126
+
},
127
+
show: function(delay) {
128
+
if (showing)
129
+
return;
130
+
if (delay) {
131
+
if (delayTimerId)
132
+
return;
133
+
delayTimerId = setTimeout(() => topbar2.show(), delay);
134
+
} else {
135
+
showing = true;
136
+
if (fadeTimerId !== null)
137
+
window2.cancelAnimationFrame(fadeTimerId);
138
+
if (!canvas)
139
+
createCanvas();
140
+
canvas.style.opacity = 1;
141
+
canvas.style.display = "block";
142
+
topbar2.progress(0);
143
+
if (options.autoRun) {
144
+
(function loop() {
145
+
progressTimerId = window2.requestAnimationFrame(loop);
146
+
topbar2.progress(
147
+
"+" + 0.05 * Math.pow(1 - Math.sqrt(currentProgress), 2)
148
+
);
149
+
})();
150
+
}
151
+
}
152
+
},
153
+
progress: function(to) {
154
+
if (typeof to === "undefined")
155
+
return currentProgress;
156
+
if (typeof to === "string") {
157
+
to = (to.indexOf("+") >= 0 || to.indexOf("-") >= 0 ? currentProgress : 0) + parseFloat(to);
158
+
}
159
+
currentProgress = to > 1 ? 1 : to;
160
+
repaint();
161
+
return currentProgress;
162
+
},
163
+
hide: function() {
164
+
clearTimeout(delayTimerId);
165
+
delayTimerId = null;
166
+
if (!showing)
167
+
return;
168
+
showing = false;
169
+
if (progressTimerId != null) {
170
+
window2.cancelAnimationFrame(progressTimerId);
171
+
progressTimerId = null;
172
+
}
173
+
(function loop() {
174
+
if (topbar2.progress("+.1") >= 1) {
175
+
canvas.style.opacity -= 0.05;
176
+
if (canvas.style.opacity <= 0.05) {
177
+
canvas.style.display = "none";
178
+
fadeTimerId = null;
179
+
return;
180
+
}
181
+
}
182
+
fadeTimerId = window2.requestAnimationFrame(loop);
183
+
})();
184
+
}
185
+
};
186
+
if (typeof module === "object" && typeof module.exports === "object") {
187
+
module.exports = topbar2;
188
+
} else if (typeof define === "function" && define.amd) {
189
+
define(function() {
190
+
return topbar2;
191
+
});
192
+
} else {
193
+
this.topbar = topbar2;
194
+
}
195
+
}).call(exports, window, document);
196
+
}
197
+
});
198
+
199
+
// ../deps/phoenix_html/priv/static/phoenix_html.js
200
+
(function() {
201
+
var PolyfillEvent = eventConstructor();
202
+
function eventConstructor() {
203
+
if (typeof window.CustomEvent === "function")
204
+
return window.CustomEvent;
205
+
function CustomEvent2(event, params) {
206
+
params = params || { bubbles: false, cancelable: false, detail: void 0 };
207
+
var evt = document.createEvent("CustomEvent");
208
+
evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);
209
+
return evt;
210
+
}
211
+
CustomEvent2.prototype = window.Event.prototype;
212
+
return CustomEvent2;
213
+
}
214
+
function buildHiddenInput(name, value) {
215
+
var input = document.createElement("input");
216
+
input.type = "hidden";
217
+
input.name = name;
218
+
input.value = value;
219
+
return input;
220
+
}
221
+
function handleClick(element, targetModifierKey) {
222
+
var to = element.getAttribute("data-to"), method = buildHiddenInput("_method", element.getAttribute("data-method")), csrf = buildHiddenInput("_csrf_token", element.getAttribute("data-csrf")), form = document.createElement("form"), submit = document.createElement("input"), target = element.getAttribute("target");
223
+
form.method = element.getAttribute("data-method") === "get" ? "get" : "post";
224
+
form.action = to;
225
+
form.style.display = "none";
226
+
if (target)
227
+
form.target = target;
228
+
else if (targetModifierKey)
229
+
form.target = "_blank";
230
+
form.appendChild(csrf);
231
+
form.appendChild(method);
232
+
document.body.appendChild(form);
233
+
submit.type = "submit";
234
+
form.appendChild(submit);
235
+
submit.click();
236
+
}
237
+
window.addEventListener("click", function(e) {
238
+
var element = e.target;
239
+
if (e.defaultPrevented)
240
+
return;
241
+
while (element && element.getAttribute) {
242
+
var phoenixLinkEvent = new PolyfillEvent("phoenix.link.click", {
243
+
"bubbles": true,
244
+
"cancelable": true
245
+
});
246
+
if (!element.dispatchEvent(phoenixLinkEvent)) {
247
+
e.preventDefault();
248
+
e.stopImmediatePropagation();
249
+
return false;
250
+
}
251
+
if (element.getAttribute("data-method") && element.getAttribute("data-to")) {
252
+
handleClick(element, e.metaKey || e.shiftKey);
253
+
e.preventDefault();
254
+
return false;
255
+
} else {
256
+
element = element.parentNode;
257
+
}
258
+
}
259
+
}, false);
260
+
window.addEventListener("phoenix.link.click", function(e) {
261
+
var message = e.target.getAttribute("data-confirm");
262
+
if (message && !window.confirm(message)) {
263
+
e.preventDefault();
264
+
}
265
+
}, false);
266
+
})();
267
+
268
+
// ../deps/phoenix/priv/static/phoenix.mjs
269
+
var closure = (value) => {
270
+
if (typeof value === "function") {
271
+
return value;
272
+
} else {
273
+
let closure22 = function() {
274
+
return value;
275
+
};
276
+
return closure22;
277
+
}
278
+
};
279
+
var globalSelf = typeof self !== "undefined" ? self : null;
280
+
var phxWindow = typeof window !== "undefined" ? window : null;
281
+
var global = globalSelf || phxWindow || global;
282
+
var DEFAULT_VSN = "2.0.0";
283
+
var SOCKET_STATES = { connecting: 0, open: 1, closing: 2, closed: 3 };
284
+
var DEFAULT_TIMEOUT = 1e4;
285
+
var WS_CLOSE_NORMAL = 1e3;
286
+
var CHANNEL_STATES = {
287
+
closed: "closed",
288
+
errored: "errored",
289
+
joined: "joined",
290
+
joining: "joining",
291
+
leaving: "leaving"
292
+
};
293
+
var CHANNEL_EVENTS = {
294
+
close: "phx_close",
295
+
error: "phx_error",
296
+
join: "phx_join",
297
+
reply: "phx_reply",
298
+
leave: "phx_leave"
299
+
};
300
+
var TRANSPORTS = {
301
+
longpoll: "longpoll",
302
+
websocket: "websocket"
303
+
};
304
+
var XHR_STATES = {
305
+
complete: 4
306
+
};
307
+
var Push = class {
308
+
constructor(channel, event, payload, timeout) {
309
+
this.channel = channel;
310
+
this.event = event;
311
+
this.payload = payload || function() {
312
+
return {};
313
+
};
314
+
this.receivedResp = null;
315
+
this.timeout = timeout;
316
+
this.timeoutTimer = null;
317
+
this.recHooks = [];
318
+
this.sent = false;
319
+
}
320
+
/**
321
+
*
322
+
* @param {number} timeout
323
+
*/
324
+
resend(timeout) {
325
+
this.timeout = timeout;
326
+
this.reset();
327
+
this.send();
328
+
}
329
+
/**
330
+
*
331
+
*/
332
+
send() {
333
+
if (this.hasReceived("timeout")) {
334
+
return;
335
+
}
336
+
this.startTimeout();
337
+
this.sent = true;
338
+
this.channel.socket.push({
339
+
topic: this.channel.topic,
340
+
event: this.event,
341
+
payload: this.payload(),
342
+
ref: this.ref,
343
+
join_ref: this.channel.joinRef()
344
+
});
345
+
}
346
+
/**
347
+
*
348
+
* @param {*} status
349
+
* @param {*} callback
350
+
*/
351
+
receive(status, callback) {
352
+
if (this.hasReceived(status)) {
353
+
callback(this.receivedResp.response);
354
+
}
355
+
this.recHooks.push({ status, callback });
356
+
return this;
357
+
}
358
+
/**
359
+
* @private
360
+
*/
361
+
reset() {
362
+
this.cancelRefEvent();
363
+
this.ref = null;
364
+
this.refEvent = null;
365
+
this.receivedResp = null;
366
+
this.sent = false;
367
+
}
368
+
/**
369
+
* @private
370
+
*/
371
+
matchReceive({ status, response, _ref }) {
372
+
this.recHooks.filter((h) => h.status === status).forEach((h) => h.callback(response));
373
+
}
374
+
/**
375
+
* @private
376
+
*/
377
+
cancelRefEvent() {
378
+
if (!this.refEvent) {
379
+
return;
380
+
}
381
+
this.channel.off(this.refEvent);
382
+
}
383
+
/**
384
+
* @private
385
+
*/
386
+
cancelTimeout() {
387
+
clearTimeout(this.timeoutTimer);
388
+
this.timeoutTimer = null;
389
+
}
390
+
/**
391
+
* @private
392
+
*/
393
+
startTimeout() {
394
+
if (this.timeoutTimer) {
395
+
this.cancelTimeout();
396
+
}
397
+
this.ref = this.channel.socket.makeRef();
398
+
this.refEvent = this.channel.replyEventName(this.ref);
399
+
this.channel.on(this.refEvent, (payload) => {
400
+
this.cancelRefEvent();
401
+
this.cancelTimeout();
402
+
this.receivedResp = payload;
403
+
this.matchReceive(payload);
404
+
});
405
+
this.timeoutTimer = setTimeout(() => {
406
+
this.trigger("timeout", {});
407
+
}, this.timeout);
408
+
}
409
+
/**
410
+
* @private
411
+
*/
412
+
hasReceived(status) {
413
+
return this.receivedResp && this.receivedResp.status === status;
414
+
}
415
+
/**
416
+
* @private
417
+
*/
418
+
trigger(status, response) {
419
+
this.channel.trigger(this.refEvent, { status, response });
420
+
}
421
+
};
422
+
var Timer = class {
423
+
constructor(callback, timerCalc) {
424
+
this.callback = callback;
425
+
this.timerCalc = timerCalc;
426
+
this.timer = null;
427
+
this.tries = 0;
428
+
}
429
+
reset() {
430
+
this.tries = 0;
431
+
clearTimeout(this.timer);
432
+
}
433
+
/**
434
+
* Cancels any previous scheduleTimeout and schedules callback
435
+
*/
436
+
scheduleTimeout() {
437
+
clearTimeout(this.timer);
438
+
this.timer = setTimeout(() => {
439
+
this.tries = this.tries + 1;
440
+
this.callback();
441
+
}, this.timerCalc(this.tries + 1));
442
+
}
443
+
};
444
+
var Channel = class {
445
+
constructor(topic, params, socket) {
446
+
this.state = CHANNEL_STATES.closed;
447
+
this.topic = topic;
448
+
this.params = closure(params || {});
449
+
this.socket = socket;
450
+
this.bindings = [];
451
+
this.bindingRef = 0;
452
+
this.timeout = this.socket.timeout;
453
+
this.joinedOnce = false;
454
+
this.joinPush = new Push(this, CHANNEL_EVENTS.join, this.params, this.timeout);
455
+
this.pushBuffer = [];
456
+
this.stateChangeRefs = [];
457
+
this.rejoinTimer = new Timer(() => {
458
+
if (this.socket.isConnected()) {
459
+
this.rejoin();
460
+
}
461
+
}, this.socket.rejoinAfterMs);
462
+
this.stateChangeRefs.push(this.socket.onError(() => this.rejoinTimer.reset()));
463
+
this.stateChangeRefs.push(
464
+
this.socket.onOpen(() => {
465
+
this.rejoinTimer.reset();
466
+
if (this.isErrored()) {
467
+
this.rejoin();
468
+
}
469
+
})
470
+
);
471
+
this.joinPush.receive("ok", () => {
472
+
this.state = CHANNEL_STATES.joined;
473
+
this.rejoinTimer.reset();
474
+
this.pushBuffer.forEach((pushEvent) => pushEvent.send());
475
+
this.pushBuffer = [];
476
+
});
477
+
this.joinPush.receive("error", () => {
478
+
this.state = CHANNEL_STATES.errored;
479
+
if (this.socket.isConnected()) {
480
+
this.rejoinTimer.scheduleTimeout();
481
+
}
482
+
});
483
+
this.onClose(() => {
484
+
this.rejoinTimer.reset();
485
+
if (this.socket.hasLogger())
486
+
this.socket.log("channel", `close ${this.topic} ${this.joinRef()}`);
487
+
this.state = CHANNEL_STATES.closed;
488
+
this.socket.remove(this);
489
+
});
490
+
this.onError((reason) => {
491
+
if (this.socket.hasLogger())
492
+
this.socket.log("channel", `error ${this.topic}`, reason);
493
+
if (this.isJoining()) {
494
+
this.joinPush.reset();
495
+
}
496
+
this.state = CHANNEL_STATES.errored;
497
+
if (this.socket.isConnected()) {
498
+
this.rejoinTimer.scheduleTimeout();
499
+
}
500
+
});
501
+
this.joinPush.receive("timeout", () => {
502
+
if (this.socket.hasLogger())
503
+
this.socket.log("channel", `timeout ${this.topic} (${this.joinRef()})`, this.joinPush.timeout);
504
+
let leavePush = new Push(this, CHANNEL_EVENTS.leave, closure({}), this.timeout);
505
+
leavePush.send();
506
+
this.state = CHANNEL_STATES.errored;
507
+
this.joinPush.reset();
508
+
if (this.socket.isConnected()) {
509
+
this.rejoinTimer.scheduleTimeout();
510
+
}
511
+
});
512
+
this.on(CHANNEL_EVENTS.reply, (payload, ref) => {
513
+
this.trigger(this.replyEventName(ref), payload);
514
+
});
515
+
}
516
+
/**
517
+
* Join the channel
518
+
* @param {integer} timeout
519
+
* @returns {Push}
520
+
*/
521
+
join(timeout = this.timeout) {
522
+
if (this.joinedOnce) {
523
+
throw new Error("tried to join multiple times. 'join' can only be called a single time per channel instance");
524
+
} else {
525
+
this.timeout = timeout;
526
+
this.joinedOnce = true;
527
+
this.rejoin();
528
+
return this.joinPush;
529
+
}
530
+
}
531
+
/**
532
+
* Hook into channel close
533
+
* @param {Function} callback
534
+
*/
535
+
onClose(callback) {
536
+
this.on(CHANNEL_EVENTS.close, callback);
537
+
}
538
+
/**
539
+
* Hook into channel errors
540
+
* @param {Function} callback
541
+
*/
542
+
onError(callback) {
543
+
return this.on(CHANNEL_EVENTS.error, (reason) => callback(reason));
544
+
}
545
+
/**
546
+
* Subscribes on channel events
547
+
*
548
+
* Subscription returns a ref counter, which can be used later to
549
+
* unsubscribe the exact event listener
550
+
*
551
+
* @example
552
+
* const ref1 = channel.on("event", do_stuff)
553
+
* const ref2 = channel.on("event", do_other_stuff)
554
+
* channel.off("event", ref1)
555
+
* // Since unsubscription, do_stuff won't fire,
556
+
* // while do_other_stuff will keep firing on the "event"
557
+
*
558
+
* @param {string} event
559
+
* @param {Function} callback
560
+
* @returns {integer} ref
561
+
*/
562
+
on(event, callback) {
563
+
let ref = this.bindingRef++;
564
+
this.bindings.push({ event, ref, callback });
565
+
return ref;
566
+
}
567
+
/**
568
+
* Unsubscribes off of channel events
569
+
*
570
+
* Use the ref returned from a channel.on() to unsubscribe one
571
+
* handler, or pass nothing for the ref to unsubscribe all
572
+
* handlers for the given event.
573
+
*
574
+
* @example
575
+
* // Unsubscribe the do_stuff handler
576
+
* const ref1 = channel.on("event", do_stuff)
577
+
* channel.off("event", ref1)
578
+
*
579
+
* // Unsubscribe all handlers from event
580
+
* channel.off("event")
581
+
*
582
+
* @param {string} event
583
+
* @param {integer} ref
584
+
*/
585
+
off(event, ref) {
586
+
this.bindings = this.bindings.filter((bind) => {
587
+
return !(bind.event === event && (typeof ref === "undefined" || ref === bind.ref));
588
+
});
589
+
}
590
+
/**
591
+
* @private
592
+
*/
593
+
canPush() {
594
+
return this.socket.isConnected() && this.isJoined();
595
+
}
596
+
/**
597
+
* Sends a message `event` to phoenix with the payload `payload`.
598
+
* Phoenix receives this in the `handle_in(event, payload, socket)`
599
+
* function. if phoenix replies or it times out (default 10000ms),
600
+
* then optionally the reply can be received.
601
+
*
602
+
* @example
603
+
* channel.push("event")
604
+
* .receive("ok", payload => console.log("phoenix replied:", payload))
605
+
* .receive("error", err => console.log("phoenix errored", err))
606
+
* .receive("timeout", () => console.log("timed out pushing"))
607
+
* @param {string} event
608
+
* @param {Object} payload
609
+
* @param {number} [timeout]
610
+
* @returns {Push}
611
+
*/
612
+
push(event, payload, timeout = this.timeout) {
613
+
payload = payload || {};
614
+
if (!this.joinedOnce) {
615
+
throw new Error(`tried to push '${event}' to '${this.topic}' before joining. Use channel.join() before pushing events`);
616
+
}
617
+
let pushEvent = new Push(this, event, function() {
618
+
return payload;
619
+
}, timeout);
620
+
if (this.canPush()) {
621
+
pushEvent.send();
622
+
} else {
623
+
pushEvent.startTimeout();
624
+
this.pushBuffer.push(pushEvent);
625
+
}
626
+
return pushEvent;
627
+
}
628
+
/** Leaves the channel
629
+
*
630
+
* Unsubscribes from server events, and
631
+
* instructs channel to terminate on server
632
+
*
633
+
* Triggers onClose() hooks
634
+
*
635
+
* To receive leave acknowledgements, use the `receive`
636
+
* hook to bind to the server ack, ie:
637
+
*
638
+
* @example
639
+
* channel.leave().receive("ok", () => alert("left!") )
640
+
*
641
+
* @param {integer} timeout
642
+
* @returns {Push}
643
+
*/
644
+
leave(timeout = this.timeout) {
645
+
this.rejoinTimer.reset();
646
+
this.joinPush.cancelTimeout();
647
+
this.state = CHANNEL_STATES.leaving;
648
+
let onClose = () => {
649
+
if (this.socket.hasLogger())
650
+
this.socket.log("channel", `leave ${this.topic}`);
651
+
this.trigger(CHANNEL_EVENTS.close, "leave");
652
+
};
653
+
let leavePush = new Push(this, CHANNEL_EVENTS.leave, closure({}), timeout);
654
+
leavePush.receive("ok", () => onClose()).receive("timeout", () => onClose());
655
+
leavePush.send();
656
+
if (!this.canPush()) {
657
+
leavePush.trigger("ok", {});
658
+
}
659
+
return leavePush;
660
+
}
661
+
/**
662
+
* Overridable message hook
663
+
*
664
+
* Receives all events for specialized message handling
665
+
* before dispatching to the channel callbacks.
666
+
*
667
+
* Must return the payload, modified or unmodified
668
+
* @param {string} event
669
+
* @param {Object} payload
670
+
* @param {integer} ref
671
+
* @returns {Object}
672
+
*/
673
+
onMessage(_event, payload, _ref) {
674
+
return payload;
675
+
}
676
+
/**
677
+
* @private
678
+
*/
679
+
isMember(topic, event, payload, joinRef) {
680
+
if (this.topic !== topic) {
681
+
return false;
682
+
}
683
+
if (joinRef && joinRef !== this.joinRef()) {
684
+
if (this.socket.hasLogger())
685
+
this.socket.log("channel", "dropping outdated message", { topic, event, payload, joinRef });
686
+
return false;
687
+
} else {
688
+
return true;
689
+
}
690
+
}
691
+
/**
692
+
* @private
693
+
*/
694
+
joinRef() {
695
+
return this.joinPush.ref;
696
+
}
697
+
/**
698
+
* @private
699
+
*/
700
+
rejoin(timeout = this.timeout) {
701
+
if (this.isLeaving()) {
702
+
return;
703
+
}
704
+
this.socket.leaveOpenTopic(this.topic);
705
+
this.state = CHANNEL_STATES.joining;
706
+
this.joinPush.resend(timeout);
707
+
}
708
+
/**
709
+
* @private
710
+
*/
711
+
trigger(event, payload, ref, joinRef) {
712
+
let handledPayload = this.onMessage(event, payload, ref, joinRef);
713
+
if (payload && !handledPayload) {
714
+
throw new Error("channel onMessage callbacks must return the payload, modified or unmodified");
715
+
}
716
+
let eventBindings = this.bindings.filter((bind) => bind.event === event);
717
+
for (let i = 0; i < eventBindings.length; i++) {
718
+
let bind = eventBindings[i];
719
+
bind.callback(handledPayload, ref, joinRef || this.joinRef());
720
+
}
721
+
}
722
+
/**
723
+
* @private
724
+
*/
725
+
replyEventName(ref) {
726
+
return `chan_reply_${ref}`;
727
+
}
728
+
/**
729
+
* @private
730
+
*/
731
+
isClosed() {
732
+
return this.state === CHANNEL_STATES.closed;
733
+
}
734
+
/**
735
+
* @private
736
+
*/
737
+
isErrored() {
738
+
return this.state === CHANNEL_STATES.errored;
739
+
}
740
+
/**
741
+
* @private
742
+
*/
743
+
isJoined() {
744
+
return this.state === CHANNEL_STATES.joined;
745
+
}
746
+
/**
747
+
* @private
748
+
*/
749
+
isJoining() {
750
+
return this.state === CHANNEL_STATES.joining;
751
+
}
752
+
/**
753
+
* @private
754
+
*/
755
+
isLeaving() {
756
+
return this.state === CHANNEL_STATES.leaving;
757
+
}
758
+
};
759
+
var Ajax = class {
760
+
static request(method, endPoint, accept, body, timeout, ontimeout, callback) {
761
+
if (global.XDomainRequest) {
762
+
let req = new global.XDomainRequest();
763
+
return this.xdomainRequest(req, method, endPoint, body, timeout, ontimeout, callback);
764
+
} else {
765
+
let req = new global.XMLHttpRequest();
766
+
return this.xhrRequest(req, method, endPoint, accept, body, timeout, ontimeout, callback);
767
+
}
768
+
}
769
+
static xdomainRequest(req, method, endPoint, body, timeout, ontimeout, callback) {
770
+
req.timeout = timeout;
771
+
req.open(method, endPoint);
772
+
req.onload = () => {
773
+
let response = this.parseJSON(req.responseText);
774
+
callback && callback(response);
775
+
};
776
+
if (ontimeout) {
777
+
req.ontimeout = ontimeout;
778
+
}
779
+
req.onprogress = () => {
780
+
};
781
+
req.send(body);
782
+
return req;
783
+
}
784
+
static xhrRequest(req, method, endPoint, accept, body, timeout, ontimeout, callback) {
785
+
req.open(method, endPoint, true);
786
+
req.timeout = timeout;
787
+
req.setRequestHeader("Content-Type", accept);
788
+
req.onerror = () => callback && callback(null);
789
+
req.onreadystatechange = () => {
790
+
if (req.readyState === XHR_STATES.complete && callback) {
791
+
let response = this.parseJSON(req.responseText);
792
+
callback(response);
793
+
}
794
+
};
795
+
if (ontimeout) {
796
+
req.ontimeout = ontimeout;
797
+
}
798
+
req.send(body);
799
+
return req;
800
+
}
801
+
static parseJSON(resp) {
802
+
if (!resp || resp === "") {
803
+
return null;
804
+
}
805
+
try {
806
+
return JSON.parse(resp);
807
+
} catch (e) {
808
+
console && console.log("failed to parse JSON response", resp);
809
+
return null;
810
+
}
811
+
}
812
+
static serialize(obj, parentKey) {
813
+
let queryStr = [];
814
+
for (var key in obj) {
815
+
if (!Object.prototype.hasOwnProperty.call(obj, key)) {
816
+
continue;
817
+
}
818
+
let paramKey = parentKey ? `${parentKey}[${key}]` : key;
819
+
let paramVal = obj[key];
820
+
if (typeof paramVal === "object") {
821
+
queryStr.push(this.serialize(paramVal, paramKey));
822
+
} else {
823
+
queryStr.push(encodeURIComponent(paramKey) + "=" + encodeURIComponent(paramVal));
824
+
}
825
+
}
826
+
return queryStr.join("&");
827
+
}
828
+
static appendParams(url, params) {
829
+
if (Object.keys(params).length === 0) {
830
+
return url;
831
+
}
832
+
let prefix = url.match(/\?/) ? "&" : "?";
833
+
return `${url}${prefix}${this.serialize(params)}`;
834
+
}
835
+
};
836
+
var arrayBufferToBase64 = (buffer) => {
837
+
let binary = "";
838
+
let bytes = new Uint8Array(buffer);
839
+
let len = bytes.byteLength;
840
+
for (let i = 0; i < len; i++) {
841
+
binary += String.fromCharCode(bytes[i]);
842
+
}
843
+
return btoa(binary);
844
+
};
845
+
var LongPoll = class {
846
+
constructor(endPoint) {
847
+
this.endPoint = null;
848
+
this.token = null;
849
+
this.skipHeartbeat = true;
850
+
this.reqs = /* @__PURE__ */ new Set();
851
+
this.awaitingBatchAck = false;
852
+
this.currentBatch = null;
853
+
this.currentBatchTimer = null;
854
+
this.batchBuffer = [];
855
+
this.onopen = function() {
856
+
};
857
+
this.onerror = function() {
858
+
};
859
+
this.onmessage = function() {
860
+
};
861
+
this.onclose = function() {
862
+
};
863
+
this.pollEndpoint = this.normalizeEndpoint(endPoint);
864
+
this.readyState = SOCKET_STATES.connecting;
865
+
setTimeout(() => this.poll(), 0);
866
+
}
867
+
normalizeEndpoint(endPoint) {
868
+
return endPoint.replace("ws://", "http://").replace("wss://", "https://").replace(new RegExp("(.*)/" + TRANSPORTS.websocket), "$1/" + TRANSPORTS.longpoll);
869
+
}
870
+
endpointURL() {
871
+
return Ajax.appendParams(this.pollEndpoint, { token: this.token });
872
+
}
873
+
closeAndRetry(code, reason, wasClean) {
874
+
this.close(code, reason, wasClean);
875
+
this.readyState = SOCKET_STATES.connecting;
876
+
}
877
+
ontimeout() {
878
+
this.onerror("timeout");
879
+
this.closeAndRetry(1005, "timeout", false);
880
+
}
881
+
isActive() {
882
+
return this.readyState === SOCKET_STATES.open || this.readyState === SOCKET_STATES.connecting;
883
+
}
884
+
poll() {
885
+
this.ajax("GET", "application/json", null, () => this.ontimeout(), (resp) => {
886
+
if (resp) {
887
+
var { status, token, messages } = resp;
888
+
this.token = token;
889
+
} else {
890
+
status = 0;
891
+
}
892
+
switch (status) {
893
+
case 200:
894
+
messages.forEach((msg) => {
895
+
setTimeout(() => this.onmessage({ data: msg }), 0);
896
+
});
897
+
this.poll();
898
+
break;
899
+
case 204:
900
+
this.poll();
901
+
break;
902
+
case 410:
903
+
this.readyState = SOCKET_STATES.open;
904
+
this.onopen({});
905
+
this.poll();
906
+
break;
907
+
case 403:
908
+
this.onerror(403);
909
+
this.close(1008, "forbidden", false);
910
+
break;
911
+
case 0:
912
+
case 500:
913
+
this.onerror(500);
914
+
this.closeAndRetry(1011, "internal server error", 500);
915
+
break;
916
+
default:
917
+
throw new Error(`unhandled poll status ${status}`);
918
+
}
919
+
});
920
+
}
921
+
// we collect all pushes within the current event loop by
922
+
// setTimeout 0, which optimizes back-to-back procedural
923
+
// pushes against an empty buffer
924
+
send(body) {
925
+
if (typeof body !== "string") {
926
+
body = arrayBufferToBase64(body);
927
+
}
928
+
if (this.currentBatch) {
929
+
this.currentBatch.push(body);
930
+
} else if (this.awaitingBatchAck) {
931
+
this.batchBuffer.push(body);
932
+
} else {
933
+
this.currentBatch = [body];
934
+
this.currentBatchTimer = setTimeout(() => {
935
+
this.batchSend(this.currentBatch);
936
+
this.currentBatch = null;
937
+
}, 0);
938
+
}
939
+
}
940
+
batchSend(messages) {
941
+
this.awaitingBatchAck = true;
942
+
this.ajax("POST", "application/x-ndjson", messages.join("\n"), () => this.onerror("timeout"), (resp) => {
943
+
this.awaitingBatchAck = false;
944
+
if (!resp || resp.status !== 200) {
945
+
this.onerror(resp && resp.status);
946
+
this.closeAndRetry(1011, "internal server error", false);
947
+
} else if (this.batchBuffer.length > 0) {
948
+
this.batchSend(this.batchBuffer);
949
+
this.batchBuffer = [];
950
+
}
951
+
});
952
+
}
953
+
close(code, reason, wasClean) {
954
+
for (let req of this.reqs) {
955
+
req.abort();
956
+
}
957
+
this.readyState = SOCKET_STATES.closed;
958
+
let opts = Object.assign({ code: 1e3, reason: void 0, wasClean: true }, { code, reason, wasClean });
959
+
this.batchBuffer = [];
960
+
clearTimeout(this.currentBatchTimer);
961
+
this.currentBatchTimer = null;
962
+
if (typeof CloseEvent !== "undefined") {
963
+
this.onclose(new CloseEvent("close", opts));
964
+
} else {
965
+
this.onclose(opts);
966
+
}
967
+
}
968
+
ajax(method, contentType, body, onCallerTimeout, callback) {
969
+
let req;
970
+
let ontimeout = () => {
971
+
this.reqs.delete(req);
972
+
onCallerTimeout();
973
+
};
974
+
req = Ajax.request(method, this.endpointURL(), contentType, body, this.timeout, ontimeout, (resp) => {
975
+
this.reqs.delete(req);
976
+
if (this.isActive()) {
977
+
callback(resp);
978
+
}
979
+
});
980
+
this.reqs.add(req);
981
+
}
982
+
};
983
+
var serializer_default = {
984
+
HEADER_LENGTH: 1,
985
+
META_LENGTH: 4,
986
+
KINDS: { push: 0, reply: 1, broadcast: 2 },
987
+
encode(msg, callback) {
988
+
if (msg.payload.constructor === ArrayBuffer) {
989
+
return callback(this.binaryEncode(msg));
990
+
} else {
991
+
let payload = [msg.join_ref, msg.ref, msg.topic, msg.event, msg.payload];
992
+
return callback(JSON.stringify(payload));
993
+
}
994
+
},
995
+
decode(rawPayload, callback) {
996
+
if (rawPayload.constructor === ArrayBuffer) {
997
+
return callback(this.binaryDecode(rawPayload));
998
+
} else {
999
+
let [join_ref, ref, topic, event, payload] = JSON.parse(rawPayload);
1000
+
return callback({ join_ref, ref, topic, event, payload });
1001
+
}
1002
+
},
1003
+
// private
1004
+
binaryEncode(message) {
1005
+
let { join_ref, ref, event, topic, payload } = message;
1006
+
let metaLength = this.META_LENGTH + join_ref.length + ref.length + topic.length + event.length;
1007
+
let header = new ArrayBuffer(this.HEADER_LENGTH + metaLength);
1008
+
let view = new DataView(header);
1009
+
let offset = 0;
1010
+
view.setUint8(offset++, this.KINDS.push);
1011
+
view.setUint8(offset++, join_ref.length);
1012
+
view.setUint8(offset++, ref.length);
1013
+
view.setUint8(offset++, topic.length);
1014
+
view.setUint8(offset++, event.length);
1015
+
Array.from(join_ref, (char) => view.setUint8(offset++, char.charCodeAt(0)));
1016
+
Array.from(ref, (char) => view.setUint8(offset++, char.charCodeAt(0)));
1017
+
Array.from(topic, (char) => view.setUint8(offset++, char.charCodeAt(0)));
1018
+
Array.from(event, (char) => view.setUint8(offset++, char.charCodeAt(0)));
1019
+
var combined = new Uint8Array(header.byteLength + payload.byteLength);
1020
+
combined.set(new Uint8Array(header), 0);
1021
+
combined.set(new Uint8Array(payload), header.byteLength);
1022
+
return combined.buffer;
1023
+
},
1024
+
binaryDecode(buffer) {
1025
+
let view = new DataView(buffer);
1026
+
let kind = view.getUint8(0);
1027
+
let decoder = new TextDecoder();
1028
+
switch (kind) {
1029
+
case this.KINDS.push:
1030
+
return this.decodePush(buffer, view, decoder);
1031
+
case this.KINDS.reply:
1032
+
return this.decodeReply(buffer, view, decoder);
1033
+
case this.KINDS.broadcast:
1034
+
return this.decodeBroadcast(buffer, view, decoder);
1035
+
}
1036
+
},
1037
+
decodePush(buffer, view, decoder) {
1038
+
let joinRefSize = view.getUint8(1);
1039
+
let topicSize = view.getUint8(2);
1040
+
let eventSize = view.getUint8(3);
1041
+
let offset = this.HEADER_LENGTH + this.META_LENGTH - 1;
1042
+
let joinRef = decoder.decode(buffer.slice(offset, offset + joinRefSize));
1043
+
offset = offset + joinRefSize;
1044
+
let topic = decoder.decode(buffer.slice(offset, offset + topicSize));
1045
+
offset = offset + topicSize;
1046
+
let event = decoder.decode(buffer.slice(offset, offset + eventSize));
1047
+
offset = offset + eventSize;
1048
+
let data = buffer.slice(offset, buffer.byteLength);
1049
+
return { join_ref: joinRef, ref: null, topic, event, payload: data };
1050
+
},
1051
+
decodeReply(buffer, view, decoder) {
1052
+
let joinRefSize = view.getUint8(1);
1053
+
let refSize = view.getUint8(2);
1054
+
let topicSize = view.getUint8(3);
1055
+
let eventSize = view.getUint8(4);
1056
+
let offset = this.HEADER_LENGTH + this.META_LENGTH;
1057
+
let joinRef = decoder.decode(buffer.slice(offset, offset + joinRefSize));
1058
+
offset = offset + joinRefSize;
1059
+
let ref = decoder.decode(buffer.slice(offset, offset + refSize));
1060
+
offset = offset + refSize;
1061
+
let topic = decoder.decode(buffer.slice(offset, offset + topicSize));
1062
+
offset = offset + topicSize;
1063
+
let event = decoder.decode(buffer.slice(offset, offset + eventSize));
1064
+
offset = offset + eventSize;
1065
+
let data = buffer.slice(offset, buffer.byteLength);
1066
+
let payload = { status: event, response: data };
1067
+
return { join_ref: joinRef, ref, topic, event: CHANNEL_EVENTS.reply, payload };
1068
+
},
1069
+
decodeBroadcast(buffer, view, decoder) {
1070
+
let topicSize = view.getUint8(1);
1071
+
let eventSize = view.getUint8(2);
1072
+
let offset = this.HEADER_LENGTH + 2;
1073
+
let topic = decoder.decode(buffer.slice(offset, offset + topicSize));
1074
+
offset = offset + topicSize;
1075
+
let event = decoder.decode(buffer.slice(offset, offset + eventSize));
1076
+
offset = offset + eventSize;
1077
+
let data = buffer.slice(offset, buffer.byteLength);
1078
+
return { join_ref: null, ref: null, topic, event, payload: data };
1079
+
}
1080
+
};
1081
+
var Socket = class {
1082
+
constructor(endPoint, opts = {}) {
1083
+
this.stateChangeCallbacks = { open: [], close: [], error: [], message: [] };
1084
+
this.channels = [];
1085
+
this.sendBuffer = [];
1086
+
this.ref = 0;
1087
+
this.timeout = opts.timeout || DEFAULT_TIMEOUT;
1088
+
this.transport = opts.transport || global.WebSocket || LongPoll;
1089
+
this.primaryPassedHealthCheck = false;
1090
+
this.longPollFallbackMs = opts.longPollFallbackMs;
1091
+
this.fallbackTimer = null;
1092
+
this.sessionStore = opts.sessionStorage || global && global.sessionStorage;
1093
+
this.establishedConnections = 0;
1094
+
this.defaultEncoder = serializer_default.encode.bind(serializer_default);
1095
+
this.defaultDecoder = serializer_default.decode.bind(serializer_default);
1096
+
this.closeWasClean = false;
1097
+
this.disconnecting = false;
1098
+
this.binaryType = opts.binaryType || "arraybuffer";
1099
+
this.connectClock = 1;
1100
+
if (this.transport !== LongPoll) {
1101
+
this.encode = opts.encode || this.defaultEncoder;
1102
+
this.decode = opts.decode || this.defaultDecoder;
1103
+
} else {
1104
+
this.encode = this.defaultEncoder;
1105
+
this.decode = this.defaultDecoder;
1106
+
}
1107
+
let awaitingConnectionOnPageShow = null;
1108
+
if (phxWindow && phxWindow.addEventListener) {
1109
+
phxWindow.addEventListener("pagehide", (_e) => {
1110
+
if (this.conn) {
1111
+
this.disconnect();
1112
+
awaitingConnectionOnPageShow = this.connectClock;
1113
+
}
1114
+
});
1115
+
phxWindow.addEventListener("pageshow", (_e) => {
1116
+
if (awaitingConnectionOnPageShow === this.connectClock) {
1117
+
awaitingConnectionOnPageShow = null;
1118
+
this.connect();
1119
+
}
1120
+
});
1121
+
}
1122
+
this.heartbeatIntervalMs = opts.heartbeatIntervalMs || 3e4;
1123
+
this.rejoinAfterMs = (tries) => {
1124
+
if (opts.rejoinAfterMs) {
1125
+
return opts.rejoinAfterMs(tries);
1126
+
} else {
1127
+
return [1e3, 2e3, 5e3][tries - 1] || 1e4;
1128
+
}
1129
+
};
1130
+
this.reconnectAfterMs = (tries) => {
1131
+
if (opts.reconnectAfterMs) {
1132
+
return opts.reconnectAfterMs(tries);
1133
+
} else {
1134
+
return [10, 50, 100, 150, 200, 250, 500, 1e3, 2e3][tries - 1] || 5e3;
1135
+
}
1136
+
};
1137
+
this.logger = opts.logger || null;
1138
+
if (!this.logger && opts.debug) {
1139
+
this.logger = (kind, msg, data) => {
1140
+
console.log(`${kind}: ${msg}`, data);
1141
+
};
1142
+
}
1143
+
this.longpollerTimeout = opts.longpollerTimeout || 2e4;
1144
+
this.params = closure(opts.params || {});
1145
+
this.endPoint = `${endPoint}/${TRANSPORTS.websocket}`;
1146
+
this.vsn = opts.vsn || DEFAULT_VSN;
1147
+
this.heartbeatTimeoutTimer = null;
1148
+
this.heartbeatTimer = null;
1149
+
this.pendingHeartbeatRef = null;
1150
+
this.reconnectTimer = new Timer(() => {
1151
+
this.teardown(() => this.connect());
1152
+
}, this.reconnectAfterMs);
1153
+
}
1154
+
/**
1155
+
* Returns the LongPoll transport reference
1156
+
*/
1157
+
getLongPollTransport() {
1158
+
return LongPoll;
1159
+
}
1160
+
/**
1161
+
* Disconnects and replaces the active transport
1162
+
*
1163
+
* @param {Function} newTransport - The new transport class to instantiate
1164
+
*
1165
+
*/
1166
+
replaceTransport(newTransport) {
1167
+
this.connectClock++;
1168
+
this.closeWasClean = true;
1169
+
clearTimeout(this.fallbackTimer);
1170
+
this.reconnectTimer.reset();
1171
+
if (this.conn) {
1172
+
this.conn.close();
1173
+
this.conn = null;
1174
+
}
1175
+
this.transport = newTransport;
1176
+
}
1177
+
/**
1178
+
* Returns the socket protocol
1179
+
*
1180
+
* @returns {string}
1181
+
*/
1182
+
protocol() {
1183
+
return location.protocol.match(/^https/) ? "wss" : "ws";
1184
+
}
1185
+
/**
1186
+
* The fully qualified socket url
1187
+
*
1188
+
* @returns {string}
1189
+
*/
1190
+
endPointURL() {
1191
+
let uri = Ajax.appendParams(
1192
+
Ajax.appendParams(this.endPoint, this.params()),
1193
+
{ vsn: this.vsn }
1194
+
);
1195
+
if (uri.charAt(0) !== "/") {
1196
+
return uri;
1197
+
}
1198
+
if (uri.charAt(1) === "/") {
1199
+
return `${this.protocol()}:${uri}`;
1200
+
}
1201
+
return `${this.protocol()}://${location.host}${uri}`;
1202
+
}
1203
+
/**
1204
+
* Disconnects the socket
1205
+
*
1206
+
* See https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes for valid status codes.
1207
+
*
1208
+
* @param {Function} callback - Optional callback which is called after socket is disconnected.
1209
+
* @param {integer} code - A status code for disconnection (Optional).
1210
+
* @param {string} reason - A textual description of the reason to disconnect. (Optional)
1211
+
*/
1212
+
disconnect(callback, code, reason) {
1213
+
this.connectClock++;
1214
+
this.disconnecting = true;
1215
+
this.closeWasClean = true;
1216
+
clearTimeout(this.fallbackTimer);
1217
+
this.reconnectTimer.reset();
1218
+
this.teardown(() => {
1219
+
this.disconnecting = false;
1220
+
callback && callback();
1221
+
}, code, reason);
1222
+
}
1223
+
/**
1224
+
*
1225
+
* @param {Object} params - The params to send when connecting, for example `{user_id: userToken}`
1226
+
*
1227
+
* Passing params to connect is deprecated; pass them in the Socket constructor instead:
1228
+
* `new Socket("/socket", {params: {user_id: userToken}})`.
1229
+
*/
1230
+
connect(params) {
1231
+
if (params) {
1232
+
console && console.log("passing params to connect is deprecated. Instead pass :params to the Socket constructor");
1233
+
this.params = closure(params);
1234
+
}
1235
+
if (this.conn && !this.disconnecting) {
1236
+
return;
1237
+
}
1238
+
if (this.longPollFallbackMs && this.transport !== LongPoll) {
1239
+
this.connectWithFallback(LongPoll, this.longPollFallbackMs);
1240
+
} else {
1241
+
this.transportConnect();
1242
+
}
1243
+
}
1244
+
/**
1245
+
* Logs the message. Override `this.logger` for specialized logging. noops by default
1246
+
* @param {string} kind
1247
+
* @param {string} msg
1248
+
* @param {Object} data
1249
+
*/
1250
+
log(kind, msg, data) {
1251
+
this.logger && this.logger(kind, msg, data);
1252
+
}
1253
+
/**
1254
+
* Returns true if a logger has been set on this socket.
1255
+
*/
1256
+
hasLogger() {
1257
+
return this.logger !== null;
1258
+
}
1259
+
/**
1260
+
* Registers callbacks for connection open events
1261
+
*
1262
+
* @example socket.onOpen(function(){ console.info("the socket was opened") })
1263
+
*
1264
+
* @param {Function} callback
1265
+
*/
1266
+
onOpen(callback) {
1267
+
let ref = this.makeRef();
1268
+
this.stateChangeCallbacks.open.push([ref, callback]);
1269
+
return ref;
1270
+
}
1271
+
/**
1272
+
* Registers callbacks for connection close events
1273
+
* @param {Function} callback
1274
+
*/
1275
+
onClose(callback) {
1276
+
let ref = this.makeRef();
1277
+
this.stateChangeCallbacks.close.push([ref, callback]);
1278
+
return ref;
1279
+
}
1280
+
/**
1281
+
* Registers callbacks for connection error events
1282
+
*
1283
+
* @example socket.onError(function(error){ alert("An error occurred") })
1284
+
*
1285
+
* @param {Function} callback
1286
+
*/
1287
+
onError(callback) {
1288
+
let ref = this.makeRef();
1289
+
this.stateChangeCallbacks.error.push([ref, callback]);
1290
+
return ref;
1291
+
}
1292
+
/**
1293
+
* Registers callbacks for connection message events
1294
+
* @param {Function} callback
1295
+
*/
1296
+
onMessage(callback) {
1297
+
let ref = this.makeRef();
1298
+
this.stateChangeCallbacks.message.push([ref, callback]);
1299
+
return ref;
1300
+
}
1301
+
/**
1302
+
* Pings the server and invokes the callback with the RTT in milliseconds
1303
+
* @param {Function} callback
1304
+
*
1305
+
* Returns true if the ping was pushed or false if unable to be pushed.
1306
+
*/
1307
+
ping(callback) {
1308
+
if (!this.isConnected()) {
1309
+
return false;
1310
+
}
1311
+
let ref = this.makeRef();
1312
+
let startTime = Date.now();
1313
+
this.push({ topic: "phoenix", event: "heartbeat", payload: {}, ref });
1314
+
let onMsgRef = this.onMessage((msg) => {
1315
+
if (msg.ref === ref) {
1316
+
this.off([onMsgRef]);
1317
+
callback(Date.now() - startTime);
1318
+
}
1319
+
});
1320
+
return true;
1321
+
}
1322
+
/**
1323
+
* @private
1324
+
*/
1325
+
transportConnect() {
1326
+
this.connectClock++;
1327
+
this.closeWasClean = false;
1328
+
this.conn = new this.transport(this.endPointURL());
1329
+
this.conn.binaryType = this.binaryType;
1330
+
this.conn.timeout = this.longpollerTimeout;
1331
+
this.conn.onopen = () => this.onConnOpen();
1332
+
this.conn.onerror = (error) => this.onConnError(error);
1333
+
this.conn.onmessage = (event) => this.onConnMessage(event);
1334
+
this.conn.onclose = (event) => this.onConnClose(event);
1335
+
}
1336
+
getSession(key) {
1337
+
return this.sessionStore && this.sessionStore.getItem(key);
1338
+
}
1339
+
storeSession(key, val) {
1340
+
this.sessionStore && this.sessionStore.setItem(key, val);
1341
+
}
1342
+
connectWithFallback(fallbackTransport, fallbackThreshold = 2500) {
1343
+
clearTimeout(this.fallbackTimer);
1344
+
let established = false;
1345
+
let primaryTransport = true;
1346
+
let openRef, errorRef;
1347
+
let fallback = (reason) => {
1348
+
this.log("transport", `falling back to ${fallbackTransport.name}...`, reason);
1349
+
this.off([openRef, errorRef]);
1350
+
primaryTransport = false;
1351
+
this.replaceTransport(fallbackTransport);
1352
+
this.transportConnect();
1353
+
};
1354
+
if (this.getSession(`phx:fallback:${fallbackTransport.name}`)) {
1355
+
return fallback("memorized");
1356
+
}
1357
+
this.fallbackTimer = setTimeout(fallback, fallbackThreshold);
1358
+
errorRef = this.onError((reason) => {
1359
+
this.log("transport", "error", reason);
1360
+
if (primaryTransport && !established) {
1361
+
clearTimeout(this.fallbackTimer);
1362
+
fallback(reason);
1363
+
}
1364
+
});
1365
+
this.onOpen(() => {
1366
+
established = true;
1367
+
if (!primaryTransport) {
1368
+
if (!this.primaryPassedHealthCheck) {
1369
+
this.storeSession(`phx:fallback:${fallbackTransport.name}`, "true");
1370
+
}
1371
+
return this.log("transport", `established ${fallbackTransport.name} fallback`);
1372
+
}
1373
+
clearTimeout(this.fallbackTimer);
1374
+
this.fallbackTimer = setTimeout(fallback, fallbackThreshold);
1375
+
this.ping((rtt) => {
1376
+
this.log("transport", "connected to primary after", rtt);
1377
+
this.primaryPassedHealthCheck = true;
1378
+
clearTimeout(this.fallbackTimer);
1379
+
});
1380
+
});
1381
+
this.transportConnect();
1382
+
}
1383
+
clearHeartbeats() {
1384
+
clearTimeout(this.heartbeatTimer);
1385
+
clearTimeout(this.heartbeatTimeoutTimer);
1386
+
}
1387
+
onConnOpen() {
1388
+
if (this.hasLogger())
1389
+
this.log("transport", `${this.transport.name} connected to ${this.endPointURL()}`);
1390
+
this.closeWasClean = false;
1391
+
this.disconnecting = false;
1392
+
this.establishedConnections++;
1393
+
this.flushSendBuffer();
1394
+
this.reconnectTimer.reset();
1395
+
this.resetHeartbeat();
1396
+
this.stateChangeCallbacks.open.forEach(([, callback]) => callback());
1397
+
}
1398
+
/**
1399
+
* @private
1400
+
*/
1401
+
heartbeatTimeout() {
1402
+
if (this.pendingHeartbeatRef) {
1403
+
this.pendingHeartbeatRef = null;
1404
+
if (this.hasLogger()) {
1405
+
this.log("transport", "heartbeat timeout. Attempting to re-establish connection");
1406
+
}
1407
+
this.triggerChanError();
1408
+
this.closeWasClean = false;
1409
+
this.teardown(() => this.reconnectTimer.scheduleTimeout(), WS_CLOSE_NORMAL, "heartbeat timeout");
1410
+
}
1411
+
}
1412
+
resetHeartbeat() {
1413
+
if (this.conn && this.conn.skipHeartbeat) {
1414
+
return;
1415
+
}
1416
+
this.pendingHeartbeatRef = null;
1417
+
this.clearHeartbeats();
1418
+
this.heartbeatTimer = setTimeout(() => this.sendHeartbeat(), this.heartbeatIntervalMs);
1419
+
}
1420
+
teardown(callback, code, reason) {
1421
+
if (!this.conn) {
1422
+
return callback && callback();
1423
+
}
1424
+
let connectClock = this.connectClock;
1425
+
this.waitForBufferDone(() => {
1426
+
if (connectClock !== this.connectClock) {
1427
+
return;
1428
+
}
1429
+
if (this.conn) {
1430
+
if (code) {
1431
+
this.conn.close(code, reason || "");
1432
+
} else {
1433
+
this.conn.close();
1434
+
}
1435
+
}
1436
+
this.waitForSocketClosed(() => {
1437
+
if (connectClock !== this.connectClock) {
1438
+
return;
1439
+
}
1440
+
if (this.conn) {
1441
+
this.conn.onopen = function() {
1442
+
};
1443
+
this.conn.onerror = function() {
1444
+
};
1445
+
this.conn.onmessage = function() {
1446
+
};
1447
+
this.conn.onclose = function() {
1448
+
};
1449
+
this.conn = null;
1450
+
}
1451
+
callback && callback();
1452
+
});
1453
+
});
1454
+
}
1455
+
waitForBufferDone(callback, tries = 1) {
1456
+
if (tries === 5 || !this.conn || !this.conn.bufferedAmount) {
1457
+
callback();
1458
+
return;
1459
+
}
1460
+
setTimeout(() => {
1461
+
this.waitForBufferDone(callback, tries + 1);
1462
+
}, 150 * tries);
1463
+
}
1464
+
waitForSocketClosed(callback, tries = 1) {
1465
+
if (tries === 5 || !this.conn || this.conn.readyState === SOCKET_STATES.closed) {
1466
+
callback();
1467
+
return;
1468
+
}
1469
+
setTimeout(() => {
1470
+
this.waitForSocketClosed(callback, tries + 1);
1471
+
}, 150 * tries);
1472
+
}
1473
+
onConnClose(event) {
1474
+
let closeCode = event && event.code;
1475
+
if (this.hasLogger())
1476
+
this.log("transport", "close", event);
1477
+
this.triggerChanError();
1478
+
this.clearHeartbeats();
1479
+
if (!this.closeWasClean && closeCode !== 1e3) {
1480
+
this.reconnectTimer.scheduleTimeout();
1481
+
}
1482
+
this.stateChangeCallbacks.close.forEach(([, callback]) => callback(event));
1483
+
}
1484
+
/**
1485
+
* @private
1486
+
*/
1487
+
onConnError(error) {
1488
+
if (this.hasLogger())
1489
+
this.log("transport", error);
1490
+
let transportBefore = this.transport;
1491
+
let establishedBefore = this.establishedConnections;
1492
+
this.stateChangeCallbacks.error.forEach(([, callback]) => {
1493
+
callback(error, transportBefore, establishedBefore);
1494
+
});
1495
+
if (transportBefore === this.transport || establishedBefore > 0) {
1496
+
this.triggerChanError();
1497
+
}
1498
+
}
1499
+
/**
1500
+
* @private
1501
+
*/
1502
+
triggerChanError() {
1503
+
this.channels.forEach((channel) => {
1504
+
if (!(channel.isErrored() || channel.isLeaving() || channel.isClosed())) {
1505
+
channel.trigger(CHANNEL_EVENTS.error);
1506
+
}
1507
+
});
1508
+
}
1509
+
/**
1510
+
* @returns {string}
1511
+
*/
1512
+
connectionState() {
1513
+
switch (this.conn && this.conn.readyState) {
1514
+
case SOCKET_STATES.connecting:
1515
+
return "connecting";
1516
+
case SOCKET_STATES.open:
1517
+
return "open";
1518
+
case SOCKET_STATES.closing:
1519
+
return "closing";
1520
+
default:
1521
+
return "closed";
1522
+
}
1523
+
}
1524
+
/**
1525
+
* @returns {boolean}
1526
+
*/
1527
+
isConnected() {
1528
+
return this.connectionState() === "open";
1529
+
}
1530
+
/**
1531
+
* @private
1532
+
*
1533
+
* @param {Channel}
1534
+
*/
1535
+
remove(channel) {
1536
+
this.off(channel.stateChangeRefs);
1537
+
this.channels = this.channels.filter((c) => c !== channel);
1538
+
}
1539
+
/**
1540
+
* Removes `onOpen`, `onClose`, `onError,` and `onMessage` registrations.
1541
+
*
1542
+
* @param {refs} - list of refs returned by calls to
1543
+
* `onOpen`, `onClose`, `onError,` and `onMessage`
1544
+
*/
1545
+
off(refs) {
1546
+
for (let key in this.stateChangeCallbacks) {
1547
+
this.stateChangeCallbacks[key] = this.stateChangeCallbacks[key].filter(([ref]) => {
1548
+
return refs.indexOf(ref) === -1;
1549
+
});
1550
+
}
1551
+
}
1552
+
/**
1553
+
* Initiates a new channel for the given topic
1554
+
*
1555
+
* @param {string} topic
1556
+
* @param {Object} chanParams - Parameters for the channel
1557
+
* @returns {Channel}
1558
+
*/
1559
+
channel(topic, chanParams = {}) {
1560
+
let chan = new Channel(topic, chanParams, this);
1561
+
this.channels.push(chan);
1562
+
return chan;
1563
+
}
1564
+
/**
1565
+
* @param {Object} data
1566
+
*/
1567
+
push(data) {
1568
+
if (this.hasLogger()) {
1569
+
let { topic, event, payload, ref, join_ref } = data;
1570
+
this.log("push", `${topic} ${event} (${join_ref}, ${ref})`, payload);
1571
+
}
1572
+
if (this.isConnected()) {
1573
+
this.encode(data, (result) => this.conn.send(result));
1574
+
} else {
1575
+
this.sendBuffer.push(() => this.encode(data, (result) => this.conn.send(result)));
1576
+
}
1577
+
}
1578
+
/**
1579
+
* Return the next message ref, accounting for overflows
1580
+
* @returns {string}
1581
+
*/
1582
+
makeRef() {
1583
+
let newRef = this.ref + 1;
1584
+
if (newRef === this.ref) {
1585
+
this.ref = 0;
1586
+
} else {
1587
+
this.ref = newRef;
1588
+
}
1589
+
return this.ref.toString();
1590
+
}
1591
+
sendHeartbeat() {
1592
+
if (this.pendingHeartbeatRef && !this.isConnected()) {
1593
+
return;
1594
+
}
1595
+
this.pendingHeartbeatRef = this.makeRef();
1596
+
this.push({ topic: "phoenix", event: "heartbeat", payload: {}, ref: this.pendingHeartbeatRef });
1597
+
this.heartbeatTimeoutTimer = setTimeout(() => this.heartbeatTimeout(), this.heartbeatIntervalMs);
1598
+
}
1599
+
flushSendBuffer() {
1600
+
if (this.isConnected() && this.sendBuffer.length > 0) {
1601
+
this.sendBuffer.forEach((callback) => callback());
1602
+
this.sendBuffer = [];
1603
+
}
1604
+
}
1605
+
onConnMessage(rawMessage) {
1606
+
this.decode(rawMessage.data, (msg) => {
1607
+
let { topic, event, payload, ref, join_ref } = msg;
1608
+
if (ref && ref === this.pendingHeartbeatRef) {
1609
+
this.clearHeartbeats();
1610
+
this.pendingHeartbeatRef = null;
1611
+
this.heartbeatTimer = setTimeout(() => this.sendHeartbeat(), this.heartbeatIntervalMs);
1612
+
}
1613
+
if (this.hasLogger())
1614
+
this.log("receive", `${payload.status || ""} ${topic} ${event} ${ref && "(" + ref + ")" || ""}`, payload);
1615
+
for (let i = 0; i < this.channels.length; i++) {
1616
+
const channel = this.channels[i];
1617
+
if (!channel.isMember(topic, event, payload, join_ref)) {
1618
+
continue;
1619
+
}
1620
+
channel.trigger(event, payload, ref, join_ref);
1621
+
}
1622
+
for (let i = 0; i < this.stateChangeCallbacks.message.length; i++) {
1623
+
let [, callback] = this.stateChangeCallbacks.message[i];
1624
+
callback(msg);
1625
+
}
1626
+
});
1627
+
}
1628
+
leaveOpenTopic(topic) {
1629
+
let dupChannel = this.channels.find((c) => c.topic === topic && (c.isJoined() || c.isJoining()));
1630
+
if (dupChannel) {
1631
+
if (this.hasLogger())
1632
+
this.log("transport", `leaving duplicate topic "${topic}"`);
1633
+
dupChannel.leave();
1634
+
}
1635
+
}
1636
+
};
1637
+
1638
+
// ../deps/phoenix_live_view/priv/static/phoenix_live_view.esm.js
1639
+
var CONSECUTIVE_RELOADS = "consecutive-reloads";
1640
+
var MAX_RELOADS = 10;
1641
+
var RELOAD_JITTER_MIN = 5e3;
1642
+
var RELOAD_JITTER_MAX = 1e4;
1643
+
var FAILSAFE_JITTER = 3e4;
1644
+
var PHX_EVENT_CLASSES = [
1645
+
"phx-click-loading",
1646
+
"phx-change-loading",
1647
+
"phx-submit-loading",
1648
+
"phx-keydown-loading",
1649
+
"phx-keyup-loading",
1650
+
"phx-blur-loading",
1651
+
"phx-focus-loading",
1652
+
"phx-hook-loading"
1653
+
];
1654
+
var PHX_COMPONENT = "data-phx-component";
1655
+
var PHX_LIVE_LINK = "data-phx-link";
1656
+
var PHX_TRACK_STATIC = "track-static";
1657
+
var PHX_LINK_STATE = "data-phx-link-state";
1658
+
var PHX_REF_LOADING = "data-phx-ref-loading";
1659
+
var PHX_REF_SRC = "data-phx-ref-src";
1660
+
var PHX_REF_LOCK = "data-phx-ref-lock";
1661
+
var PHX_PENDING_REFS = "phx-pending-refs";
1662
+
var PHX_TRACK_UPLOADS = "track-uploads";
1663
+
var PHX_UPLOAD_REF = "data-phx-upload-ref";
1664
+
var PHX_PREFLIGHTED_REFS = "data-phx-preflighted-refs";
1665
+
var PHX_DONE_REFS = "data-phx-done-refs";
1666
+
var PHX_DROP_TARGET = "drop-target";
1667
+
var PHX_ACTIVE_ENTRY_REFS = "data-phx-active-refs";
1668
+
var PHX_LIVE_FILE_UPDATED = "phx:live-file:updated";
1669
+
var PHX_SKIP = "data-phx-skip";
1670
+
var PHX_MAGIC_ID = "data-phx-id";
1671
+
var PHX_PRUNE = "data-phx-prune";
1672
+
var PHX_CONNECTED_CLASS = "phx-connected";
1673
+
var PHX_LOADING_CLASS = "phx-loading";
1674
+
var PHX_ERROR_CLASS = "phx-error";
1675
+
var PHX_CLIENT_ERROR_CLASS = "phx-client-error";
1676
+
var PHX_SERVER_ERROR_CLASS = "phx-server-error";
1677
+
var PHX_PARENT_ID = "data-phx-parent-id";
1678
+
var PHX_MAIN = "data-phx-main";
1679
+
var PHX_ROOT_ID = "data-phx-root-id";
1680
+
var PHX_VIEWPORT_TOP = "viewport-top";
1681
+
var PHX_VIEWPORT_BOTTOM = "viewport-bottom";
1682
+
var PHX_TRIGGER_ACTION = "trigger-action";
1683
+
var PHX_HAS_FOCUSED = "phx-has-focused";
1684
+
var FOCUSABLE_INPUTS = ["text", "textarea", "number", "email", "password", "search", "tel", "url", "date", "time", "datetime-local", "color", "range"];
1685
+
var CHECKABLE_INPUTS = ["checkbox", "radio"];
1686
+
var PHX_HAS_SUBMITTED = "phx-has-submitted";
1687
+
var PHX_SESSION = "data-phx-session";
1688
+
var PHX_VIEW_SELECTOR = `[${PHX_SESSION}]`;
1689
+
var PHX_STICKY = "data-phx-sticky";
1690
+
var PHX_STATIC = "data-phx-static";
1691
+
var PHX_READONLY = "data-phx-readonly";
1692
+
var PHX_DISABLED = "data-phx-disabled";
1693
+
var PHX_DISABLE_WITH = "disable-with";
1694
+
var PHX_DISABLE_WITH_RESTORE = "data-phx-disable-with-restore";
1695
+
var PHX_HOOK = "hook";
1696
+
var PHX_DEBOUNCE = "debounce";
1697
+
var PHX_THROTTLE = "throttle";
1698
+
var PHX_UPDATE = "update";
1699
+
var PHX_STREAM = "stream";
1700
+
var PHX_STREAM_REF = "data-phx-stream";
1701
+
var PHX_KEY = "key";
1702
+
var PHX_PRIVATE = "phxPrivate";
1703
+
var PHX_AUTO_RECOVER = "auto-recover";
1704
+
var PHX_LV_DEBUG = "phx:live-socket:debug";
1705
+
var PHX_LV_PROFILE = "phx:live-socket:profiling";
1706
+
var PHX_LV_LATENCY_SIM = "phx:live-socket:latency-sim";
1707
+
var PHX_LV_HISTORY_POSITION = "phx:nav-history-position";
1708
+
var PHX_PROGRESS = "progress";
1709
+
var PHX_MOUNTED = "mounted";
1710
+
var PHX_RELOAD_STATUS = "__phoenix_reload_status__";
1711
+
var LOADER_TIMEOUT = 1;
1712
+
var MAX_CHILD_JOIN_ATTEMPTS = 3;
1713
+
var BEFORE_UNLOAD_LOADER_TIMEOUT = 200;
1714
+
var DISCONNECTED_TIMEOUT = 500;
1715
+
var BINDING_PREFIX = "phx-";
1716
+
var PUSH_TIMEOUT = 3e4;
1717
+
var DEBOUNCE_TRIGGER = "debounce-trigger";
1718
+
var THROTTLED = "throttled";
1719
+
var DEBOUNCE_PREV_KEY = "debounce-prev-key";
1720
+
var DEFAULTS = {
1721
+
debounce: 300,
1722
+
throttle: 300
1723
+
};
1724
+
var PHX_PENDING_ATTRS = [PHX_REF_LOADING, PHX_REF_SRC, PHX_REF_LOCK];
1725
+
var DYNAMICS = "d";
1726
+
var STATIC = "s";
1727
+
var ROOT = "r";
1728
+
var COMPONENTS = "c";
1729
+
var EVENTS = "e";
1730
+
var REPLY = "r";
1731
+
var TITLE = "t";
1732
+
var TEMPLATES = "p";
1733
+
var STREAM = "stream";
1734
+
var EntryUploader = class {
1735
+
constructor(entry, config, liveSocket2) {
1736
+
let { chunk_size, chunk_timeout } = config;
1737
+
this.liveSocket = liveSocket2;
1738
+
this.entry = entry;
1739
+
this.offset = 0;
1740
+
this.chunkSize = chunk_size;
1741
+
this.chunkTimeout = chunk_timeout;
1742
+
this.chunkTimer = null;
1743
+
this.errored = false;
1744
+
this.uploadChannel = liveSocket2.channel(`lvu:${entry.ref}`, { token: entry.metadata() });
1745
+
}
1746
+
error(reason) {
1747
+
if (this.errored) {
1748
+
return;
1749
+
}
1750
+
this.uploadChannel.leave();
1751
+
this.errored = true;
1752
+
clearTimeout(this.chunkTimer);
1753
+
this.entry.error(reason);
1754
+
}
1755
+
upload() {
1756
+
this.uploadChannel.onError((reason) => this.error(reason));
1757
+
this.uploadChannel.join().receive("ok", (_data) => this.readNextChunk()).receive("error", (reason) => this.error(reason));
1758
+
}
1759
+
isDone() {
1760
+
return this.offset >= this.entry.file.size;
1761
+
}
1762
+
readNextChunk() {
1763
+
let reader = new window.FileReader();
1764
+
let blob = this.entry.file.slice(this.offset, this.chunkSize + this.offset);
1765
+
reader.onload = (e) => {
1766
+
if (e.target.error === null) {
1767
+
this.offset += e.target.result.byteLength;
1768
+
this.pushChunk(e.target.result);
1769
+
} else {
1770
+
return logError("Read error: " + e.target.error);
1771
+
}
1772
+
};
1773
+
reader.readAsArrayBuffer(blob);
1774
+
}
1775
+
pushChunk(chunk) {
1776
+
if (!this.uploadChannel.isJoined()) {
1777
+
return;
1778
+
}
1779
+
this.uploadChannel.push("chunk", chunk, this.chunkTimeout).receive("ok", () => {
1780
+
this.entry.progress(this.offset / this.entry.file.size * 100);
1781
+
if (!this.isDone()) {
1782
+
this.chunkTimer = setTimeout(() => this.readNextChunk(), this.liveSocket.getLatencySim() || 0);
1783
+
}
1784
+
}).receive("error", ({ reason }) => this.error(reason));
1785
+
}
1786
+
};
1787
+
var logError = (msg, obj) => console.error && console.error(msg, obj);
1788
+
var isCid = (cid) => {
1789
+
let type = typeof cid;
1790
+
return type === "number" || type === "string" && /^(0|[1-9]\d*)$/.test(cid);
1791
+
};
1792
+
function detectDuplicateIds() {
1793
+
let ids = /* @__PURE__ */ new Set();
1794
+
let elems = document.querySelectorAll("*[id]");
1795
+
for (let i = 0, len = elems.length; i < len; i++) {
1796
+
if (ids.has(elems[i].id)) {
1797
+
console.error(`Multiple IDs detected: ${elems[i].id}. Ensure unique element ids.`);
1798
+
} else {
1799
+
ids.add(elems[i].id);
1800
+
}
1801
+
}
1802
+
}
1803
+
function detectInvalidStreamInserts(inserts) {
1804
+
const errors = /* @__PURE__ */ new Set();
1805
+
Object.keys(inserts).forEach((id) => {
1806
+
const streamEl = document.getElementById(id);
1807
+
if (streamEl && streamEl.parentElement && streamEl.parentElement.getAttribute("phx-update") !== "stream") {
1808
+
errors.add(`The stream container with id "${streamEl.parentElement.id}" is missing the phx-update="stream" attribute. Ensure it is set for streams to work properly.`);
1809
+
}
1810
+
});
1811
+
errors.forEach((error) => console.error(error));
1812
+
}
1813
+
var debug = (view, kind, msg, obj) => {
1814
+
if (view.liveSocket.isDebugEnabled()) {
1815
+
console.log(`${view.id} ${kind}: ${msg} - `, obj);
1816
+
}
1817
+
};
1818
+
var closure2 = (val) => typeof val === "function" ? val : function() {
1819
+
return val;
1820
+
};
1821
+
var clone = (obj) => {
1822
+
return JSON.parse(JSON.stringify(obj));
1823
+
};
1824
+
var closestPhxBinding = (el, binding, borderEl) => {
1825
+
do {
1826
+
if (el.matches(`[${binding}]`) && !el.disabled) {
1827
+
return el;
1828
+
}
1829
+
el = el.parentElement || el.parentNode;
1830
+
} while (el !== null && el.nodeType === 1 && !(borderEl && borderEl.isSameNode(el) || el.matches(PHX_VIEW_SELECTOR)));
1831
+
return null;
1832
+
};
1833
+
var isObject = (obj) => {
1834
+
return obj !== null && typeof obj === "object" && !(obj instanceof Array);
1835
+
};
1836
+
var isEqualObj = (obj1, obj2) => JSON.stringify(obj1) === JSON.stringify(obj2);
1837
+
var isEmpty = (obj) => {
1838
+
for (let x in obj) {
1839
+
return false;
1840
+
}
1841
+
return true;
1842
+
};
1843
+
var maybe = (el, callback) => el && callback(el);
1844
+
var channelUploader = function(entries, onError, resp, liveSocket2) {
1845
+
entries.forEach((entry) => {
1846
+
let entryUploader = new EntryUploader(entry, resp.config, liveSocket2);
1847
+
entryUploader.upload();
1848
+
});
1849
+
};
1850
+
var Browser = {
1851
+
canPushState() {
1852
+
return typeof history.pushState !== "undefined";
1853
+
},
1854
+
dropLocal(localStorage, namespace, subkey) {
1855
+
return localStorage.removeItem(this.localKey(namespace, subkey));
1856
+
},
1857
+
updateLocal(localStorage, namespace, subkey, initial, func) {
1858
+
let current = this.getLocal(localStorage, namespace, subkey);
1859
+
let key = this.localKey(namespace, subkey);
1860
+
let newVal = current === null ? initial : func(current);
1861
+
localStorage.setItem(key, JSON.stringify(newVal));
1862
+
return newVal;
1863
+
},
1864
+
getLocal(localStorage, namespace, subkey) {
1865
+
return JSON.parse(localStorage.getItem(this.localKey(namespace, subkey)));
1866
+
},
1867
+
updateCurrentState(callback) {
1868
+
if (!this.canPushState()) {
1869
+
return;
1870
+
}
1871
+
history.replaceState(callback(history.state || {}), "", window.location.href);
1872
+
},
1873
+
pushState(kind, meta, to) {
1874
+
if (this.canPushState()) {
1875
+
if (to !== window.location.href) {
1876
+
if (meta.type == "redirect" && meta.scroll) {
1877
+
let currentState = history.state || {};
1878
+
currentState.scroll = meta.scroll;
1879
+
history.replaceState(currentState, "", window.location.href);
1880
+
}
1881
+
delete meta.scroll;
1882
+
history[kind + "State"](meta, "", to || null);
1883
+
window.requestAnimationFrame(() => {
1884
+
let hashEl = this.getHashTargetEl(window.location.hash);
1885
+
if (hashEl) {
1886
+
hashEl.scrollIntoView();
1887
+
} else if (meta.type === "redirect") {
1888
+
window.scroll(0, 0);
1889
+
}
1890
+
});
1891
+
}
1892
+
} else {
1893
+
this.redirect(to);
1894
+
}
1895
+
},
1896
+
setCookie(name, value, maxAgeSeconds) {
1897
+
let expires = typeof maxAgeSeconds === "number" ? ` max-age=${maxAgeSeconds};` : "";
1898
+
document.cookie = `${name}=${value};${expires} path=/`;
1899
+
},
1900
+
getCookie(name) {
1901
+
return document.cookie.replace(new RegExp(`(?:(?:^|.*;s*)${name}s*=s*([^;]*).*$)|^.*$`), "$1");
1902
+
},
1903
+
deleteCookie(name) {
1904
+
document.cookie = `${name}=; max-age=-1; path=/`;
1905
+
},
1906
+
redirect(toURL, flash) {
1907
+
if (flash) {
1908
+
this.setCookie("__phoenix_flash__", flash, 60);
1909
+
}
1910
+
window.location = toURL;
1911
+
},
1912
+
localKey(namespace, subkey) {
1913
+
return `${namespace}-${subkey}`;
1914
+
},
1915
+
getHashTargetEl(maybeHash) {
1916
+
let hash = maybeHash.toString().substring(1);
1917
+
if (hash === "") {
1918
+
return;
1919
+
}
1920
+
return document.getElementById(hash) || document.querySelector(`a[name="${hash}"]`);
1921
+
}
1922
+
};
1923
+
var browser_default = Browser;
1924
+
var DOM = {
1925
+
byId(id) {
1926
+
return document.getElementById(id) || logError(`no id found for ${id}`);
1927
+
},
1928
+
removeClass(el, className) {
1929
+
el.classList.remove(className);
1930
+
if (el.classList.length === 0) {
1931
+
el.removeAttribute("class");
1932
+
}
1933
+
},
1934
+
all(node, query, callback) {
1935
+
if (!node) {
1936
+
return [];
1937
+
}
1938
+
let array = Array.from(node.querySelectorAll(query));
1939
+
return callback ? array.forEach(callback) : array;
1940
+
},
1941
+
childNodeLength(html) {
1942
+
let template = document.createElement("template");
1943
+
template.innerHTML = html;
1944
+
return template.content.childElementCount;
1945
+
},
1946
+
isUploadInput(el) {
1947
+
return el.type === "file" && el.getAttribute(PHX_UPLOAD_REF) !== null;
1948
+
},
1949
+
isAutoUpload(inputEl) {
1950
+
return inputEl.hasAttribute("data-phx-auto-upload");
1951
+
},
1952
+
findUploadInputs(node) {
1953
+
const formId = node.id;
1954
+
const inputsOutsideForm = this.all(document, `input[type="file"][${PHX_UPLOAD_REF}][form="${formId}"]`);
1955
+
return this.all(node, `input[type="file"][${PHX_UPLOAD_REF}]`).concat(inputsOutsideForm);
1956
+
},
1957
+
findComponentNodeList(node, cid) {
1958
+
return this.filterWithinSameLiveView(this.all(node, `[${PHX_COMPONENT}="${cid}"]`), node);
1959
+
},
1960
+
isPhxDestroyed(node) {
1961
+
return node.id && DOM.private(node, "destroyed") ? true : false;
1962
+
},
1963
+
wantsNewTab(e) {
1964
+
let wantsNewTab = e.ctrlKey || e.shiftKey || e.metaKey || e.button && e.button === 1;
1965
+
let isDownload = e.target instanceof HTMLAnchorElement && e.target.hasAttribute("download");
1966
+
let isTargetBlank = e.target.hasAttribute("target") && e.target.getAttribute("target").toLowerCase() === "_blank";
1967
+
let isTargetNamedTab = e.target.hasAttribute("target") && !e.target.getAttribute("target").startsWith("_");
1968
+
return wantsNewTab || isTargetBlank || isDownload || isTargetNamedTab;
1969
+
},
1970
+
isUnloadableFormSubmit(e) {
1971
+
let isDialogSubmit = e.target && e.target.getAttribute("method") === "dialog" || e.submitter && e.submitter.getAttribute("formmethod") === "dialog";
1972
+
if (isDialogSubmit) {
1973
+
return false;
1974
+
} else {
1975
+
return !e.defaultPrevented && !this.wantsNewTab(e);
1976
+
}
1977
+
},
1978
+
isNewPageClick(e, currentLocation) {
1979
+
let href = e.target instanceof HTMLAnchorElement ? e.target.getAttribute("href") : null;
1980
+
let url;
1981
+
if (e.defaultPrevented || href === null || this.wantsNewTab(e)) {
1982
+
return false;
1983
+
}
1984
+
if (href.startsWith("mailto:") || href.startsWith("tel:")) {
1985
+
return false;
1986
+
}
1987
+
if (e.target.isContentEditable) {
1988
+
return false;
1989
+
}
1990
+
try {
1991
+
url = new URL(href);
1992
+
} catch (e2) {
1993
+
try {
1994
+
url = new URL(href, currentLocation);
1995
+
} catch (e3) {
1996
+
return true;
1997
+
}
1998
+
}
1999
+
if (url.host === currentLocation.host && url.protocol === currentLocation.protocol) {
2000
+
if (url.pathname === currentLocation.pathname && url.search === currentLocation.search) {
2001
+
return url.hash === "" && !url.href.endsWith("#");
2002
+
}
2003
+
}
2004
+
return url.protocol.startsWith("http");
2005
+
},
2006
+
markPhxChildDestroyed(el) {
2007
+
if (this.isPhxChild(el)) {
2008
+
el.setAttribute(PHX_SESSION, "");
2009
+
}
2010
+
this.putPrivate(el, "destroyed", true);
2011
+
},
2012
+
findPhxChildrenInFragment(html, parentId) {
2013
+
let template = document.createElement("template");
2014
+
template.innerHTML = html;
2015
+
return this.findPhxChildren(template.content, parentId);
2016
+
},
2017
+
isIgnored(el, phxUpdate) {
2018
+
return (el.getAttribute(phxUpdate) || el.getAttribute("data-phx-update")) === "ignore";
2019
+
},
2020
+
isPhxUpdate(el, phxUpdate, updateTypes) {
2021
+
return el.getAttribute && updateTypes.indexOf(el.getAttribute(phxUpdate)) >= 0;
2022
+
},
2023
+
findPhxSticky(el) {
2024
+
return this.all(el, `[${PHX_STICKY}]`);
2025
+
},
2026
+
findPhxChildren(el, parentId) {
2027
+
return this.all(el, `${PHX_VIEW_SELECTOR}[${PHX_PARENT_ID}="${parentId}"]`);
2028
+
},
2029
+
findExistingParentCIDs(node, cids) {
2030
+
let parentCids = /* @__PURE__ */ new Set();
2031
+
let childrenCids = /* @__PURE__ */ new Set();
2032
+
cids.forEach((cid) => {
2033
+
this.filterWithinSameLiveView(this.all(node, `[${PHX_COMPONENT}="${cid}"]`), node).forEach((parent) => {
2034
+
parentCids.add(cid);
2035
+
this.filterWithinSameLiveView(this.all(parent, `[${PHX_COMPONENT}]`), parent).map((el) => parseInt(el.getAttribute(PHX_COMPONENT))).forEach((childCID) => childrenCids.add(childCID));
2036
+
});
2037
+
});
2038
+
childrenCids.forEach((childCid) => parentCids.delete(childCid));
2039
+
return parentCids;
2040
+
},
2041
+
filterWithinSameLiveView(nodes, parent) {
2042
+
if (parent.querySelector(PHX_VIEW_SELECTOR)) {
2043
+
return nodes.filter((el) => this.withinSameLiveView(el, parent));
2044
+
} else {
2045
+
return nodes;
2046
+
}
2047
+
},
2048
+
withinSameLiveView(node, parent) {
2049
+
while (node = node.parentNode) {
2050
+
if (node.isSameNode(parent)) {
2051
+
return true;
2052
+
}
2053
+
if (node.getAttribute(PHX_SESSION) !== null) {
2054
+
return false;
2055
+
}
2056
+
}
2057
+
},
2058
+
private(el, key) {
2059
+
return el[PHX_PRIVATE] && el[PHX_PRIVATE][key];
2060
+
},
2061
+
deletePrivate(el, key) {
2062
+
el[PHX_PRIVATE] && delete el[PHX_PRIVATE][key];
2063
+
},
2064
+
putPrivate(el, key, value) {
2065
+
if (!el[PHX_PRIVATE]) {
2066
+
el[PHX_PRIVATE] = {};
2067
+
}
2068
+
el[PHX_PRIVATE][key] = value;
2069
+
},
2070
+
updatePrivate(el, key, defaultVal, updateFunc) {
2071
+
let existing = this.private(el, key);
2072
+
if (existing === void 0) {
2073
+
this.putPrivate(el, key, updateFunc(defaultVal));
2074
+
} else {
2075
+
this.putPrivate(el, key, updateFunc(existing));
2076
+
}
2077
+
},
2078
+
syncPendingAttrs(fromEl, toEl) {
2079
+
if (!fromEl.hasAttribute(PHX_REF_SRC)) {
2080
+
return;
2081
+
}
2082
+
PHX_EVENT_CLASSES.forEach((className) => {
2083
+
fromEl.classList.contains(className) && toEl.classList.add(className);
2084
+
});
2085
+
PHX_PENDING_ATTRS.filter((attr) => fromEl.hasAttribute(attr)).forEach((attr) => {
2086
+
toEl.setAttribute(attr, fromEl.getAttribute(attr));
2087
+
});
2088
+
},
2089
+
copyPrivates(target, source) {
2090
+
if (source[PHX_PRIVATE]) {
2091
+
target[PHX_PRIVATE] = source[PHX_PRIVATE];
2092
+
}
2093
+
},
2094
+
putTitle(str) {
2095
+
let titleEl = document.querySelector("title");
2096
+
if (titleEl) {
2097
+
let { prefix, suffix, default: defaultTitle } = titleEl.dataset;
2098
+
let isEmpty2 = typeof str !== "string" || str.trim() === "";
2099
+
if (isEmpty2 && typeof defaultTitle !== "string") {
2100
+
return;
2101
+
}
2102
+
let inner = isEmpty2 ? defaultTitle : str;
2103
+
document.title = `${prefix || ""}${inner || ""}${suffix || ""}`;
2104
+
} else {
2105
+
document.title = str;
2106
+
}
2107
+
},
2108
+
debounce(el, event, phxDebounce, defaultDebounce, phxThrottle, defaultThrottle, asyncFilter, callback) {
2109
+
let debounce = el.getAttribute(phxDebounce);
2110
+
let throttle = el.getAttribute(phxThrottle);
2111
+
if (debounce === "") {
2112
+
debounce = defaultDebounce;
2113
+
}
2114
+
if (throttle === "") {
2115
+
throttle = defaultThrottle;
2116
+
}
2117
+
let value = debounce || throttle;
2118
+
switch (value) {
2119
+
case null:
2120
+
return callback();
2121
+
case "blur":
2122
+
this.incCycle(el, "debounce-blur-cycle", () => {
2123
+
if (asyncFilter()) {
2124
+
callback();
2125
+
}
2126
+
});
2127
+
if (this.once(el, "debounce-blur")) {
2128
+
el.addEventListener("blur", () => this.triggerCycle(el, "debounce-blur-cycle"));
2129
+
}
2130
+
return;
2131
+
default:
2132
+
let timeout = parseInt(value);
2133
+
let trigger = () => throttle ? this.deletePrivate(el, THROTTLED) : callback();
2134
+
let currentCycle = this.incCycle(el, DEBOUNCE_TRIGGER, trigger);
2135
+
if (isNaN(timeout)) {
2136
+
return logError(`invalid throttle/debounce value: ${value}`);
2137
+
}
2138
+
if (throttle) {
2139
+
let newKeyDown = false;
2140
+
if (event.type === "keydown") {
2141
+
let prevKey = this.private(el, DEBOUNCE_PREV_KEY);
2142
+
this.putPrivate(el, DEBOUNCE_PREV_KEY, event.key);
2143
+
newKeyDown = prevKey !== event.key;
2144
+
}
2145
+
if (!newKeyDown && this.private(el, THROTTLED)) {
2146
+
return false;
2147
+
} else {
2148
+
callback();
2149
+
const t = setTimeout(() => {
2150
+
if (asyncFilter()) {
2151
+
this.triggerCycle(el, DEBOUNCE_TRIGGER);
2152
+
}
2153
+
}, timeout);
2154
+
this.putPrivate(el, THROTTLED, t);
2155
+
}
2156
+
} else {
2157
+
setTimeout(() => {
2158
+
if (asyncFilter()) {
2159
+
this.triggerCycle(el, DEBOUNCE_TRIGGER, currentCycle);
2160
+
}
2161
+
}, timeout);
2162
+
}
2163
+
let form = el.form;
2164
+
if (form && this.once(form, "bind-debounce")) {
2165
+
form.addEventListener("submit", () => {
2166
+
Array.from(new FormData(form).entries(), ([name]) => {
2167
+
let input = form.querySelector(`[name="${name}"]`);
2168
+
this.incCycle(input, DEBOUNCE_TRIGGER);
2169
+
this.deletePrivate(input, THROTTLED);
2170
+
});
2171
+
});
2172
+
}
2173
+
if (this.once(el, "bind-debounce")) {
2174
+
el.addEventListener("blur", () => {
2175
+
clearTimeout(this.private(el, THROTTLED));
2176
+
this.triggerCycle(el, DEBOUNCE_TRIGGER);
2177
+
});
2178
+
}
2179
+
}
2180
+
},
2181
+
triggerCycle(el, key, currentCycle) {
2182
+
let [cycle, trigger] = this.private(el, key);
2183
+
if (!currentCycle) {
2184
+
currentCycle = cycle;
2185
+
}
2186
+
if (currentCycle === cycle) {
2187
+
this.incCycle(el, key);
2188
+
trigger();
2189
+
}
2190
+
},
2191
+
once(el, key) {
2192
+
if (this.private(el, key) === true) {
2193
+
return false;
2194
+
}
2195
+
this.putPrivate(el, key, true);
2196
+
return true;
2197
+
},
2198
+
incCycle(el, key, trigger = function() {
2199
+
}) {
2200
+
let [currentCycle] = this.private(el, key) || [0, trigger];
2201
+
currentCycle++;
2202
+
this.putPrivate(el, key, [currentCycle, trigger]);
2203
+
return currentCycle;
2204
+
},
2205
+
// maintains or adds privately used hook information
2206
+
// fromEl and toEl can be the same element in the case of a newly added node
2207
+
// fromEl and toEl can be any HTML node type, so we need to check if it's an element node
2208
+
maintainPrivateHooks(fromEl, toEl, phxViewportTop, phxViewportBottom) {
2209
+
if (fromEl.hasAttribute && fromEl.hasAttribute("data-phx-hook") && !toEl.hasAttribute("data-phx-hook")) {
2210
+
toEl.setAttribute("data-phx-hook", fromEl.getAttribute("data-phx-hook"));
2211
+
}
2212
+
if (toEl.hasAttribute && (toEl.hasAttribute(phxViewportTop) || toEl.hasAttribute(phxViewportBottom))) {
2213
+
toEl.setAttribute("data-phx-hook", "Phoenix.InfiniteScroll");
2214
+
}
2215
+
},
2216
+
putCustomElHook(el, hook) {
2217
+
if (el.isConnected) {
2218
+
el.setAttribute("data-phx-hook", "");
2219
+
} else {
2220
+
console.error(`
2221
+
hook attached to non-connected DOM element
2222
+
ensure you are calling createHook within your connectedCallback. ${el.outerHTML}
2223
+
`);
2224
+
}
2225
+
this.putPrivate(el, "custom-el-hook", hook);
2226
+
},
2227
+
getCustomElHook(el) {
2228
+
return this.private(el, "custom-el-hook");
2229
+
},
2230
+
isUsedInput(el) {
2231
+
return el.nodeType === Node.ELEMENT_NODE && (this.private(el, PHX_HAS_FOCUSED) || this.private(el, PHX_HAS_SUBMITTED));
2232
+
},
2233
+
resetForm(form) {
2234
+
Array.from(form.elements).forEach((input) => {
2235
+
this.deletePrivate(input, PHX_HAS_FOCUSED);
2236
+
this.deletePrivate(input, PHX_HAS_SUBMITTED);
2237
+
});
2238
+
},
2239
+
isPhxChild(node) {
2240
+
return node.getAttribute && node.getAttribute(PHX_PARENT_ID);
2241
+
},
2242
+
isPhxSticky(node) {
2243
+
return node.getAttribute && node.getAttribute(PHX_STICKY) !== null;
2244
+
},
2245
+
isChildOfAny(el, parents) {
2246
+
return !!parents.find((parent) => parent.contains(el));
2247
+
},
2248
+
firstPhxChild(el) {
2249
+
return this.isPhxChild(el) ? el : this.all(el, `[${PHX_PARENT_ID}]`)[0];
2250
+
},
2251
+
dispatchEvent(target, name, opts = {}) {
2252
+
let defaultBubble = true;
2253
+
let isUploadTarget = target.nodeName === "INPUT" && target.type === "file";
2254
+
if (isUploadTarget && name === "click") {
2255
+
defaultBubble = false;
2256
+
}
2257
+
let bubbles = opts.bubbles === void 0 ? defaultBubble : !!opts.bubbles;
2258
+
let eventOpts = { bubbles, cancelable: true, detail: opts.detail || {} };
2259
+
let event = name === "click" ? new MouseEvent("click", eventOpts) : new CustomEvent(name, eventOpts);
2260
+
target.dispatchEvent(event);
2261
+
},
2262
+
cloneNode(node, html) {
2263
+
if (typeof html === "undefined") {
2264
+
return node.cloneNode(true);
2265
+
} else {
2266
+
let cloned = node.cloneNode(false);
2267
+
cloned.innerHTML = html;
2268
+
return cloned;
2269
+
}
2270
+
},
2271
+
// merge attributes from source to target
2272
+
// if an element is ignored, we only merge data attributes
2273
+
// including removing data attributes that are no longer in the source
2274
+
mergeAttrs(target, source, opts = {}) {
2275
+
let exclude = new Set(opts.exclude || []);
2276
+
let isIgnored = opts.isIgnored;
2277
+
let sourceAttrs = source.attributes;
2278
+
for (let i = sourceAttrs.length - 1; i >= 0; i--) {
2279
+
let name = sourceAttrs[i].name;
2280
+
if (!exclude.has(name)) {
2281
+
const sourceValue = source.getAttribute(name);
2282
+
if (target.getAttribute(name) !== sourceValue && (!isIgnored || isIgnored && name.startsWith("data-"))) {
2283
+
target.setAttribute(name, sourceValue);
2284
+
}
2285
+
} else {
2286
+
if (name === "value" && target.value === source.value) {
2287
+
target.setAttribute("value", source.getAttribute(name));
2288
+
}
2289
+
}
2290
+
}
2291
+
let targetAttrs = target.attributes;
2292
+
for (let i = targetAttrs.length - 1; i >= 0; i--) {
2293
+
let name = targetAttrs[i].name;
2294
+
if (isIgnored) {
2295
+
if (name.startsWith("data-") && !source.hasAttribute(name) && !PHX_PENDING_ATTRS.includes(name)) {
2296
+
target.removeAttribute(name);
2297
+
}
2298
+
} else {
2299
+
if (!source.hasAttribute(name)) {
2300
+
target.removeAttribute(name);
2301
+
}
2302
+
}
2303
+
}
2304
+
},
2305
+
mergeFocusedInput(target, source) {
2306
+
if (!(target instanceof HTMLSelectElement)) {
2307
+
DOM.mergeAttrs(target, source, { exclude: ["value"] });
2308
+
}
2309
+
if (source.readOnly) {
2310
+
target.setAttribute("readonly", true);
2311
+
} else {
2312
+
target.removeAttribute("readonly");
2313
+
}
2314
+
},
2315
+
hasSelectionRange(el) {
2316
+
return el.setSelectionRange && (el.type === "text" || el.type === "textarea");
2317
+
},
2318
+
restoreFocus(focused, selectionStart, selectionEnd) {
2319
+
if (focused instanceof HTMLSelectElement) {
2320
+
focused.focus();
2321
+
}
2322
+
if (!DOM.isTextualInput(focused)) {
2323
+
return;
2324
+
}
2325
+
let wasFocused = focused.matches(":focus");
2326
+
if (!wasFocused) {
2327
+
focused.focus();
2328
+
}
2329
+
if (this.hasSelectionRange(focused)) {
2330
+
focused.setSelectionRange(selectionStart, selectionEnd);
2331
+
}
2332
+
},
2333
+
isFormInput(el) {
2334
+
return /^(?:input|select|textarea)$/i.test(el.tagName) && el.type !== "button";
2335
+
},
2336
+
syncAttrsToProps(el) {
2337
+
if (el instanceof HTMLInputElement && CHECKABLE_INPUTS.indexOf(el.type.toLocaleLowerCase()) >= 0) {
2338
+
el.checked = el.getAttribute("checked") !== null;
2339
+
}
2340
+
},
2341
+
isTextualInput(el) {
2342
+
return FOCUSABLE_INPUTS.indexOf(el.type) >= 0;
2343
+
},
2344
+
isNowTriggerFormExternal(el, phxTriggerExternal) {
2345
+
return el.getAttribute && el.getAttribute(phxTriggerExternal) !== null && document.body.contains(el);
2346
+
},
2347
+
cleanChildNodes(container, phxUpdate) {
2348
+
if (DOM.isPhxUpdate(container, phxUpdate, ["append", "prepend", PHX_STREAM])) {
2349
+
let toRemove = [];
2350
+
container.childNodes.forEach((childNode) => {
2351
+
if (!childNode.id) {
2352
+
let isEmptyTextNode = childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue.trim() === "";
2353
+
if (!isEmptyTextNode && childNode.nodeType !== Node.COMMENT_NODE) {
2354
+
logError(`only HTML element tags with an id are allowed inside containers with phx-update.
2355
+
2356
+
removing illegal node: "${(childNode.outerHTML || childNode.nodeValue).trim()}"
2357
+
2358
+
`);
2359
+
}
2360
+
toRemove.push(childNode);
2361
+
}
2362
+
});
2363
+
toRemove.forEach((childNode) => childNode.remove());
2364
+
}
2365
+
},
2366
+
replaceRootContainer(container, tagName, attrs) {
2367
+
let retainedAttrs = /* @__PURE__ */ new Set(["id", PHX_SESSION, PHX_STATIC, PHX_MAIN, PHX_ROOT_ID]);
2368
+
if (container.tagName.toLowerCase() === tagName.toLowerCase()) {
2369
+
Array.from(container.attributes).filter((attr) => !retainedAttrs.has(attr.name.toLowerCase())).forEach((attr) => container.removeAttribute(attr.name));
2370
+
Object.keys(attrs).filter((name) => !retainedAttrs.has(name.toLowerCase())).forEach((attr) => container.setAttribute(attr, attrs[attr]));
2371
+
return container;
2372
+
} else {
2373
+
let newContainer = document.createElement(tagName);
2374
+
Object.keys(attrs).forEach((attr) => newContainer.setAttribute(attr, attrs[attr]));
2375
+
retainedAttrs.forEach((attr) => newContainer.setAttribute(attr, container.getAttribute(attr)));
2376
+
newContainer.innerHTML = container.innerHTML;
2377
+
container.replaceWith(newContainer);
2378
+
return newContainer;
2379
+
}
2380
+
},
2381
+
getSticky(el, name, defaultVal) {
2382
+
let op = (DOM.private(el, "sticky") || []).find(([existingName]) => name === existingName);
2383
+
if (op) {
2384
+
let [_name, _op, stashedResult] = op;
2385
+
return stashedResult;
2386
+
} else {
2387
+
return typeof defaultVal === "function" ? defaultVal() : defaultVal;
2388
+
}
2389
+
},
2390
+
deleteSticky(el, name) {
2391
+
this.updatePrivate(el, "sticky", [], (ops) => {
2392
+
return ops.filter(([existingName, _]) => existingName !== name);
2393
+
});
2394
+
},
2395
+
putSticky(el, name, op) {
2396
+
let stashedResult = op(el);
2397
+
this.updatePrivate(el, "sticky", [], (ops) => {
2398
+
let existingIndex = ops.findIndex(([existingName]) => name === existingName);
2399
+
if (existingIndex >= 0) {
2400
+
ops[existingIndex] = [name, op, stashedResult];
2401
+
} else {
2402
+
ops.push([name, op, stashedResult]);
2403
+
}
2404
+
return ops;
2405
+
});
2406
+
},
2407
+
applyStickyOperations(el) {
2408
+
let ops = DOM.private(el, "sticky");
2409
+
if (!ops) {
2410
+
return;
2411
+
}
2412
+
ops.forEach(([name, op, _stashed]) => this.putSticky(el, name, op));
2413
+
},
2414
+
isLocked(el) {
2415
+
return el.hasAttribute && el.hasAttribute(PHX_REF_LOCK);
2416
+
}
2417
+
};
2418
+
var dom_default = DOM;
2419
+
var UploadEntry = class {
2420
+
static isActive(fileEl, file) {
2421
+
let isNew = file._phxRef === void 0;
2422
+
let activeRefs = fileEl.getAttribute(PHX_ACTIVE_ENTRY_REFS).split(",");
2423
+
let isActive = activeRefs.indexOf(LiveUploader.genFileRef(file)) >= 0;
2424
+
return file.size > 0 && (isNew || isActive);
2425
+
}
2426
+
static isPreflighted(fileEl, file) {
2427
+
let preflightedRefs = fileEl.getAttribute(PHX_PREFLIGHTED_REFS).split(",");
2428
+
let isPreflighted = preflightedRefs.indexOf(LiveUploader.genFileRef(file)) >= 0;
2429
+
return isPreflighted && this.isActive(fileEl, file);
2430
+
}
2431
+
static isPreflightInProgress(file) {
2432
+
return file._preflightInProgress === true;
2433
+
}
2434
+
static markPreflightInProgress(file) {
2435
+
file._preflightInProgress = true;
2436
+
}
2437
+
constructor(fileEl, file, view, autoUpload) {
2438
+
this.ref = LiveUploader.genFileRef(file);
2439
+
this.fileEl = fileEl;
2440
+
this.file = file;
2441
+
this.view = view;
2442
+
this.meta = null;
2443
+
this._isCancelled = false;
2444
+
this._isDone = false;
2445
+
this._progress = 0;
2446
+
this._lastProgressSent = -1;
2447
+
this._onDone = function() {
2448
+
};
2449
+
this._onElUpdated = this.onElUpdated.bind(this);
2450
+
this.fileEl.addEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated);
2451
+
this.autoUpload = autoUpload;
2452
+
}
2453
+
metadata() {
2454
+
return this.meta;
2455
+
}
2456
+
progress(progress) {
2457
+
this._progress = Math.floor(progress);
2458
+
if (this._progress > this._lastProgressSent) {
2459
+
if (this._progress >= 100) {
2460
+
this._progress = 100;
2461
+
this._lastProgressSent = 100;
2462
+
this._isDone = true;
2463
+
this.view.pushFileProgress(this.fileEl, this.ref, 100, () => {
2464
+
LiveUploader.untrackFile(this.fileEl, this.file);
2465
+
this._onDone();
2466
+
});
2467
+
} else {
2468
+
this._lastProgressSent = this._progress;
2469
+
this.view.pushFileProgress(this.fileEl, this.ref, this._progress);
2470
+
}
2471
+
}
2472
+
}
2473
+
isCancelled() {
2474
+
return this._isCancelled;
2475
+
}
2476
+
cancel() {
2477
+
this.file._preflightInProgress = false;
2478
+
this._isCancelled = true;
2479
+
this._isDone = true;
2480
+
this._onDone();
2481
+
}
2482
+
isDone() {
2483
+
return this._isDone;
2484
+
}
2485
+
error(reason = "failed") {
2486
+
this.fileEl.removeEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated);
2487
+
this.view.pushFileProgress(this.fileEl, this.ref, { error: reason });
2488
+
if (!this.isAutoUpload()) {
2489
+
LiveUploader.clearFiles(this.fileEl);
2490
+
}
2491
+
}
2492
+
isAutoUpload() {
2493
+
return this.autoUpload;
2494
+
}
2495
+
//private
2496
+
onDone(callback) {
2497
+
this._onDone = () => {
2498
+
this.fileEl.removeEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated);
2499
+
callback();
2500
+
};
2501
+
}
2502
+
onElUpdated() {
2503
+
let activeRefs = this.fileEl.getAttribute(PHX_ACTIVE_ENTRY_REFS).split(",");
2504
+
if (activeRefs.indexOf(this.ref) === -1) {
2505
+
LiveUploader.untrackFile(this.fileEl, this.file);
2506
+
this.cancel();
2507
+
}
2508
+
}
2509
+
toPreflightPayload() {
2510
+
return {
2511
+
last_modified: this.file.lastModified,
2512
+
name: this.file.name,
2513
+
relative_path: this.file.webkitRelativePath,
2514
+
size: this.file.size,
2515
+
type: this.file.type,
2516
+
ref: this.ref,
2517
+
meta: typeof this.file.meta === "function" ? this.file.meta() : void 0
2518
+
};
2519
+
}
2520
+
uploader(uploaders) {
2521
+
if (this.meta.uploader) {
2522
+
let callback = uploaders[this.meta.uploader] || logError(`no uploader configured for ${this.meta.uploader}`);
2523
+
return { name: this.meta.uploader, callback };
2524
+
} else {
2525
+
return { name: "channel", callback: channelUploader };
2526
+
}
2527
+
}
2528
+
zipPostFlight(resp) {
2529
+
this.meta = resp.entries[this.ref];
2530
+
if (!this.meta) {
2531
+
logError(`no preflight upload response returned with ref ${this.ref}`, { input: this.fileEl, response: resp });
2532
+
}
2533
+
}
2534
+
};
2535
+
var liveUploaderFileRef = 0;
2536
+
var LiveUploader = class _LiveUploader {
2537
+
static genFileRef(file) {
2538
+
let ref = file._phxRef;
2539
+
if (ref !== void 0) {
2540
+
return ref;
2541
+
} else {
2542
+
file._phxRef = (liveUploaderFileRef++).toString();
2543
+
return file._phxRef;
2544
+
}
2545
+
}
2546
+
static getEntryDataURL(inputEl, ref, callback) {
2547
+
let file = this.activeFiles(inputEl).find((file2) => this.genFileRef(file2) === ref);
2548
+
callback(URL.createObjectURL(file));
2549
+
}
2550
+
static hasUploadsInProgress(formEl) {
2551
+
let active = 0;
2552
+
dom_default.findUploadInputs(formEl).forEach((input) => {
2553
+
if (input.getAttribute(PHX_PREFLIGHTED_REFS) !== input.getAttribute(PHX_DONE_REFS)) {
2554
+
active++;
2555
+
}
2556
+
});
2557
+
return active > 0;
2558
+
}
2559
+
static serializeUploads(inputEl) {
2560
+
let files = this.activeFiles(inputEl);
2561
+
let fileData = {};
2562
+
files.forEach((file) => {
2563
+
let entry = { path: inputEl.name };
2564
+
let uploadRef = inputEl.getAttribute(PHX_UPLOAD_REF);
2565
+
fileData[uploadRef] = fileData[uploadRef] || [];
2566
+
entry.ref = this.genFileRef(file);
2567
+
entry.last_modified = file.lastModified;
2568
+
entry.name = file.name || entry.ref;
2569
+
entry.relative_path = file.webkitRelativePath;
2570
+
entry.type = file.type;
2571
+
entry.size = file.size;
2572
+
if (typeof file.meta === "function") {
2573
+
entry.meta = file.meta();
2574
+
}
2575
+
fileData[uploadRef].push(entry);
2576
+
});
2577
+
return fileData;
2578
+
}
2579
+
static clearFiles(inputEl) {
2580
+
inputEl.value = null;
2581
+
inputEl.removeAttribute(PHX_UPLOAD_REF);
2582
+
dom_default.putPrivate(inputEl, "files", []);
2583
+
}
2584
+
static untrackFile(inputEl, file) {
2585
+
dom_default.putPrivate(inputEl, "files", dom_default.private(inputEl, "files").filter((f) => !Object.is(f, file)));
2586
+
}
2587
+
static trackFiles(inputEl, files, dataTransfer) {
2588
+
if (inputEl.getAttribute("multiple") !== null) {
2589
+
let newFiles = files.filter((file) => !this.activeFiles(inputEl).find((f) => Object.is(f, file)));
2590
+
dom_default.updatePrivate(inputEl, "files", [], (existing) => existing.concat(newFiles));
2591
+
inputEl.value = null;
2592
+
} else {
2593
+
if (dataTransfer && dataTransfer.files.length > 0) {
2594
+
inputEl.files = dataTransfer.files;
2595
+
}
2596
+
dom_default.putPrivate(inputEl, "files", files);
2597
+
}
2598
+
}
2599
+
static activeFileInputs(formEl) {
2600
+
let fileInputs = dom_default.findUploadInputs(formEl);
2601
+
return Array.from(fileInputs).filter((el) => el.files && this.activeFiles(el).length > 0);
2602
+
}
2603
+
static activeFiles(input) {
2604
+
return (dom_default.private(input, "files") || []).filter((f) => UploadEntry.isActive(input, f));
2605
+
}
2606
+
static inputsAwaitingPreflight(formEl) {
2607
+
let fileInputs = dom_default.findUploadInputs(formEl);
2608
+
return Array.from(fileInputs).filter((input) => this.filesAwaitingPreflight(input).length > 0);
2609
+
}
2610
+
static filesAwaitingPreflight(input) {
2611
+
return this.activeFiles(input).filter((f) => !UploadEntry.isPreflighted(input, f) && !UploadEntry.isPreflightInProgress(f));
2612
+
}
2613
+
static markPreflightInProgress(entries) {
2614
+
entries.forEach((entry) => UploadEntry.markPreflightInProgress(entry.file));
2615
+
}
2616
+
constructor(inputEl, view, onComplete) {
2617
+
this.autoUpload = dom_default.isAutoUpload(inputEl);
2618
+
this.view = view;
2619
+
this.onComplete = onComplete;
2620
+
this._entries = Array.from(_LiveUploader.filesAwaitingPreflight(inputEl) || []).map((file) => new UploadEntry(inputEl, file, view, this.autoUpload));
2621
+
_LiveUploader.markPreflightInProgress(this._entries);
2622
+
this.numEntriesInProgress = this._entries.length;
2623
+
}
2624
+
isAutoUpload() {
2625
+
return this.autoUpload;
2626
+
}
2627
+
entries() {
2628
+
return this._entries;
2629
+
}
2630
+
initAdapterUpload(resp, onError, liveSocket2) {
2631
+
this._entries = this._entries.map((entry) => {
2632
+
if (entry.isCancelled()) {
2633
+
this.numEntriesInProgress--;
2634
+
if (this.numEntriesInProgress === 0) {
2635
+
this.onComplete();
2636
+
}
2637
+
} else {
2638
+
entry.zipPostFlight(resp);
2639
+
entry.onDone(() => {
2640
+
this.numEntriesInProgress--;
2641
+
if (this.numEntriesInProgress === 0) {
2642
+
this.onComplete();
2643
+
}
2644
+
});
2645
+
}
2646
+
return entry;
2647
+
});
2648
+
let groupedEntries = this._entries.reduce((acc, entry) => {
2649
+
if (!entry.meta) {
2650
+
return acc;
2651
+
}
2652
+
let { name, callback } = entry.uploader(liveSocket2.uploaders);
2653
+
acc[name] = acc[name] || { callback, entries: [] };
2654
+
acc[name].entries.push(entry);
2655
+
return acc;
2656
+
}, {});
2657
+
for (let name in groupedEntries) {
2658
+
let { callback, entries } = groupedEntries[name];
2659
+
callback(entries, onError, resp, liveSocket2);
2660
+
}
2661
+
}
2662
+
};
2663
+
var ARIA = {
2664
+
anyOf(instance, classes) {
2665
+
return classes.find((name) => instance instanceof name);
2666
+
},
2667
+
isFocusable(el, interactiveOnly) {
2668
+
return el instanceof HTMLAnchorElement && el.rel !== "ignore" || el instanceof HTMLAreaElement && el.href !== void 0 || !el.disabled && this.anyOf(el, [HTMLInputElement, HTMLSelectElement, HTMLTextAreaElement, HTMLButtonElement]) || el instanceof HTMLIFrameElement || (el.tabIndex >= 0 && el.getAttribute("aria-hidden") !== "true" || !interactiveOnly && el.getAttribute("tabindex") !== null && el.getAttribute("aria-hidden") !== "true");
2669
+
},
2670
+
attemptFocus(el, interactiveOnly) {
2671
+
if (this.isFocusable(el, interactiveOnly)) {
2672
+
try {
2673
+
el.focus();
2674
+
} catch (e) {
2675
+
}
2676
+
}
2677
+
return !!document.activeElement && document.activeElement.isSameNode(el);
2678
+
},
2679
+
focusFirstInteractive(el) {
2680
+
let child = el.firstElementChild;
2681
+
while (child) {
2682
+
if (this.attemptFocus(child, true) || this.focusFirstInteractive(child, true)) {
2683
+
return true;
2684
+
}
2685
+
child = child.nextElementSibling;
2686
+
}
2687
+
},
2688
+
focusFirst(el) {
2689
+
let child = el.firstElementChild;
2690
+
while (child) {
2691
+
if (this.attemptFocus(child) || this.focusFirst(child)) {
2692
+
return true;
2693
+
}
2694
+
child = child.nextElementSibling;
2695
+
}
2696
+
},
2697
+
focusLast(el) {
2698
+
let child = el.lastElementChild;
2699
+
while (child) {
2700
+
if (this.attemptFocus(child) || this.focusLast(child)) {
2701
+
return true;
2702
+
}
2703
+
child = child.previousElementSibling;
2704
+
}
2705
+
}
2706
+
};
2707
+
var aria_default = ARIA;
2708
+
var Hooks = {
2709
+
LiveFileUpload: {
2710
+
activeRefs() {
2711
+
return this.el.getAttribute(PHX_ACTIVE_ENTRY_REFS);
2712
+
},
2713
+
preflightedRefs() {
2714
+
return this.el.getAttribute(PHX_PREFLIGHTED_REFS);
2715
+
},
2716
+
mounted() {
2717
+
this.preflightedWas = this.preflightedRefs();
2718
+
},
2719
+
updated() {
2720
+
let newPreflights = this.preflightedRefs();
2721
+
if (this.preflightedWas !== newPreflights) {
2722
+
this.preflightedWas = newPreflights;
2723
+
if (newPreflights === "") {
2724
+
this.__view().cancelSubmit(this.el.form);
2725
+
}
2726
+
}
2727
+
if (this.activeRefs() === "") {
2728
+
this.el.value = null;
2729
+
}
2730
+
this.el.dispatchEvent(new CustomEvent(PHX_LIVE_FILE_UPDATED));
2731
+
}
2732
+
},
2733
+
LiveImgPreview: {
2734
+
mounted() {
2735
+
this.ref = this.el.getAttribute("data-phx-entry-ref");
2736
+
this.inputEl = document.getElementById(this.el.getAttribute(PHX_UPLOAD_REF));
2737
+
LiveUploader.getEntryDataURL(this.inputEl, this.ref, (url) => {
2738
+
this.url = url;
2739
+
this.el.src = url;
2740
+
});
2741
+
},
2742
+
destroyed() {
2743
+
URL.revokeObjectURL(this.url);
2744
+
}
2745
+
},
2746
+
FocusWrap: {
2747
+
mounted() {
2748
+
this.focusStart = this.el.firstElementChild;
2749
+
this.focusEnd = this.el.lastElementChild;
2750
+
this.focusStart.addEventListener("focus", (e) => {
2751
+
if (!e.relatedTarget || !this.el.contains(e.relatedTarget)) {
2752
+
const nextFocus = e.target.nextElementSibling;
2753
+
aria_default.attemptFocus(nextFocus) || aria_default.focusFirst(nextFocus);
2754
+
} else {
2755
+
aria_default.focusLast(this.el);
2756
+
}
2757
+
});
2758
+
this.focusEnd.addEventListener("focus", (e) => {
2759
+
if (!e.relatedTarget || !this.el.contains(e.relatedTarget)) {
2760
+
const nextFocus = e.target.previousElementSibling;
2761
+
aria_default.attemptFocus(nextFocus) || aria_default.focusLast(nextFocus);
2762
+
} else {
2763
+
aria_default.focusFirst(this.el);
2764
+
}
2765
+
});
2766
+
this.el.addEventListener("phx:show-end", () => this.el.focus());
2767
+
if (window.getComputedStyle(this.el).display !== "none") {
2768
+
aria_default.focusFirst(this.el);
2769
+
}
2770
+
}
2771
+
}
2772
+
};
2773
+
var findScrollContainer = (el) => {
2774
+
if (["HTML", "BODY"].indexOf(el.nodeName.toUpperCase()) >= 0)
2775
+
return null;
2776
+
if (["scroll", "auto"].indexOf(getComputedStyle(el).overflowY) >= 0)
2777
+
return el;
2778
+
return findScrollContainer(el.parentElement);
2779
+
};
2780
+
var scrollTop = (scrollContainer) => {
2781
+
if (scrollContainer) {
2782
+
return scrollContainer.scrollTop;
2783
+
} else {
2784
+
return document.documentElement.scrollTop || document.body.scrollTop;
2785
+
}
2786
+
};
2787
+
var bottom = (scrollContainer) => {
2788
+
if (scrollContainer) {
2789
+
return scrollContainer.getBoundingClientRect().bottom;
2790
+
} else {
2791
+
return window.innerHeight || document.documentElement.clientHeight;
2792
+
}
2793
+
};
2794
+
var top = (scrollContainer) => {
2795
+
if (scrollContainer) {
2796
+
return scrollContainer.getBoundingClientRect().top;
2797
+
} else {
2798
+
return 0;
2799
+
}
2800
+
};
2801
+
var isAtViewportTop = (el, scrollContainer) => {
2802
+
let rect = el.getBoundingClientRect();
2803
+
return Math.ceil(rect.top) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.top) <= bottom(scrollContainer);
2804
+
};
2805
+
var isAtViewportBottom = (el, scrollContainer) => {
2806
+
let rect = el.getBoundingClientRect();
2807
+
return Math.ceil(rect.bottom) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.bottom) <= bottom(scrollContainer);
2808
+
};
2809
+
var isWithinViewport = (el, scrollContainer) => {
2810
+
let rect = el.getBoundingClientRect();
2811
+
return Math.ceil(rect.top) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.top) <= bottom(scrollContainer);
2812
+
};
2813
+
Hooks.InfiniteScroll = {
2814
+
mounted() {
2815
+
this.scrollContainer = findScrollContainer(this.el);
2816
+
let scrollBefore = scrollTop(this.scrollContainer);
2817
+
let topOverran = false;
2818
+
let throttleInterval = 500;
2819
+
let pendingOp = null;
2820
+
let onTopOverrun = this.throttle(throttleInterval, (topEvent, firstChild) => {
2821
+
pendingOp = () => true;
2822
+
this.liveSocket.execJSHookPush(this.el, topEvent, { id: firstChild.id, _overran: true }, () => {
2823
+
pendingOp = null;
2824
+
});
2825
+
});
2826
+
let onFirstChildAtTop = this.throttle(throttleInterval, (topEvent, firstChild) => {
2827
+
pendingOp = () => firstChild.scrollIntoView({ block: "start" });
2828
+
this.liveSocket.execJSHookPush(this.el, topEvent, { id: firstChild.id }, () => {
2829
+
pendingOp = null;
2830
+
window.requestAnimationFrame(() => {
2831
+
if (!isWithinViewport(firstChild, this.scrollContainer)) {
2832
+
firstChild.scrollIntoView({ block: "start" });
2833
+
}
2834
+
});
2835
+
});
2836
+
});
2837
+
let onLastChildAtBottom = this.throttle(throttleInterval, (bottomEvent, lastChild) => {
2838
+
pendingOp = () => lastChild.scrollIntoView({ block: "end" });
2839
+
this.liveSocket.execJSHookPush(this.el, bottomEvent, { id: lastChild.id }, () => {
2840
+
pendingOp = null;
2841
+
window.requestAnimationFrame(() => {
2842
+
if (!isWithinViewport(lastChild, this.scrollContainer)) {
2843
+
lastChild.scrollIntoView({ block: "end" });
2844
+
}
2845
+
});
2846
+
});
2847
+
});
2848
+
this.onScroll = (_e) => {
2849
+
let scrollNow = scrollTop(this.scrollContainer);
2850
+
if (pendingOp) {
2851
+
scrollBefore = scrollNow;
2852
+
return pendingOp();
2853
+
}
2854
+
let rect = this.el.getBoundingClientRect();
2855
+
let topEvent = this.el.getAttribute(this.liveSocket.binding("viewport-top"));
2856
+
let bottomEvent = this.el.getAttribute(this.liveSocket.binding("viewport-bottom"));
2857
+
let lastChild = this.el.lastElementChild;
2858
+
let firstChild = this.el.firstElementChild;
2859
+
let isScrollingUp = scrollNow < scrollBefore;
2860
+
let isScrollingDown = scrollNow > scrollBefore;
2861
+
if (isScrollingUp && topEvent && !topOverran && rect.top >= 0) {
2862
+
topOverran = true;
2863
+
onTopOverrun(topEvent, firstChild);
2864
+
} else if (isScrollingDown && topOverran && rect.top <= 0) {
2865
+
topOverran = false;
2866
+
}
2867
+
if (topEvent && isScrollingUp && isAtViewportTop(firstChild, this.scrollContainer)) {
2868
+
onFirstChildAtTop(topEvent, firstChild);
2869
+
} else if (bottomEvent && isScrollingDown && isAtViewportBottom(lastChild, this.scrollContainer)) {
2870
+
onLastChildAtBottom(bottomEvent, lastChild);
2871
+
}
2872
+
scrollBefore = scrollNow;
2873
+
};
2874
+
if (this.scrollContainer) {
2875
+
this.scrollContainer.addEventListener("scroll", this.onScroll);
2876
+
} else {
2877
+
window.addEventListener("scroll", this.onScroll);
2878
+
}
2879
+
},
2880
+
destroyed() {
2881
+
if (this.scrollContainer) {
2882
+
this.scrollContainer.removeEventListener("scroll", this.onScroll);
2883
+
} else {
2884
+
window.removeEventListener("scroll", this.onScroll);
2885
+
}
2886
+
},
2887
+
throttle(interval, callback) {
2888
+
let lastCallAt = 0;
2889
+
let timer;
2890
+
return (...args) => {
2891
+
let now = Date.now();
2892
+
let remainingTime = interval - (now - lastCallAt);
2893
+
if (remainingTime <= 0 || remainingTime > interval) {
2894
+
if (timer) {
2895
+
clearTimeout(timer);
2896
+
timer = null;
2897
+
}
2898
+
lastCallAt = now;
2899
+
callback(...args);
2900
+
} else if (!timer) {
2901
+
timer = setTimeout(() => {
2902
+
lastCallAt = Date.now();
2903
+
timer = null;
2904
+
callback(...args);
2905
+
}, remainingTime);
2906
+
}
2907
+
};
2908
+
}
2909
+
};
2910
+
var hooks_default = Hooks;
2911
+
var ElementRef = class {
2912
+
static onUnlock(el, callback) {
2913
+
if (!dom_default.isLocked(el) && !el.closest(`[${PHX_REF_LOCK}]`)) {
2914
+
return callback();
2915
+
}
2916
+
const closestLock = el.closest(`[${PHX_REF_LOCK}]`);
2917
+
const ref = closestLock.closest(`[${PHX_REF_LOCK}]`).getAttribute(PHX_REF_LOCK);
2918
+
closestLock.addEventListener(`phx:undo-lock:${ref}`, () => {
2919
+
callback();
2920
+
}, { once: true });
2921
+
}
2922
+
constructor(el) {
2923
+
this.el = el;
2924
+
this.loadingRef = el.hasAttribute(PHX_REF_LOADING) ? parseInt(el.getAttribute(PHX_REF_LOADING), 10) : null;
2925
+
this.lockRef = el.hasAttribute(PHX_REF_LOCK) ? parseInt(el.getAttribute(PHX_REF_LOCK), 10) : null;
2926
+
}
2927
+
// public
2928
+
maybeUndo(ref, phxEvent, eachCloneCallback) {
2929
+
if (!this.isWithin(ref)) {
2930
+
dom_default.updatePrivate(this.el, PHX_PENDING_REFS, [], (pendingRefs) => {
2931
+
pendingRefs.push(ref);
2932
+
return pendingRefs;
2933
+
});
2934
+
return;
2935
+
}
2936
+
this.undoLocks(ref, phxEvent, eachCloneCallback);
2937
+
this.undoLoading(ref, phxEvent);
2938
+
dom_default.updatePrivate(this.el, PHX_PENDING_REFS, [], (pendingRefs) => {
2939
+
return pendingRefs.filter((pendingRef) => {
2940
+
let opts = {
2941
+
detail: { ref: pendingRef, event: phxEvent },
2942
+
bubbles: true,
2943
+
cancelable: false
2944
+
};
2945
+
if (this.loadingRef && this.loadingRef > pendingRef) {
2946
+
this.el.dispatchEvent(
2947
+
new CustomEvent(`phx:undo-loading:${pendingRef}`, opts)
2948
+
);
2949
+
}
2950
+
if (this.lockRef && this.lockRef > pendingRef) {
2951
+
this.el.dispatchEvent(
2952
+
new CustomEvent(`phx:undo-lock:${pendingRef}`, opts)
2953
+
);
2954
+
}
2955
+
return pendingRef > ref;
2956
+
});
2957
+
});
2958
+
if (this.isFullyResolvedBy(ref)) {
2959
+
this.el.removeAttribute(PHX_REF_SRC);
2960
+
}
2961
+
}
2962
+
// private
2963
+
isWithin(ref) {
2964
+
return !(this.loadingRef !== null && this.loadingRef > ref && (this.lockRef !== null && this.lockRef > ref));
2965
+
}
2966
+
// Check for cloned PHX_REF_LOCK element that has been morphed behind
2967
+
// the scenes while this element was locked in the DOM.
2968
+
// When we apply the cloned tree to the active DOM element, we must
2969
+
//
2970
+
// 1. execute pending mounted hooks for nodes now in the DOM
2971
+
// 2. undo any ref inside the cloned tree that has since been ack'd
2972
+
undoLocks(ref, phxEvent, eachCloneCallback) {
2973
+
if (!this.isLockUndoneBy(ref)) {
2974
+
return;
2975
+
}
2976
+
let clonedTree = dom_default.private(this.el, PHX_REF_LOCK);
2977
+
if (clonedTree) {
2978
+
eachCloneCallback(clonedTree);
2979
+
dom_default.deletePrivate(this.el, PHX_REF_LOCK);
2980
+
}
2981
+
this.el.removeAttribute(PHX_REF_LOCK);
2982
+
let opts = { detail: { ref, event: phxEvent }, bubbles: true, cancelable: false };
2983
+
this.el.dispatchEvent(new CustomEvent(`phx:undo-lock:${this.lockRef}`, opts));
2984
+
}
2985
+
undoLoading(ref, phxEvent) {
2986
+
if (!this.isLoadingUndoneBy(ref)) {
2987
+
if (this.canUndoLoading(ref) && this.el.classList.contains("phx-submit-loading")) {
2988
+
this.el.classList.remove("phx-change-loading");
2989
+
}
2990
+
return;
2991
+
}
2992
+
if (this.canUndoLoading(ref)) {
2993
+
this.el.removeAttribute(PHX_REF_LOADING);
2994
+
let disabledVal = this.el.getAttribute(PHX_DISABLED);
2995
+
let readOnlyVal = this.el.getAttribute(PHX_READONLY);
2996
+
if (readOnlyVal !== null) {
2997
+
this.el.readOnly = readOnlyVal === "true" ? true : false;
2998
+
this.el.removeAttribute(PHX_READONLY);
2999
+
}
3000
+
if (disabledVal !== null) {
3001
+
this.el.disabled = disabledVal === "true" ? true : false;
3002
+
this.el.removeAttribute(PHX_DISABLED);
3003
+
}
3004
+
let disableRestore = this.el.getAttribute(PHX_DISABLE_WITH_RESTORE);
3005
+
if (disableRestore !== null) {
3006
+
this.el.innerText = disableRestore;
3007
+
this.el.removeAttribute(PHX_DISABLE_WITH_RESTORE);
3008
+
}
3009
+
let opts = { detail: { ref, event: phxEvent }, bubbles: true, cancelable: false };
3010
+
this.el.dispatchEvent(new CustomEvent(`phx:undo-loading:${this.loadingRef}`, opts));
3011
+
}
3012
+
PHX_EVENT_CLASSES.forEach((name) => {
3013
+
if (name !== "phx-submit-loading" || this.canUndoLoading(ref)) {
3014
+
dom_default.removeClass(this.el, name);
3015
+
}
3016
+
});
3017
+
}
3018
+
isLoadingUndoneBy(ref) {
3019
+
return this.loadingRef === null ? false : this.loadingRef <= ref;
3020
+
}
3021
+
isLockUndoneBy(ref) {
3022
+
return this.lockRef === null ? false : this.lockRef <= ref;
3023
+
}
3024
+
isFullyResolvedBy(ref) {
3025
+
return (this.loadingRef === null || this.loadingRef <= ref) && (this.lockRef === null || this.lockRef <= ref);
3026
+
}
3027
+
// only remove the phx-submit-loading class if we are not locked
3028
+
canUndoLoading(ref) {
3029
+
return this.lockRef === null || this.lockRef <= ref;
3030
+
}
3031
+
};
3032
+
var DOMPostMorphRestorer = class {
3033
+
constructor(containerBefore, containerAfter, updateType) {
3034
+
let idsBefore = /* @__PURE__ */ new Set();
3035
+
let idsAfter = new Set([...containerAfter.children].map((child) => child.id));
3036
+
let elementsToModify = [];
3037
+
Array.from(containerBefore.children).forEach((child) => {
3038
+
if (child.id) {
3039
+
idsBefore.add(child.id);
3040
+
if (idsAfter.has(child.id)) {
3041
+
let previousElementId = child.previousElementSibling && child.previousElementSibling.id;
3042
+
elementsToModify.push({ elementId: child.id, previousElementId });
3043
+
}
3044
+
}
3045
+
});
3046
+
this.containerId = containerAfter.id;
3047
+
this.updateType = updateType;
3048
+
this.elementsToModify = elementsToModify;
3049
+
this.elementIdsToAdd = [...idsAfter].filter((id) => !idsBefore.has(id));
3050
+
}
3051
+
// We do the following to optimize append/prepend operations:
3052
+
// 1) Track ids of modified elements & of new elements
3053
+
// 2) All the modified elements are put back in the correct position in the DOM tree
3054
+
// by storing the id of their previous sibling
3055
+
// 3) New elements are going to be put in the right place by morphdom during append.
3056
+
// For prepend, we move them to the first position in the container
3057
+
perform() {
3058
+
let container = dom_default.byId(this.containerId);
3059
+
this.elementsToModify.forEach((elementToModify) => {
3060
+
if (elementToModify.previousElementId) {
3061
+
maybe(document.getElementById(elementToModify.previousElementId), (previousElem) => {
3062
+
maybe(document.getElementById(elementToModify.elementId), (elem) => {
3063
+
let isInRightPlace = elem.previousElementSibling && elem.previousElementSibling.id == previousElem.id;
3064
+
if (!isInRightPlace) {
3065
+
previousElem.insertAdjacentElement("afterend", elem);
3066
+
}
3067
+
});
3068
+
});
3069
+
} else {
3070
+
maybe(document.getElementById(elementToModify.elementId), (elem) => {
3071
+
let isInRightPlace = elem.previousElementSibling == null;
3072
+
if (!isInRightPlace) {
3073
+
container.insertAdjacentElement("afterbegin", elem);
3074
+
}
3075
+
});
3076
+
}
3077
+
});
3078
+
if (this.updateType == "prepend") {
3079
+
this.elementIdsToAdd.reverse().forEach((elemId) => {
3080
+
maybe(document.getElementById(elemId), (elem) => container.insertAdjacentElement("afterbegin", elem));
3081
+
});
3082
+
}
3083
+
}
3084
+
};
3085
+
var DOCUMENT_FRAGMENT_NODE = 11;
3086
+
function morphAttrs(fromNode, toNode) {
3087
+
var toNodeAttrs = toNode.attributes;
3088
+
var attr;
3089
+
var attrName;
3090
+
var attrNamespaceURI;
3091
+
var attrValue;
3092
+
var fromValue;
3093
+
if (toNode.nodeType === DOCUMENT_FRAGMENT_NODE || fromNode.nodeType === DOCUMENT_FRAGMENT_NODE) {
3094
+
return;
3095
+
}
3096
+
for (var i = toNodeAttrs.length - 1; i >= 0; i--) {
3097
+
attr = toNodeAttrs[i];
3098
+
attrName = attr.name;
3099
+
attrNamespaceURI = attr.namespaceURI;
3100
+
attrValue = attr.value;
3101
+
if (attrNamespaceURI) {
3102
+
attrName = attr.localName || attrName;
3103
+
fromValue = fromNode.getAttributeNS(attrNamespaceURI, attrName);
3104
+
if (fromValue !== attrValue) {
3105
+
if (attr.prefix === "xmlns") {
3106
+
attrName = attr.name;
3107
+
}
3108
+
fromNode.setAttributeNS(attrNamespaceURI, attrName, attrValue);
3109
+
}
3110
+
} else {
3111
+
fromValue = fromNode.getAttribute(attrName);
3112
+
if (fromValue !== attrValue) {
3113
+
fromNode.setAttribute(attrName, attrValue);
3114
+
}
3115
+
}
3116
+
}
3117
+
var fromNodeAttrs = fromNode.attributes;
3118
+
for (var d = fromNodeAttrs.length - 1; d >= 0; d--) {
3119
+
attr = fromNodeAttrs[d];
3120
+
attrName = attr.name;
3121
+
attrNamespaceURI = attr.namespaceURI;
3122
+
if (attrNamespaceURI) {
3123
+
attrName = attr.localName || attrName;
3124
+
if (!toNode.hasAttributeNS(attrNamespaceURI, attrName)) {
3125
+
fromNode.removeAttributeNS(attrNamespaceURI, attrName);
3126
+
}
3127
+
} else {
3128
+
if (!toNode.hasAttribute(attrName)) {
3129
+
fromNode.removeAttribute(attrName);
3130
+
}
3131
+
}
3132
+
}
3133
+
}
3134
+
var range;
3135
+
var NS_XHTML = "http://www.w3.org/1999/xhtml";
3136
+
var doc = typeof document === "undefined" ? void 0 : document;
3137
+
var HAS_TEMPLATE_SUPPORT = !!doc && "content" in doc.createElement("template");
3138
+
var HAS_RANGE_SUPPORT = !!doc && doc.createRange && "createContextualFragment" in doc.createRange();
3139
+
function createFragmentFromTemplate(str) {
3140
+
var template = doc.createElement("template");
3141
+
template.innerHTML = str;
3142
+
return template.content.childNodes[0];
3143
+
}
3144
+
function createFragmentFromRange(str) {
3145
+
if (!range) {
3146
+
range = doc.createRange();
3147
+
range.selectNode(doc.body);
3148
+
}
3149
+
var fragment = range.createContextualFragment(str);
3150
+
return fragment.childNodes[0];
3151
+
}
3152
+
function createFragmentFromWrap(str) {
3153
+
var fragment = doc.createElement("body");
3154
+
fragment.innerHTML = str;
3155
+
return fragment.childNodes[0];
3156
+
}
3157
+
function toElement(str) {
3158
+
str = str.trim();
3159
+
if (HAS_TEMPLATE_SUPPORT) {
3160
+
return createFragmentFromTemplate(str);
3161
+
} else if (HAS_RANGE_SUPPORT) {
3162
+
return createFragmentFromRange(str);
3163
+
}
3164
+
return createFragmentFromWrap(str);
3165
+
}
3166
+
function compareNodeNames(fromEl, toEl) {
3167
+
var fromNodeName = fromEl.nodeName;
3168
+
var toNodeName = toEl.nodeName;
3169
+
var fromCodeStart, toCodeStart;
3170
+
if (fromNodeName === toNodeName) {
3171
+
return true;
3172
+
}
3173
+
fromCodeStart = fromNodeName.charCodeAt(0);
3174
+
toCodeStart = toNodeName.charCodeAt(0);
3175
+
if (fromCodeStart <= 90 && toCodeStart >= 97) {
3176
+
return fromNodeName === toNodeName.toUpperCase();
3177
+
} else if (toCodeStart <= 90 && fromCodeStart >= 97) {
3178
+
return toNodeName === fromNodeName.toUpperCase();
3179
+
} else {
3180
+
return false;
3181
+
}
3182
+
}
3183
+
function createElementNS(name, namespaceURI) {
3184
+
return !namespaceURI || namespaceURI === NS_XHTML ? doc.createElement(name) : doc.createElementNS(namespaceURI, name);
3185
+
}
3186
+
function moveChildren(fromEl, toEl) {
3187
+
var curChild = fromEl.firstChild;
3188
+
while (curChild) {
3189
+
var nextChild = curChild.nextSibling;
3190
+
toEl.appendChild(curChild);
3191
+
curChild = nextChild;
3192
+
}
3193
+
return toEl;
3194
+
}
3195
+
function syncBooleanAttrProp(fromEl, toEl, name) {
3196
+
if (fromEl[name] !== toEl[name]) {
3197
+
fromEl[name] = toEl[name];
3198
+
if (fromEl[name]) {
3199
+
fromEl.setAttribute(name, "");
3200
+
} else {
3201
+
fromEl.removeAttribute(name);
3202
+
}
3203
+
}
3204
+
}
3205
+
var specialElHandlers = {
3206
+
OPTION: function(fromEl, toEl) {
3207
+
var parentNode = fromEl.parentNode;
3208
+
if (parentNode) {
3209
+
var parentName = parentNode.nodeName.toUpperCase();
3210
+
if (parentName === "OPTGROUP") {
3211
+
parentNode = parentNode.parentNode;
3212
+
parentName = parentNode && parentNode.nodeName.toUpperCase();
3213
+
}
3214
+
if (parentName === "SELECT" && !parentNode.hasAttribute("multiple")) {
3215
+
if (fromEl.hasAttribute("selected") && !toEl.selected) {
3216
+
fromEl.setAttribute("selected", "selected");
3217
+
fromEl.removeAttribute("selected");
3218
+
}
3219
+
parentNode.selectedIndex = -1;
3220
+
}
3221
+
}
3222
+
syncBooleanAttrProp(fromEl, toEl, "selected");
3223
+
},
3224
+
/**
3225
+
* The "value" attribute is special for the <input> element since it sets
3226
+
* the initial value. Changing the "value" attribute without changing the
3227
+
* "value" property will have no effect since it is only used to the set the
3228
+
* initial value. Similar for the "checked" attribute, and "disabled".
3229
+
*/
3230
+
INPUT: function(fromEl, toEl) {
3231
+
syncBooleanAttrProp(fromEl, toEl, "checked");
3232
+
syncBooleanAttrProp(fromEl, toEl, "disabled");
3233
+
if (fromEl.value !== toEl.value) {
3234
+
fromEl.value = toEl.value;
3235
+
}
3236
+
if (!toEl.hasAttribute("value")) {
3237
+
fromEl.removeAttribute("value");
3238
+
}
3239
+
},
3240
+
TEXTAREA: function(fromEl, toEl) {
3241
+
var newValue = toEl.value;
3242
+
if (fromEl.value !== newValue) {
3243
+
fromEl.value = newValue;
3244
+
}
3245
+
var firstChild = fromEl.firstChild;
3246
+
if (firstChild) {
3247
+
var oldValue = firstChild.nodeValue;
3248
+
if (oldValue == newValue || !newValue && oldValue == fromEl.placeholder) {
3249
+
return;
3250
+
}
3251
+
firstChild.nodeValue = newValue;
3252
+
}
3253
+
},
3254
+
SELECT: function(fromEl, toEl) {
3255
+
if (!toEl.hasAttribute("multiple")) {
3256
+
var selectedIndex = -1;
3257
+
var i = 0;
3258
+
var curChild = fromEl.firstChild;
3259
+
var optgroup;
3260
+
var nodeName;
3261
+
while (curChild) {
3262
+
nodeName = curChild.nodeName && curChild.nodeName.toUpperCase();
3263
+
if (nodeName === "OPTGROUP") {
3264
+
optgroup = curChild;
3265
+
curChild = optgroup.firstChild;
3266
+
} else {
3267
+
if (nodeName === "OPTION") {
3268
+
if (curChild.hasAttribute("selected")) {
3269
+
selectedIndex = i;
3270
+
break;
3271
+
}
3272
+
i++;
3273
+
}
3274
+
curChild = curChild.nextSibling;
3275
+
if (!curChild && optgroup) {
3276
+
curChild = optgroup.nextSibling;
3277
+
optgroup = null;
3278
+
}
3279
+
}
3280
+
}
3281
+
fromEl.selectedIndex = selectedIndex;
3282
+
}
3283
+
}
3284
+
};
3285
+
var ELEMENT_NODE = 1;
3286
+
var DOCUMENT_FRAGMENT_NODE$1 = 11;
3287
+
var TEXT_NODE = 3;
3288
+
var COMMENT_NODE = 8;
3289
+
function noop() {
3290
+
}
3291
+
function defaultGetNodeKey(node) {
3292
+
if (node) {
3293
+
return node.getAttribute && node.getAttribute("id") || node.id;
3294
+
}
3295
+
}
3296
+
function morphdomFactory(morphAttrs2) {
3297
+
return function morphdom2(fromNode, toNode, options) {
3298
+
if (!options) {
3299
+
options = {};
3300
+
}
3301
+
if (typeof toNode === "string") {
3302
+
if (fromNode.nodeName === "#document" || fromNode.nodeName === "HTML" || fromNode.nodeName === "BODY") {
3303
+
var toNodeHtml = toNode;
3304
+
toNode = doc.createElement("html");
3305
+
toNode.innerHTML = toNodeHtml;
3306
+
} else {
3307
+
toNode = toElement(toNode);
3308
+
}
3309
+
} else if (toNode.nodeType === DOCUMENT_FRAGMENT_NODE$1) {
3310
+
toNode = toNode.firstElementChild;
3311
+
}
3312
+
var getNodeKey = options.getNodeKey || defaultGetNodeKey;
3313
+
var onBeforeNodeAdded = options.onBeforeNodeAdded || noop;
3314
+
var onNodeAdded = options.onNodeAdded || noop;
3315
+
var onBeforeElUpdated = options.onBeforeElUpdated || noop;
3316
+
var onElUpdated = options.onElUpdated || noop;
3317
+
var onBeforeNodeDiscarded = options.onBeforeNodeDiscarded || noop;
3318
+
var onNodeDiscarded = options.onNodeDiscarded || noop;
3319
+
var onBeforeElChildrenUpdated = options.onBeforeElChildrenUpdated || noop;
3320
+
var skipFromChildren = options.skipFromChildren || noop;
3321
+
var addChild = options.addChild || function(parent, child) {
3322
+
return parent.appendChild(child);
3323
+
};
3324
+
var childrenOnly = options.childrenOnly === true;
3325
+
var fromNodesLookup = /* @__PURE__ */ Object.create(null);
3326
+
var keyedRemovalList = [];
3327
+
function addKeyedRemoval(key) {
3328
+
keyedRemovalList.push(key);
3329
+
}
3330
+
function walkDiscardedChildNodes(node, skipKeyedNodes) {
3331
+
if (node.nodeType === ELEMENT_NODE) {
3332
+
var curChild = node.firstChild;
3333
+
while (curChild) {
3334
+
var key = void 0;
3335
+
if (skipKeyedNodes && (key = getNodeKey(curChild))) {
3336
+
addKeyedRemoval(key);
3337
+
} else {
3338
+
onNodeDiscarded(curChild);
3339
+
if (curChild.firstChild) {
3340
+
walkDiscardedChildNodes(curChild, skipKeyedNodes);
3341
+
}
3342
+
}
3343
+
curChild = curChild.nextSibling;
3344
+
}
3345
+
}
3346
+
}
3347
+
function removeNode(node, parentNode, skipKeyedNodes) {
3348
+
if (onBeforeNodeDiscarded(node) === false) {
3349
+
return;
3350
+
}
3351
+
if (parentNode) {
3352
+
parentNode.removeChild(node);
3353
+
}
3354
+
onNodeDiscarded(node);
3355
+
walkDiscardedChildNodes(node, skipKeyedNodes);
3356
+
}
3357
+
function indexTree(node) {
3358
+
if (node.nodeType === ELEMENT_NODE || node.nodeType === DOCUMENT_FRAGMENT_NODE$1) {
3359
+
var curChild = node.firstChild;
3360
+
while (curChild) {
3361
+
var key = getNodeKey(curChild);
3362
+
if (key) {
3363
+
fromNodesLookup[key] = curChild;
3364
+
}
3365
+
indexTree(curChild);
3366
+
curChild = curChild.nextSibling;
3367
+
}
3368
+
}
3369
+
}
3370
+
indexTree(fromNode);
3371
+
function handleNodeAdded(el) {
3372
+
onNodeAdded(el);
3373
+
var curChild = el.firstChild;
3374
+
while (curChild) {
3375
+
var nextSibling = curChild.nextSibling;
3376
+
var key = getNodeKey(curChild);
3377
+
if (key) {
3378
+
var unmatchedFromEl = fromNodesLookup[key];
3379
+
if (unmatchedFromEl && compareNodeNames(curChild, unmatchedFromEl)) {
3380
+
curChild.parentNode.replaceChild(unmatchedFromEl, curChild);
3381
+
morphEl(unmatchedFromEl, curChild);
3382
+
} else {
3383
+
handleNodeAdded(curChild);
3384
+
}
3385
+
} else {
3386
+
handleNodeAdded(curChild);
3387
+
}
3388
+
curChild = nextSibling;
3389
+
}
3390
+
}
3391
+
function cleanupFromEl(fromEl, curFromNodeChild, curFromNodeKey) {
3392
+
while (curFromNodeChild) {
3393
+
var fromNextSibling = curFromNodeChild.nextSibling;
3394
+
if (curFromNodeKey = getNodeKey(curFromNodeChild)) {
3395
+
addKeyedRemoval(curFromNodeKey);
3396
+
} else {
3397
+
removeNode(
3398
+
curFromNodeChild,
3399
+
fromEl,
3400
+
true
3401
+
/* skip keyed nodes */
3402
+
);
3403
+
}
3404
+
curFromNodeChild = fromNextSibling;
3405
+
}
3406
+
}
3407
+
function morphEl(fromEl, toEl, childrenOnly2) {
3408
+
var toElKey = getNodeKey(toEl);
3409
+
if (toElKey) {
3410
+
delete fromNodesLookup[toElKey];
3411
+
}
3412
+
if (!childrenOnly2) {
3413
+
var beforeUpdateResult = onBeforeElUpdated(fromEl, toEl);
3414
+
if (beforeUpdateResult === false) {
3415
+
return;
3416
+
} else if (beforeUpdateResult instanceof HTMLElement) {
3417
+
fromEl = beforeUpdateResult;
3418
+
indexTree(fromEl);
3419
+
}
3420
+
morphAttrs2(fromEl, toEl);
3421
+
onElUpdated(fromEl);
3422
+
if (onBeforeElChildrenUpdated(fromEl, toEl) === false) {
3423
+
return;
3424
+
}
3425
+
}
3426
+
if (fromEl.nodeName !== "TEXTAREA") {
3427
+
morphChildren(fromEl, toEl);
3428
+
} else {
3429
+
specialElHandlers.TEXTAREA(fromEl, toEl);
3430
+
}
3431
+
}
3432
+
function morphChildren(fromEl, toEl) {
3433
+
var skipFrom = skipFromChildren(fromEl, toEl);
3434
+
var curToNodeChild = toEl.firstChild;
3435
+
var curFromNodeChild = fromEl.firstChild;
3436
+
var curToNodeKey;
3437
+
var curFromNodeKey;
3438
+
var fromNextSibling;
3439
+
var toNextSibling;
3440
+
var matchingFromEl;
3441
+
outer:
3442
+
while (curToNodeChild) {
3443
+
toNextSibling = curToNodeChild.nextSibling;
3444
+
curToNodeKey = getNodeKey(curToNodeChild);
3445
+
while (!skipFrom && curFromNodeChild) {
3446
+
fromNextSibling = curFromNodeChild.nextSibling;
3447
+
if (curToNodeChild.isSameNode && curToNodeChild.isSameNode(curFromNodeChild)) {
3448
+
curToNodeChild = toNextSibling;
3449
+
curFromNodeChild = fromNextSibling;
3450
+
continue outer;
3451
+
}
3452
+
curFromNodeKey = getNodeKey(curFromNodeChild);
3453
+
var curFromNodeType = curFromNodeChild.nodeType;
3454
+
var isCompatible = void 0;
3455
+
if (curFromNodeType === curToNodeChild.nodeType) {
3456
+
if (curFromNodeType === ELEMENT_NODE) {
3457
+
if (curToNodeKey) {
3458
+
if (curToNodeKey !== curFromNodeKey) {
3459
+
if (matchingFromEl = fromNodesLookup[curToNodeKey]) {
3460
+
if (fromNextSibling === matchingFromEl) {
3461
+
isCompatible = false;
3462
+
} else {
3463
+
fromEl.insertBefore(matchingFromEl, curFromNodeChild);
3464
+
if (curFromNodeKey) {
3465
+
addKeyedRemoval(curFromNodeKey);
3466
+
} else {
3467
+
removeNode(
3468
+
curFromNodeChild,
3469
+
fromEl,
3470
+
true
3471
+
/* skip keyed nodes */
3472
+
);
3473
+
}
3474
+
curFromNodeChild = matchingFromEl;
3475
+
curFromNodeKey = getNodeKey(curFromNodeChild);
3476
+
}
3477
+
} else {
3478
+
isCompatible = false;
3479
+
}
3480
+
}
3481
+
} else if (curFromNodeKey) {
3482
+
isCompatible = false;
3483
+
}
3484
+
isCompatible = isCompatible !== false && compareNodeNames(curFromNodeChild, curToNodeChild);
3485
+
if (isCompatible) {
3486
+
morphEl(curFromNodeChild, curToNodeChild);
3487
+
}
3488
+
} else if (curFromNodeType === TEXT_NODE || curFromNodeType == COMMENT_NODE) {
3489
+
isCompatible = true;
3490
+
if (curFromNodeChild.nodeValue !== curToNodeChild.nodeValue) {
3491
+
curFromNodeChild.nodeValue = curToNodeChild.nodeValue;
3492
+
}
3493
+
}
3494
+
}
3495
+
if (isCompatible) {
3496
+
curToNodeChild = toNextSibling;
3497
+
curFromNodeChild = fromNextSibling;
3498
+
continue outer;
3499
+
}
3500
+
if (curFromNodeKey) {
3501
+
addKeyedRemoval(curFromNodeKey);
3502
+
} else {
3503
+
removeNode(
3504
+
curFromNodeChild,
3505
+
fromEl,
3506
+
true
3507
+
/* skip keyed nodes */
3508
+
);
3509
+
}
3510
+
curFromNodeChild = fromNextSibling;
3511
+
}
3512
+
if (curToNodeKey && (matchingFromEl = fromNodesLookup[curToNodeKey]) && compareNodeNames(matchingFromEl, curToNodeChild)) {
3513
+
if (!skipFrom) {
3514
+
addChild(fromEl, matchingFromEl);
3515
+
}
3516
+
morphEl(matchingFromEl, curToNodeChild);
3517
+
} else {
3518
+
var onBeforeNodeAddedResult = onBeforeNodeAdded(curToNodeChild);
3519
+
if (onBeforeNodeAddedResult !== false) {
3520
+
if (onBeforeNodeAddedResult) {
3521
+
curToNodeChild = onBeforeNodeAddedResult;
3522
+
}
3523
+
if (curToNodeChild.actualize) {
3524
+
curToNodeChild = curToNodeChild.actualize(fromEl.ownerDocument || doc);
3525
+
}
3526
+
addChild(fromEl, curToNodeChild);
3527
+
handleNodeAdded(curToNodeChild);
3528
+
}
3529
+
}
3530
+
curToNodeChild = toNextSibling;
3531
+
curFromNodeChild = fromNextSibling;
3532
+
}
3533
+
cleanupFromEl(fromEl, curFromNodeChild, curFromNodeKey);
3534
+
var specialElHandler = specialElHandlers[fromEl.nodeName];
3535
+
if (specialElHandler) {
3536
+
specialElHandler(fromEl, toEl);
3537
+
}
3538
+
}
3539
+
var morphedNode = fromNode;
3540
+
var morphedNodeType = morphedNode.nodeType;
3541
+
var toNodeType = toNode.nodeType;
3542
+
if (!childrenOnly) {
3543
+
if (morphedNodeType === ELEMENT_NODE) {
3544
+
if (toNodeType === ELEMENT_NODE) {
3545
+
if (!compareNodeNames(fromNode, toNode)) {
3546
+
onNodeDiscarded(fromNode);
3547
+
morphedNode = moveChildren(fromNode, createElementNS(toNode.nodeName, toNode.namespaceURI));
3548
+
}
3549
+
} else {
3550
+
morphedNode = toNode;
3551
+
}
3552
+
} else if (morphedNodeType === TEXT_NODE || morphedNodeType === COMMENT_NODE) {
3553
+
if (toNodeType === morphedNodeType) {
3554
+
if (morphedNode.nodeValue !== toNode.nodeValue) {
3555
+
morphedNode.nodeValue = toNode.nodeValue;
3556
+
}
3557
+
return morphedNode;
3558
+
} else {
3559
+
morphedNode = toNode;
3560
+
}
3561
+
}
3562
+
}
3563
+
if (morphedNode === toNode) {
3564
+
onNodeDiscarded(fromNode);
3565
+
} else {
3566
+
if (toNode.isSameNode && toNode.isSameNode(morphedNode)) {
3567
+
return;
3568
+
}
3569
+
morphEl(morphedNode, toNode, childrenOnly);
3570
+
if (keyedRemovalList) {
3571
+
for (var i = 0, len = keyedRemovalList.length; i < len; i++) {
3572
+
var elToRemove = fromNodesLookup[keyedRemovalList[i]];
3573
+
if (elToRemove) {
3574
+
removeNode(elToRemove, elToRemove.parentNode, false);
3575
+
}
3576
+
}
3577
+
}
3578
+
}
3579
+
if (!childrenOnly && morphedNode !== fromNode && fromNode.parentNode) {
3580
+
if (morphedNode.actualize) {
3581
+
morphedNode = morphedNode.actualize(fromNode.ownerDocument || doc);
3582
+
}
3583
+
fromNode.parentNode.replaceChild(morphedNode, fromNode);
3584
+
}
3585
+
return morphedNode;
3586
+
};
3587
+
}
3588
+
var morphdom = morphdomFactory(morphAttrs);
3589
+
var morphdom_esm_default = morphdom;
3590
+
var DOMPatch = class {
3591
+
constructor(view, container, id, html, streams, targetCID, opts = {}) {
3592
+
this.view = view;
3593
+
this.liveSocket = view.liveSocket;
3594
+
this.container = container;
3595
+
this.id = id;
3596
+
this.rootID = view.root.id;
3597
+
this.html = html;
3598
+
this.streams = streams;
3599
+
this.streamInserts = {};
3600
+
this.streamComponentRestore = {};
3601
+
this.targetCID = targetCID;
3602
+
this.cidPatch = isCid(this.targetCID);
3603
+
this.pendingRemoves = [];
3604
+
this.phxRemove = this.liveSocket.binding("remove");
3605
+
this.targetContainer = this.isCIDPatch() ? this.targetCIDContainer(html) : container;
3606
+
this.callbacks = {
3607
+
beforeadded: [],
3608
+
beforeupdated: [],
3609
+
beforephxChildAdded: [],
3610
+
afteradded: [],
3611
+
afterupdated: [],
3612
+
afterdiscarded: [],
3613
+
afterphxChildAdded: [],
3614
+
aftertransitionsDiscarded: []
3615
+
};
3616
+
this.withChildren = opts.withChildren || opts.undoRef || false;
3617
+
this.undoRef = opts.undoRef;
3618
+
}
3619
+
before(kind, callback) {
3620
+
this.callbacks[`before${kind}`].push(callback);
3621
+
}
3622
+
after(kind, callback) {
3623
+
this.callbacks[`after${kind}`].push(callback);
3624
+
}
3625
+
trackBefore(kind, ...args) {
3626
+
this.callbacks[`before${kind}`].forEach((callback) => callback(...args));
3627
+
}
3628
+
trackAfter(kind, ...args) {
3629
+
this.callbacks[`after${kind}`].forEach((callback) => callback(...args));
3630
+
}
3631
+
markPrunableContentForRemoval() {
3632
+
let phxUpdate = this.liveSocket.binding(PHX_UPDATE);
3633
+
dom_default.all(this.container, `[${phxUpdate}=append] > *, [${phxUpdate}=prepend] > *`, (el) => {
3634
+
el.setAttribute(PHX_PRUNE, "");
3635
+
});
3636
+
}
3637
+
perform(isJoinPatch) {
3638
+
let { view, liveSocket: liveSocket2, html, container, targetContainer } = this;
3639
+
if (this.isCIDPatch() && !targetContainer) {
3640
+
return;
3641
+
}
3642
+
let focused = liveSocket2.getActiveElement();
3643
+
let { selectionStart, selectionEnd } = focused && dom_default.hasSelectionRange(focused) ? focused : {};
3644
+
let phxUpdate = liveSocket2.binding(PHX_UPDATE);
3645
+
let phxViewportTop = liveSocket2.binding(PHX_VIEWPORT_TOP);
3646
+
let phxViewportBottom = liveSocket2.binding(PHX_VIEWPORT_BOTTOM);
3647
+
let phxTriggerExternal = liveSocket2.binding(PHX_TRIGGER_ACTION);
3648
+
let added = [];
3649
+
let updates = [];
3650
+
let appendPrependUpdates = [];
3651
+
let externalFormTriggered = null;
3652
+
function morph(targetContainer2, source, withChildren = this.withChildren) {
3653
+
let morphCallbacks = {
3654
+
// normally, we are running with childrenOnly, as the patch HTML for a LV
3655
+
// does not include the LV attrs (data-phx-session, etc.)
3656
+
// when we are patching a live component, we do want to patch the root element as well;
3657
+
// another case is the recursive patch of a stream item that was kept on reset (-> onBeforeNodeAdded)
3658
+
childrenOnly: targetContainer2.getAttribute(PHX_COMPONENT) === null && !withChildren,
3659
+
getNodeKey: (node) => {
3660
+
if (dom_default.isPhxDestroyed(node)) {
3661
+
return null;
3662
+
}
3663
+
if (isJoinPatch) {
3664
+
return node.id;
3665
+
}
3666
+
return node.id || node.getAttribute && node.getAttribute(PHX_MAGIC_ID);
3667
+
},
3668
+
// skip indexing from children when container is stream
3669
+
skipFromChildren: (from) => {
3670
+
return from.getAttribute(phxUpdate) === PHX_STREAM;
3671
+
},
3672
+
// tell morphdom how to add a child
3673
+
addChild: (parent, child) => {
3674
+
let { ref, streamAt } = this.getStreamInsert(child);
3675
+
if (ref === void 0) {
3676
+
return parent.appendChild(child);
3677
+
}
3678
+
this.setStreamRef(child, ref);
3679
+
if (streamAt === 0) {
3680
+
parent.insertAdjacentElement("afterbegin", child);
3681
+
} else if (streamAt === -1) {
3682
+
let lastChild = parent.lastElementChild;
3683
+
if (lastChild && !lastChild.hasAttribute(PHX_STREAM_REF)) {
3684
+
let nonStreamChild = Array.from(parent.children).find((c) => !c.hasAttribute(PHX_STREAM_REF));
3685
+
parent.insertBefore(child, nonStreamChild);
3686
+
} else {
3687
+
parent.appendChild(child);
3688
+
}
3689
+
} else if (streamAt > 0) {
3690
+
let sibling = Array.from(parent.children)[streamAt];
3691
+
parent.insertBefore(child, sibling);
3692
+
}
3693
+
},
3694
+
onBeforeNodeAdded: (el) => {
3695
+
dom_default.maintainPrivateHooks(el, el, phxViewportTop, phxViewportBottom);
3696
+
this.trackBefore("added", el);
3697
+
let morphedEl = el;
3698
+
if (this.streamComponentRestore[el.id]) {
3699
+
morphedEl = this.streamComponentRestore[el.id];
3700
+
delete this.streamComponentRestore[el.id];
3701
+
morph.call(this, morphedEl, el, true);
3702
+
}
3703
+
return morphedEl;
3704
+
},
3705
+
onNodeAdded: (el) => {
3706
+
if (el.getAttribute) {
3707
+
this.maybeReOrderStream(el, true);
3708
+
}
3709
+
if (el instanceof HTMLImageElement && el.srcset) {
3710
+
el.srcset = el.srcset;
3711
+
} else if (el instanceof HTMLVideoElement && el.autoplay) {
3712
+
el.play();
3713
+
}
3714
+
if (dom_default.isNowTriggerFormExternal(el, phxTriggerExternal)) {
3715
+
externalFormTriggered = el;
3716
+
}
3717
+
if (dom_default.isPhxChild(el) && view.ownsElement(el) || dom_default.isPhxSticky(el) && view.ownsElement(el.parentNode)) {
3718
+
this.trackAfter("phxChildAdded", el);
3719
+
}
3720
+
added.push(el);
3721
+
},
3722
+
onNodeDiscarded: (el) => this.onNodeDiscarded(el),
3723
+
onBeforeNodeDiscarded: (el) => {
3724
+
if (el.getAttribute && el.getAttribute(PHX_PRUNE) !== null) {
3725
+
return true;
3726
+
}
3727
+
if (el.parentElement !== null && el.id && dom_default.isPhxUpdate(el.parentElement, phxUpdate, [PHX_STREAM, "append", "prepend"])) {
3728
+
return false;
3729
+
}
3730
+
if (this.maybePendingRemove(el)) {
3731
+
return false;
3732
+
}
3733
+
if (this.skipCIDSibling(el)) {
3734
+
return false;
3735
+
}
3736
+
return true;
3737
+
},
3738
+
onElUpdated: (el) => {
3739
+
if (dom_default.isNowTriggerFormExternal(el, phxTriggerExternal)) {
3740
+
externalFormTriggered = el;
3741
+
}
3742
+
updates.push(el);
3743
+
this.maybeReOrderStream(el, false);
3744
+
},
3745
+
onBeforeElUpdated: (fromEl, toEl) => {
3746
+
if (fromEl.id && fromEl.isSameNode(targetContainer2) && fromEl.id !== toEl.id) {
3747
+
morphCallbacks.onNodeDiscarded(fromEl);
3748
+
fromEl.replaceWith(toEl);
3749
+
return morphCallbacks.onNodeAdded(toEl);
3750
+
}
3751
+
dom_default.syncPendingAttrs(fromEl, toEl);
3752
+
dom_default.maintainPrivateHooks(fromEl, toEl, phxViewportTop, phxViewportBottom);
3753
+
dom_default.cleanChildNodes(toEl, phxUpdate);
3754
+
if (this.skipCIDSibling(toEl)) {
3755
+
this.maybeReOrderStream(fromEl);
3756
+
return false;
3757
+
}
3758
+
if (dom_default.isPhxSticky(fromEl)) {
3759
+
[PHX_SESSION, PHX_STATIC, PHX_ROOT_ID].map((attr) => [attr, fromEl.getAttribute(attr), toEl.getAttribute(attr)]).forEach(([attr, fromVal, toVal]) => {
3760
+
if (toVal && fromVal !== toVal) {
3761
+
fromEl.setAttribute(attr, toVal);
3762
+
}
3763
+
});
3764
+
return false;
3765
+
}
3766
+
if (dom_default.isIgnored(fromEl, phxUpdate) || fromEl.form && fromEl.form.isSameNode(externalFormTriggered)) {
3767
+
this.trackBefore("updated", fromEl, toEl);
3768
+
dom_default.mergeAttrs(fromEl, toEl, { isIgnored: dom_default.isIgnored(fromEl, phxUpdate) });
3769
+
updates.push(fromEl);
3770
+
dom_default.applyStickyOperations(fromEl);
3771
+
return false;
3772
+
}
3773
+
if (fromEl.type === "number" && (fromEl.validity && fromEl.validity.badInput)) {
3774
+
return false;
3775
+
}
3776
+
let isFocusedFormEl = focused && fromEl.isSameNode(focused) && dom_default.isFormInput(fromEl);
3777
+
let focusedSelectChanged = isFocusedFormEl && this.isChangedSelect(fromEl, toEl);
3778
+
if (fromEl.hasAttribute(PHX_REF_SRC)) {
3779
+
const ref = new ElementRef(fromEl);
3780
+
if (ref.lockRef && (!this.undoRef || !ref.isLockUndoneBy(this.undoRef))) {
3781
+
if (dom_default.isUploadInput(fromEl)) {
3782
+
dom_default.mergeAttrs(fromEl, toEl, { isIgnored: true });
3783
+
this.trackBefore("updated", fromEl, toEl);
3784
+
updates.push(fromEl);
3785
+
}
3786
+
dom_default.applyStickyOperations(fromEl);
3787
+
let isLocked = fromEl.hasAttribute(PHX_REF_LOCK);
3788
+
let clone2 = isLocked ? dom_default.private(fromEl, PHX_REF_LOCK) || fromEl.cloneNode(true) : null;
3789
+
if (clone2) {
3790
+
dom_default.putPrivate(fromEl, PHX_REF_LOCK, clone2);
3791
+
if (!isFocusedFormEl) {
3792
+
fromEl = clone2;
3793
+
}
3794
+
}
3795
+
}
3796
+
}
3797
+
if (dom_default.isPhxChild(toEl)) {
3798
+
let prevSession = fromEl.getAttribute(PHX_SESSION);
3799
+
dom_default.mergeAttrs(fromEl, toEl, { exclude: [PHX_STATIC] });
3800
+
if (prevSession !== "") {
3801
+
fromEl.setAttribute(PHX_SESSION, prevSession);
3802
+
}
3803
+
fromEl.setAttribute(PHX_ROOT_ID, this.rootID);
3804
+
dom_default.applyStickyOperations(fromEl);
3805
+
return false;
3806
+
}
3807
+
if (this.undoRef && dom_default.private(toEl, PHX_REF_LOCK)) {
3808
+
dom_default.putPrivate(fromEl, PHX_REF_LOCK, dom_default.private(toEl, PHX_REF_LOCK));
3809
+
}
3810
+
dom_default.copyPrivates(toEl, fromEl);
3811
+
if (isFocusedFormEl && fromEl.type !== "hidden" && !focusedSelectChanged) {
3812
+
this.trackBefore("updated", fromEl, toEl);
3813
+
dom_default.mergeFocusedInput(fromEl, toEl);
3814
+
dom_default.syncAttrsToProps(fromEl);
3815
+
updates.push(fromEl);
3816
+
dom_default.applyStickyOperations(fromEl);
3817
+
return false;
3818
+
} else {
3819
+
if (focusedSelectChanged) {
3820
+
fromEl.blur();
3821
+
}
3822
+
if (dom_default.isPhxUpdate(toEl, phxUpdate, ["append", "prepend"])) {
3823
+
appendPrependUpdates.push(new DOMPostMorphRestorer(fromEl, toEl, toEl.getAttribute(phxUpdate)));
3824
+
}
3825
+
dom_default.syncAttrsToProps(toEl);
3826
+
dom_default.applyStickyOperations(toEl);
3827
+
this.trackBefore("updated", fromEl, toEl);
3828
+
return fromEl;
3829
+
}
3830
+
}
3831
+
};
3832
+
morphdom_esm_default(targetContainer2, source, morphCallbacks);
3833
+
}
3834
+
this.trackBefore("added", container);
3835
+
this.trackBefore("updated", container, container);
3836
+
liveSocket2.time("morphdom", () => {
3837
+
this.streams.forEach(([ref, inserts, deleteIds, reset]) => {
3838
+
inserts.forEach(([key, streamAt, limit]) => {
3839
+
this.streamInserts[key] = { ref, streamAt, limit, reset };
3840
+
});
3841
+
if (reset !== void 0) {
3842
+
dom_default.all(container, `[${PHX_STREAM_REF}="${ref}"]`, (child) => {
3843
+
this.removeStreamChildElement(child);
3844
+
});
3845
+
}
3846
+
deleteIds.forEach((id) => {
3847
+
let child = container.querySelector(`[id="${id}"]`);
3848
+
if (child) {
3849
+
this.removeStreamChildElement(child);
3850
+
}
3851
+
});
3852
+
});
3853
+
if (isJoinPatch) {
3854
+
dom_default.all(this.container, `[${phxUpdate}=${PHX_STREAM}]`).filter((el) => this.view.ownsElement(el)).forEach((el) => {
3855
+
Array.from(el.children).forEach((child) => {
3856
+
this.removeStreamChildElement(child, true);
3857
+
});
3858
+
});
3859
+
}
3860
+
morph.call(this, targetContainer, html);
3861
+
});
3862
+
if (liveSocket2.isDebugEnabled()) {
3863
+
detectDuplicateIds();
3864
+
detectInvalidStreamInserts(this.streamInserts);
3865
+
Array.from(document.querySelectorAll("input[name=id]")).forEach((node) => {
3866
+
if (node.form) {
3867
+
console.error('Detected an input with name="id" inside a form! This will cause problems when patching the DOM.\n', node);
3868
+
}
3869
+
});
3870
+
}
3871
+
if (appendPrependUpdates.length > 0) {
3872
+
liveSocket2.time("post-morph append/prepend restoration", () => {
3873
+
appendPrependUpdates.forEach((update) => update.perform());
3874
+
});
3875
+
}
3876
+
liveSocket2.silenceEvents(() => dom_default.restoreFocus(focused, selectionStart, selectionEnd));
3877
+
dom_default.dispatchEvent(document, "phx:update");
3878
+
added.forEach((el) => this.trackAfter("added", el));
3879
+
updates.forEach((el) => this.trackAfter("updated", el));
3880
+
this.transitionPendingRemoves();
3881
+
if (externalFormTriggered) {
3882
+
liveSocket2.unload();
3883
+
const submitter = dom_default.private(externalFormTriggered, "submitter");
3884
+
if (submitter && submitter.name && targetContainer.contains(submitter)) {
3885
+
const input = document.createElement("input");
3886
+
input.type = "hidden";
3887
+
const formId = submitter.getAttribute("form");
3888
+
if (formId) {
3889
+
input.setAttribute("form", formId);
3890
+
}
3891
+
input.name = submitter.name;
3892
+
input.value = submitter.value;
3893
+
submitter.parentElement.insertBefore(input, submitter);
3894
+
}
3895
+
Object.getPrototypeOf(externalFormTriggered).submit.call(externalFormTriggered);
3896
+
}
3897
+
return true;
3898
+
}
3899
+
onNodeDiscarded(el) {
3900
+
if (dom_default.isPhxChild(el) || dom_default.isPhxSticky(el)) {
3901
+
this.liveSocket.destroyViewByEl(el);
3902
+
}
3903
+
this.trackAfter("discarded", el);
3904
+
}
3905
+
maybePendingRemove(node) {
3906
+
if (node.getAttribute && node.getAttribute(this.phxRemove) !== null) {
3907
+
this.pendingRemoves.push(node);
3908
+
return true;
3909
+
} else {
3910
+
return false;
3911
+
}
3912
+
}
3913
+
removeStreamChildElement(child, force = false) {
3914
+
if (!force && !this.view.ownsElement(child)) {
3915
+
return;
3916
+
}
3917
+
if (this.streamInserts[child.id]) {
3918
+
this.streamComponentRestore[child.id] = child;
3919
+
child.remove();
3920
+
} else {
3921
+
if (!this.maybePendingRemove(child)) {
3922
+
child.remove();
3923
+
this.onNodeDiscarded(child);
3924
+
}
3925
+
}
3926
+
}
3927
+
getStreamInsert(el) {
3928
+
let insert = el.id ? this.streamInserts[el.id] : {};
3929
+
return insert || {};
3930
+
}
3931
+
setStreamRef(el, ref) {
3932
+
dom_default.putSticky(el, PHX_STREAM_REF, (el2) => el2.setAttribute(PHX_STREAM_REF, ref));
3933
+
}
3934
+
maybeReOrderStream(el, isNew) {
3935
+
let { ref, streamAt, reset } = this.getStreamInsert(el);
3936
+
if (streamAt === void 0) {
3937
+
return;
3938
+
}
3939
+
this.setStreamRef(el, ref);
3940
+
if (!reset && !isNew) {
3941
+
return;
3942
+
}
3943
+
if (!el.parentElement) {
3944
+
return;
3945
+
}
3946
+
if (streamAt === 0) {
3947
+
el.parentElement.insertBefore(el, el.parentElement.firstElementChild);
3948
+
} else if (streamAt > 0) {
3949
+
let children = Array.from(el.parentElement.children);
3950
+
let oldIndex = children.indexOf(el);
3951
+
if (streamAt >= children.length - 1) {
3952
+
el.parentElement.appendChild(el);
3953
+
} else {
3954
+
let sibling = children[streamAt];
3955
+
if (oldIndex > streamAt) {
3956
+
el.parentElement.insertBefore(el, sibling);
3957
+
} else {
3958
+
el.parentElement.insertBefore(el, sibling.nextElementSibling);
3959
+
}
3960
+
}
3961
+
}
3962
+
this.maybeLimitStream(el);
3963
+
}
3964
+
maybeLimitStream(el) {
3965
+
let { limit } = this.getStreamInsert(el);
3966
+
let children = limit !== null && Array.from(el.parentElement.children);
3967
+
if (limit && limit < 0 && children.length > limit * -1) {
3968
+
children.slice(0, children.length + limit).forEach((child) => this.removeStreamChildElement(child));
3969
+
} else if (limit && limit >= 0 && children.length > limit) {
3970
+
children.slice(limit).forEach((child) => this.removeStreamChildElement(child));
3971
+
}
3972
+
}
3973
+
transitionPendingRemoves() {
3974
+
let { pendingRemoves, liveSocket: liveSocket2 } = this;
3975
+
if (pendingRemoves.length > 0) {
3976
+
liveSocket2.transitionRemoves(pendingRemoves, () => {
3977
+
pendingRemoves.forEach((el) => {
3978
+
let child = dom_default.firstPhxChild(el);
3979
+
if (child) {
3980
+
liveSocket2.destroyViewByEl(child);
3981
+
}
3982
+
el.remove();
3983
+
});
3984
+
this.trackAfter("transitionsDiscarded", pendingRemoves);
3985
+
});
3986
+
}
3987
+
}
3988
+
isChangedSelect(fromEl, toEl) {
3989
+
if (!(fromEl instanceof HTMLSelectElement) || fromEl.multiple) {
3990
+
return false;
3991
+
}
3992
+
if (fromEl.options.length !== toEl.options.length) {
3993
+
return true;
3994
+
}
3995
+
toEl.value = fromEl.value;
3996
+
return !fromEl.isEqualNode(toEl);
3997
+
}
3998
+
isCIDPatch() {
3999
+
return this.cidPatch;
4000
+
}
4001
+
skipCIDSibling(el) {
4002
+
return el.nodeType === Node.ELEMENT_NODE && el.hasAttribute(PHX_SKIP);
4003
+
}
4004
+
targetCIDContainer(html) {
4005
+
if (!this.isCIDPatch()) {
4006
+
return;
4007
+
}
4008
+
let [first, ...rest] = dom_default.findComponentNodeList(this.container, this.targetCID);
4009
+
if (rest.length === 0 && dom_default.childNodeLength(html) === 1) {
4010
+
return first;
4011
+
} else {
4012
+
return first && first.parentNode;
4013
+
}
4014
+
}
4015
+
indexOf(parent, child) {
4016
+
return Array.from(parent.children).indexOf(child);
4017
+
}
4018
+
};
4019
+
var VOID_TAGS = /* @__PURE__ */ new Set([
4020
+
"area",
4021
+
"base",
4022
+
"br",
4023
+
"col",
4024
+
"command",
4025
+
"embed",
4026
+
"hr",
4027
+
"img",
4028
+
"input",
4029
+
"keygen",
4030
+
"link",
4031
+
"meta",
4032
+
"param",
4033
+
"source",
4034
+
"track",
4035
+
"wbr"
4036
+
]);
4037
+
var quoteChars = /* @__PURE__ */ new Set(["'", '"']);
4038
+
var modifyRoot = (html, attrs, clearInnerHTML) => {
4039
+
let i = 0;
4040
+
let insideComment = false;
4041
+
let beforeTag, afterTag, tag, tagNameEndsAt, id, newHTML;
4042
+
let lookahead = html.match(/^(\s*(?:<!--.*?-->\s*)*)<([^\s\/>]+)/);
4043
+
if (lookahead === null) {
4044
+
throw new Error(`malformed html ${html}`);
4045
+
}
4046
+
i = lookahead[0].length;
4047
+
beforeTag = lookahead[1];
4048
+
tag = lookahead[2];
4049
+
tagNameEndsAt = i;
4050
+
for (i; i < html.length; i++) {
4051
+
if (html.charAt(i) === ">") {
4052
+
break;
4053
+
}
4054
+
if (html.charAt(i) === "=") {
4055
+
let isId = html.slice(i - 3, i) === " id";
4056
+
i++;
4057
+
let char = html.charAt(i);
4058
+
if (quoteChars.has(char)) {
4059
+
let attrStartsAt = i;
4060
+
i++;
4061
+
for (i; i < html.length; i++) {
4062
+
if (html.charAt(i) === char) {
4063
+
break;
4064
+
}
4065
+
}
4066
+
if (isId) {
4067
+
id = html.slice(attrStartsAt + 1, i);
4068
+
break;
4069
+
}
4070
+
}
4071
+
}
4072
+
}
4073
+
let closeAt = html.length - 1;
4074
+
insideComment = false;
4075
+
while (closeAt >= beforeTag.length + tag.length) {
4076
+
let char = html.charAt(closeAt);
4077
+
if (insideComment) {
4078
+
if (char === "-" && html.slice(closeAt - 3, closeAt) === "<!-") {
4079
+
insideComment = false;
4080
+
closeAt -= 4;
4081
+
} else {
4082
+
closeAt -= 1;
4083
+
}
4084
+
} else if (char === ">" && html.slice(closeAt - 2, closeAt) === "--") {
4085
+
insideComment = true;
4086
+
closeAt -= 3;
4087
+
} else if (char === ">") {
4088
+
break;
4089
+
} else {
4090
+
closeAt -= 1;
4091
+
}
4092
+
}
4093
+
afterTag = html.slice(closeAt + 1, html.length);
4094
+
let attrsStr = Object.keys(attrs).map((attr) => attrs[attr] === true ? attr : `${attr}="${attrs[attr]}"`).join(" ");
4095
+
if (clearInnerHTML) {
4096
+
let idAttrStr = id ? ` id="${id}"` : "";
4097
+
if (VOID_TAGS.has(tag)) {
4098
+
newHTML = `<${tag}${idAttrStr}${attrsStr === "" ? "" : " "}${attrsStr}/>`;
4099
+
} else {
4100
+
newHTML = `<${tag}${idAttrStr}${attrsStr === "" ? "" : " "}${attrsStr}></${tag}>`;
4101
+
}
4102
+
} else {
4103
+
let rest = html.slice(tagNameEndsAt, closeAt + 1);
4104
+
newHTML = `<${tag}${attrsStr === "" ? "" : " "}${attrsStr}${rest}`;
4105
+
}
4106
+
return [newHTML, beforeTag, afterTag];
4107
+
};
4108
+
var Rendered = class {
4109
+
static extract(diff) {
4110
+
let { [REPLY]: reply, [EVENTS]: events, [TITLE]: title } = diff;
4111
+
delete diff[REPLY];
4112
+
delete diff[EVENTS];
4113
+
delete diff[TITLE];
4114
+
return { diff, title, reply: reply || null, events: events || [] };
4115
+
}
4116
+
constructor(viewId, rendered) {
4117
+
this.viewId = viewId;
4118
+
this.rendered = {};
4119
+
this.magicId = 0;
4120
+
this.mergeDiff(rendered);
4121
+
}
4122
+
parentViewId() {
4123
+
return this.viewId;
4124
+
}
4125
+
toString(onlyCids) {
4126
+
let [str, streams] = this.recursiveToString(this.rendered, this.rendered[COMPONENTS], onlyCids, true, {});
4127
+
return [str, streams];
4128
+
}
4129
+
recursiveToString(rendered, components = rendered[COMPONENTS], onlyCids, changeTracking, rootAttrs) {
4130
+
onlyCids = onlyCids ? new Set(onlyCids) : null;
4131
+
let output = { buffer: "", components, onlyCids, streams: /* @__PURE__ */ new Set() };
4132
+
this.toOutputBuffer(rendered, null, output, changeTracking, rootAttrs);
4133
+
return [output.buffer, output.streams];
4134
+
}
4135
+
componentCIDs(diff) {
4136
+
return Object.keys(diff[COMPONENTS] || {}).map((i) => parseInt(i));
4137
+
}
4138
+
isComponentOnlyDiff(diff) {
4139
+
if (!diff[COMPONENTS]) {
4140
+
return false;
4141
+
}
4142
+
return Object.keys(diff).length === 1;
4143
+
}
4144
+
getComponent(diff, cid) {
4145
+
return diff[COMPONENTS][cid];
4146
+
}
4147
+
resetRender(cid) {
4148
+
if (this.rendered[COMPONENTS][cid]) {
4149
+
this.rendered[COMPONENTS][cid].reset = true;
4150
+
}
4151
+
}
4152
+
mergeDiff(diff) {
4153
+
let newc = diff[COMPONENTS];
4154
+
let cache = {};
4155
+
delete diff[COMPONENTS];
4156
+
this.rendered = this.mutableMerge(this.rendered, diff);
4157
+
this.rendered[COMPONENTS] = this.rendered[COMPONENTS] || {};
4158
+
if (newc) {
4159
+
let oldc = this.rendered[COMPONENTS];
4160
+
for (let cid in newc) {
4161
+
newc[cid] = this.cachedFindComponent(cid, newc[cid], oldc, newc, cache);
4162
+
}
4163
+
for (let cid in newc) {
4164
+
oldc[cid] = newc[cid];
4165
+
}
4166
+
diff[COMPONENTS] = newc;
4167
+
}
4168
+
}
4169
+
cachedFindComponent(cid, cdiff, oldc, newc, cache) {
4170
+
if (cache[cid]) {
4171
+
return cache[cid];
4172
+
} else {
4173
+
let ndiff, stat, scid = cdiff[STATIC];
4174
+
if (isCid(scid)) {
4175
+
let tdiff;
4176
+
if (scid > 0) {
4177
+
tdiff = this.cachedFindComponent(scid, newc[scid], oldc, newc, cache);
4178
+
} else {
4179
+
tdiff = oldc[-scid];
4180
+
}
4181
+
stat = tdiff[STATIC];
4182
+
ndiff = this.cloneMerge(tdiff, cdiff, true);
4183
+
ndiff[STATIC] = stat;
4184
+
} else {
4185
+
ndiff = cdiff[STATIC] !== void 0 || oldc[cid] === void 0 ? cdiff : this.cloneMerge(oldc[cid], cdiff, false);
4186
+
}
4187
+
cache[cid] = ndiff;
4188
+
return ndiff;
4189
+
}
4190
+
}
4191
+
mutableMerge(target, source) {
4192
+
if (source[STATIC] !== void 0) {
4193
+
return source;
4194
+
} else {
4195
+
this.doMutableMerge(target, source);
4196
+
return target;
4197
+
}
4198
+
}
4199
+
doMutableMerge(target, source) {
4200
+
for (let key in source) {
4201
+
let val = source[key];
4202
+
let targetVal = target[key];
4203
+
let isObjVal = isObject(val);
4204
+
if (isObjVal && val[STATIC] === void 0 && isObject(targetVal)) {
4205
+
this.doMutableMerge(targetVal, val);
4206
+
} else {
4207
+
target[key] = val;
4208
+
}
4209
+
}
4210
+
if (target[ROOT]) {
4211
+
target.newRender = true;
4212
+
}
4213
+
}
4214
+
// Merges cid trees together, copying statics from source tree.
4215
+
//
4216
+
// The `pruneMagicId` is passed to control pruning the magicId of the
4217
+
// target. We must always prune the magicId when we are sharing statics
4218
+
// from another component. If not pruning, we replicate the logic from
4219
+
// mutableMerge, where we set newRender to true if there is a root
4220
+
// (effectively forcing the new version to be rendered instead of skipped)
4221
+
//
4222
+
cloneMerge(target, source, pruneMagicId) {
4223
+
let merged = __spreadValues(__spreadValues({}, target), source);
4224
+
for (let key in merged) {
4225
+
let val = source[key];
4226
+
let targetVal = target[key];
4227
+
if (isObject(val) && val[STATIC] === void 0 && isObject(targetVal)) {
4228
+
merged[key] = this.cloneMerge(targetVal, val, pruneMagicId);
4229
+
} else if (val === void 0 && isObject(targetVal)) {
4230
+
merged[key] = this.cloneMerge(targetVal, {}, pruneMagicId);
4231
+
}
4232
+
}
4233
+
if (pruneMagicId) {
4234
+
delete merged.magicId;
4235
+
delete merged.newRender;
4236
+
} else if (target[ROOT]) {
4237
+
merged.newRender = true;
4238
+
}
4239
+
return merged;
4240
+
}
4241
+
componentToString(cid) {
4242
+
let [str, streams] = this.recursiveCIDToString(this.rendered[COMPONENTS], cid, null);
4243
+
let [strippedHTML, _before, _after] = modifyRoot(str, {});
4244
+
return [strippedHTML, streams];
4245
+
}
4246
+
pruneCIDs(cids) {
4247
+
cids.forEach((cid) => delete this.rendered[COMPONENTS][cid]);
4248
+
}
4249
+
// private
4250
+
get() {
4251
+
return this.rendered;
4252
+
}
4253
+
isNewFingerprint(diff = {}) {
4254
+
return !!diff[STATIC];
4255
+
}
4256
+
templateStatic(part, templates) {
4257
+
if (typeof part === "number") {
4258
+
return templates[part];
4259
+
} else {
4260
+
return part;
4261
+
}
4262
+
}
4263
+
nextMagicID() {
4264
+
this.magicId++;
4265
+
return `m${this.magicId}-${this.parentViewId()}`;
4266
+
}
4267
+
// Converts rendered tree to output buffer.
4268
+
//
4269
+
// changeTracking controls if we can apply the PHX_SKIP optimization.
4270
+
// It is disabled for comprehensions since we must re-render the entire collection
4271
+
// and no individual element is tracked inside the comprehension.
4272
+
toOutputBuffer(rendered, templates, output, changeTracking, rootAttrs = {}) {
4273
+
if (rendered[DYNAMICS]) {
4274
+
return this.comprehensionToBuffer(rendered, templates, output);
4275
+
}
4276
+
let { [STATIC]: statics } = rendered;
4277
+
statics = this.templateStatic(statics, templates);
4278
+
let isRoot = rendered[ROOT];
4279
+
let prevBuffer = output.buffer;
4280
+
if (isRoot) {
4281
+
output.buffer = "";
4282
+
}
4283
+
if (changeTracking && isRoot && !rendered.magicId) {
4284
+
rendered.newRender = true;
4285
+
rendered.magicId = this.nextMagicID();
4286
+
}
4287
+
output.buffer += statics[0];
4288
+
for (let i = 1; i < statics.length; i++) {
4289
+
this.dynamicToBuffer(rendered[i - 1], templates, output, changeTracking);
4290
+
output.buffer += statics[i];
4291
+
}
4292
+
if (isRoot) {
4293
+
let skip = false;
4294
+
let attrs;
4295
+
if (changeTracking || rendered.magicId) {
4296
+
skip = changeTracking && !rendered.newRender;
4297
+
attrs = __spreadValues({ [PHX_MAGIC_ID]: rendered.magicId }, rootAttrs);
4298
+
} else {
4299
+
attrs = rootAttrs;
4300
+
}
4301
+
if (skip) {
4302
+
attrs[PHX_SKIP] = true;
4303
+
}
4304
+
let [newRoot, commentBefore, commentAfter] = modifyRoot(output.buffer, attrs, skip);
4305
+
rendered.newRender = false;
4306
+
output.buffer = prevBuffer + commentBefore + newRoot + commentAfter;
4307
+
}
4308
+
}
4309
+
comprehensionToBuffer(rendered, templates, output) {
4310
+
let { [DYNAMICS]: dynamics, [STATIC]: statics, [STREAM]: stream } = rendered;
4311
+
let [_ref, _inserts, deleteIds, reset] = stream || [null, {}, [], null];
4312
+
statics = this.templateStatic(statics, templates);
4313
+
let compTemplates = templates || rendered[TEMPLATES];
4314
+
for (let d = 0; d < dynamics.length; d++) {
4315
+
let dynamic = dynamics[d];
4316
+
output.buffer += statics[0];
4317
+
for (let i = 1; i < statics.length; i++) {
4318
+
let changeTracking = false;
4319
+
this.dynamicToBuffer(dynamic[i - 1], compTemplates, output, changeTracking);
4320
+
output.buffer += statics[i];
4321
+
}
4322
+
}
4323
+
if (stream !== void 0 && (rendered[DYNAMICS].length > 0 || deleteIds.length > 0 || reset)) {
4324
+
delete rendered[STREAM];
4325
+
rendered[DYNAMICS] = [];
4326
+
output.streams.add(stream);
4327
+
}
4328
+
}
4329
+
dynamicToBuffer(rendered, templates, output, changeTracking) {
4330
+
if (typeof rendered === "number") {
4331
+
let [str, streams] = this.recursiveCIDToString(output.components, rendered, output.onlyCids);
4332
+
output.buffer += str;
4333
+
output.streams = /* @__PURE__ */ new Set([...output.streams, ...streams]);
4334
+
} else if (isObject(rendered)) {
4335
+
this.toOutputBuffer(rendered, templates, output, changeTracking, {});
4336
+
} else {
4337
+
output.buffer += rendered;
4338
+
}
4339
+
}
4340
+
recursiveCIDToString(components, cid, onlyCids) {
4341
+
let component = components[cid] || logError(`no component for CID ${cid}`, components);
4342
+
let attrs = { [PHX_COMPONENT]: cid };
4343
+
let skip = onlyCids && !onlyCids.has(cid);
4344
+
component.newRender = !skip;
4345
+
component.magicId = `c${cid}-${this.parentViewId()}`;
4346
+
let changeTracking = !component.reset;
4347
+
let [html, streams] = this.recursiveToString(component, components, onlyCids, changeTracking, attrs);
4348
+
delete component.reset;
4349
+
return [html, streams];
4350
+
}
4351
+
};
4352
+
var focusStack = [];
4353
+
var default_transition_time = 200;
4354
+
var JS = {
4355
+
// private
4356
+
exec(e, eventType, phxEvent, view, sourceEl, defaults) {
4357
+
let [defaultKind, defaultArgs] = defaults || [null, { callback: defaults && defaults.callback }];
4358
+
let commands = phxEvent.charAt(0) === "[" ? JSON.parse(phxEvent) : [[defaultKind, defaultArgs]];
4359
+
commands.forEach(([kind, args]) => {
4360
+
if (kind === defaultKind) {
4361
+
args = __spreadValues(__spreadValues({}, defaultArgs), args);
4362
+
args.callback = args.callback || defaultArgs.callback;
4363
+
}
4364
+
this.filterToEls(view.liveSocket, sourceEl, args).forEach((el) => {
4365
+
this[`exec_${kind}`](e, eventType, phxEvent, view, sourceEl, el, args);
4366
+
});
4367
+
});
4368
+
},
4369
+
isVisible(el) {
4370
+
return !!(el.offsetWidth || el.offsetHeight || el.getClientRects().length > 0);
4371
+
},
4372
+
// returns true if any part of the element is inside the viewport
4373
+
isInViewport(el) {
4374
+
const rect = el.getBoundingClientRect();
4375
+
const windowHeight = window.innerHeight || document.documentElement.clientHeight;
4376
+
const windowWidth = window.innerWidth || document.documentElement.clientWidth;
4377
+
return rect.right > 0 && rect.bottom > 0 && rect.left < windowWidth && rect.top < windowHeight;
4378
+
},
4379
+
// private
4380
+
// commands
4381
+
exec_exec(e, eventType, phxEvent, view, sourceEl, el, { attr, to }) {
4382
+
let encodedJS = el.getAttribute(attr);
4383
+
if (!encodedJS) {
4384
+
throw new Error(`expected ${attr} to contain JS command on "${to}"`);
4385
+
}
4386
+
view.liveSocket.execJS(el, encodedJS, eventType);
4387
+
},
4388
+
exec_dispatch(e, eventType, phxEvent, view, sourceEl, el, { event, detail, bubbles }) {
4389
+
detail = detail || {};
4390
+
detail.dispatcher = sourceEl;
4391
+
dom_default.dispatchEvent(el, event, { detail, bubbles });
4392
+
},
4393
+
exec_push(e, eventType, phxEvent, view, sourceEl, el, args) {
4394
+
let { event, data, target, page_loading, loading, value, dispatcher, callback } = args;
4395
+
let pushOpts = { loading, value, target, page_loading: !!page_loading };
4396
+
let targetSrc = eventType === "change" && dispatcher ? dispatcher : sourceEl;
4397
+
let phxTarget = target || targetSrc.getAttribute(view.binding("target")) || targetSrc;
4398
+
const handler = (targetView, targetCtx) => {
4399
+
if (!targetView.isConnected()) {
4400
+
return;
4401
+
}
4402
+
if (eventType === "change") {
4403
+
let { newCid, _target } = args;
4404
+
_target = _target || (dom_default.isFormInput(sourceEl) ? sourceEl.name : void 0);
4405
+
if (_target) {
4406
+
pushOpts._target = _target;
4407
+
}
4408
+
targetView.pushInput(sourceEl, targetCtx, newCid, event || phxEvent, pushOpts, callback);
4409
+
} else if (eventType === "submit") {
4410
+
let { submitter } = args;
4411
+
targetView.submitForm(sourceEl, targetCtx, event || phxEvent, submitter, pushOpts, callback);
4412
+
} else {
4413
+
targetView.pushEvent(eventType, sourceEl, targetCtx, event || phxEvent, data, pushOpts, callback);
4414
+
}
4415
+
};
4416
+
if (args.targetView && args.targetCtx) {
4417
+
handler(args.targetView, args.targetCtx);
4418
+
} else {
4419
+
view.withinTargets(phxTarget, handler);
4420
+
}
4421
+
},
4422
+
exec_navigate(e, eventType, phxEvent, view, sourceEl, el, { href, replace }) {
4423
+
view.liveSocket.historyRedirect(e, href, replace ? "replace" : "push", null, sourceEl);
4424
+
},
4425
+
exec_patch(e, eventType, phxEvent, view, sourceEl, el, { href, replace }) {
4426
+
view.liveSocket.pushHistoryPatch(e, href, replace ? "replace" : "push", sourceEl);
4427
+
},
4428
+
exec_focus(e, eventType, phxEvent, view, sourceEl, el) {
4429
+
aria_default.attemptFocus(el);
4430
+
window.requestAnimationFrame(() => {
4431
+
window.requestAnimationFrame(() => aria_default.attemptFocus(el));
4432
+
});
4433
+
},
4434
+
exec_focus_first(e, eventType, phxEvent, view, sourceEl, el) {
4435
+
aria_default.focusFirstInteractive(el) || aria_default.focusFirst(el);
4436
+
window.requestAnimationFrame(() => {
4437
+
window.requestAnimationFrame(() => aria_default.focusFirstInteractive(el) || aria_default.focusFirst(el));
4438
+
});
4439
+
},
4440
+
exec_push_focus(e, eventType, phxEvent, view, sourceEl, el) {
4441
+
focusStack.push(el || sourceEl);
4442
+
},
4443
+
exec_pop_focus(_e, _eventType, _phxEvent, _view, _sourceEl, _el) {
4444
+
const el = focusStack.pop();
4445
+
if (el) {
4446
+
el.focus();
4447
+
window.requestAnimationFrame(() => {
4448
+
window.requestAnimationFrame(() => el.focus());
4449
+
});
4450
+
}
4451
+
},
4452
+
exec_add_class(e, eventType, phxEvent, view, sourceEl, el, { names, transition, time, blocking }) {
4453
+
this.addOrRemoveClasses(el, names, [], transition, time, view, blocking);
4454
+
},
4455
+
exec_remove_class(e, eventType, phxEvent, view, sourceEl, el, { names, transition, time, blocking }) {
4456
+
this.addOrRemoveClasses(el, [], names, transition, time, view, blocking);
4457
+
},
4458
+
exec_toggle_class(e, eventType, phxEvent, view, sourceEl, el, { names, transition, time, blocking }) {
4459
+
this.toggleClasses(el, names, transition, time, view, blocking);
4460
+
},
4461
+
exec_toggle_attr(e, eventType, phxEvent, view, sourceEl, el, { attr: [attr, val1, val2] }) {
4462
+
this.toggleAttr(el, attr, val1, val2);
4463
+
},
4464
+
exec_transition(e, eventType, phxEvent, view, sourceEl, el, { time, transition, blocking }) {
4465
+
this.addOrRemoveClasses(el, [], [], transition, time, view, blocking);
4466
+
},
4467
+
exec_toggle(e, eventType, phxEvent, view, sourceEl, el, { display, ins, outs, time, blocking }) {
4468
+
this.toggle(eventType, view, el, display, ins, outs, time, blocking);
4469
+
},
4470
+
exec_show(e, eventType, phxEvent, view, sourceEl, el, { display, transition, time, blocking }) {
4471
+
this.show(eventType, view, el, display, transition, time, blocking);
4472
+
},
4473
+
exec_hide(e, eventType, phxEvent, view, sourceEl, el, { display, transition, time, blocking }) {
4474
+
this.hide(eventType, view, el, display, transition, time, blocking);
4475
+
},
4476
+
exec_set_attr(e, eventType, phxEvent, view, sourceEl, el, { attr: [attr, val] }) {
4477
+
this.setOrRemoveAttrs(el, [[attr, val]], []);
4478
+
},
4479
+
exec_remove_attr(e, eventType, phxEvent, view, sourceEl, el, { attr }) {
4480
+
this.setOrRemoveAttrs(el, [], [attr]);
4481
+
},
4482
+
// utils for commands
4483
+
show(eventType, view, el, display, transition, time, blocking) {
4484
+
if (!this.isVisible(el)) {
4485
+
this.toggle(eventType, view, el, display, transition, null, time, blocking);
4486
+
}
4487
+
},
4488
+
hide(eventType, view, el, display, transition, time, blocking) {
4489
+
if (this.isVisible(el)) {
4490
+
this.toggle(eventType, view, el, display, null, transition, time, blocking);
4491
+
}
4492
+
},
4493
+
toggle(eventType, view, el, display, ins, outs, time, blocking) {
4494
+
time = time || default_transition_time;
4495
+
let [inClasses, inStartClasses, inEndClasses] = ins || [[], [], []];
4496
+
let [outClasses, outStartClasses, outEndClasses] = outs || [[], [], []];
4497
+
if (inClasses.length > 0 || outClasses.length > 0) {
4498
+
if (this.isVisible(el)) {
4499
+
let onStart = () => {
4500
+
this.addOrRemoveClasses(el, outStartClasses, inClasses.concat(inStartClasses).concat(inEndClasses));
4501
+
window.requestAnimationFrame(() => {
4502
+
this.addOrRemoveClasses(el, outClasses, []);
4503
+
window.requestAnimationFrame(() => this.addOrRemoveClasses(el, outEndClasses, outStartClasses));
4504
+
});
4505
+
};
4506
+
let onEnd = () => {
4507
+
this.addOrRemoveClasses(el, [], outClasses.concat(outEndClasses));
4508
+
dom_default.putSticky(el, "toggle", (currentEl) => currentEl.style.display = "none");
4509
+
el.dispatchEvent(new Event("phx:hide-end"));
4510
+
};
4511
+
el.dispatchEvent(new Event("phx:hide-start"));
4512
+
if (blocking === false) {
4513
+
onStart();
4514
+
setTimeout(onEnd, time);
4515
+
} else {
4516
+
view.transition(time, onStart, onEnd);
4517
+
}
4518
+
} else {
4519
+
if (eventType === "remove") {
4520
+
return;
4521
+
}
4522
+
let onStart = () => {
4523
+
this.addOrRemoveClasses(el, inStartClasses, outClasses.concat(outStartClasses).concat(outEndClasses));
4524
+
const stickyDisplay = display || this.defaultDisplay(el);
4525
+
window.requestAnimationFrame(() => {
4526
+
this.addOrRemoveClasses(el, inClasses, []);
4527
+
window.requestAnimationFrame(() => {
4528
+
dom_default.putSticky(el, "toggle", (currentEl) => currentEl.style.display = stickyDisplay);
4529
+
this.addOrRemoveClasses(el, inEndClasses, inStartClasses);
4530
+
});
4531
+
});
4532
+
};
4533
+
let onEnd = () => {
4534
+
this.addOrRemoveClasses(el, [], inClasses.concat(inEndClasses));
4535
+
el.dispatchEvent(new Event("phx:show-end"));
4536
+
};
4537
+
el.dispatchEvent(new Event("phx:show-start"));
4538
+
if (blocking === false) {
4539
+
onStart();
4540
+
setTimeout(onEnd, time);
4541
+
} else {
4542
+
view.transition(time, onStart, onEnd);
4543
+
}
4544
+
}
4545
+
} else {
4546
+
if (this.isVisible(el)) {
4547
+
window.requestAnimationFrame(() => {
4548
+
el.dispatchEvent(new Event("phx:hide-start"));
4549
+
dom_default.putSticky(el, "toggle", (currentEl) => currentEl.style.display = "none");
4550
+
el.dispatchEvent(new Event("phx:hide-end"));
4551
+
});
4552
+
} else {
4553
+
window.requestAnimationFrame(() => {
4554
+
el.dispatchEvent(new Event("phx:show-start"));
4555
+
let stickyDisplay = display || this.defaultDisplay(el);
4556
+
dom_default.putSticky(el, "toggle", (currentEl) => currentEl.style.display = stickyDisplay);
4557
+
el.dispatchEvent(new Event("phx:show-end"));
4558
+
});
4559
+
}
4560
+
}
4561
+
},
4562
+
toggleClasses(el, classes, transition, time, view, blocking) {
4563
+
window.requestAnimationFrame(() => {
4564
+
let [prevAdds, prevRemoves] = dom_default.getSticky(el, "classes", [[], []]);
4565
+
let newAdds = classes.filter((name) => prevAdds.indexOf(name) < 0 && !el.classList.contains(name));
4566
+
let newRemoves = classes.filter((name) => prevRemoves.indexOf(name) < 0 && el.classList.contains(name));
4567
+
this.addOrRemoveClasses(el, newAdds, newRemoves, transition, time, view, blocking);
4568
+
});
4569
+
},
4570
+
toggleAttr(el, attr, val1, val2) {
4571
+
if (el.hasAttribute(attr)) {
4572
+
if (val2 !== void 0) {
4573
+
if (el.getAttribute(attr) === val1) {
4574
+
this.setOrRemoveAttrs(el, [[attr, val2]], []);
4575
+
} else {
4576
+
this.setOrRemoveAttrs(el, [[attr, val1]], []);
4577
+
}
4578
+
} else {
4579
+
this.setOrRemoveAttrs(el, [], [attr]);
4580
+
}
4581
+
} else {
4582
+
this.setOrRemoveAttrs(el, [[attr, val1]], []);
4583
+
}
4584
+
},
4585
+
addOrRemoveClasses(el, adds, removes, transition, time, view, blocking) {
4586
+
time = time || default_transition_time;
4587
+
let [transitionRun, transitionStart, transitionEnd] = transition || [[], [], []];
4588
+
if (transitionRun.length > 0) {
4589
+
let onStart = () => {
4590
+
this.addOrRemoveClasses(el, transitionStart, [].concat(transitionRun).concat(transitionEnd));
4591
+
window.requestAnimationFrame(() => {
4592
+
this.addOrRemoveClasses(el, transitionRun, []);
4593
+
window.requestAnimationFrame(() => this.addOrRemoveClasses(el, transitionEnd, transitionStart));
4594
+
});
4595
+
};
4596
+
let onDone = () => this.addOrRemoveClasses(el, adds.concat(transitionEnd), removes.concat(transitionRun).concat(transitionStart));
4597
+
if (blocking === false) {
4598
+
onStart();
4599
+
setTimeout(onDone, time);
4600
+
} else {
4601
+
view.transition(time, onStart, onDone);
4602
+
}
4603
+
return;
4604
+
}
4605
+
window.requestAnimationFrame(() => {
4606
+
let [prevAdds, prevRemoves] = dom_default.getSticky(el, "classes", [[], []]);
4607
+
let keepAdds = adds.filter((name) => prevAdds.indexOf(name) < 0 && !el.classList.contains(name));
4608
+
let keepRemoves = removes.filter((name) => prevRemoves.indexOf(name) < 0 && el.classList.contains(name));
4609
+
let newAdds = prevAdds.filter((name) => removes.indexOf(name) < 0).concat(keepAdds);
4610
+
let newRemoves = prevRemoves.filter((name) => adds.indexOf(name) < 0).concat(keepRemoves);
4611
+
dom_default.putSticky(el, "classes", (currentEl) => {
4612
+
currentEl.classList.remove(...newRemoves);
4613
+
currentEl.classList.add(...newAdds);
4614
+
return [newAdds, newRemoves];
4615
+
});
4616
+
});
4617
+
},
4618
+
setOrRemoveAttrs(el, sets, removes) {
4619
+
let [prevSets, prevRemoves] = dom_default.getSticky(el, "attrs", [[], []]);
4620
+
let alteredAttrs = sets.map(([attr, _val]) => attr).concat(removes);
4621
+
let newSets = prevSets.filter(([attr, _val]) => !alteredAttrs.includes(attr)).concat(sets);
4622
+
let newRemoves = prevRemoves.filter((attr) => !alteredAttrs.includes(attr)).concat(removes);
4623
+
dom_default.putSticky(el, "attrs", (currentEl) => {
4624
+
newRemoves.forEach((attr) => currentEl.removeAttribute(attr));
4625
+
newSets.forEach(([attr, val]) => currentEl.setAttribute(attr, val));
4626
+
return [newSets, newRemoves];
4627
+
});
4628
+
},
4629
+
hasAllClasses(el, classes) {
4630
+
return classes.every((name) => el.classList.contains(name));
4631
+
},
4632
+
isToggledOut(el, outClasses) {
4633
+
return !this.isVisible(el) || this.hasAllClasses(el, outClasses);
4634
+
},
4635
+
filterToEls(liveSocket2, sourceEl, { to }) {
4636
+
let defaultQuery = () => {
4637
+
if (typeof to === "string") {
4638
+
return document.querySelectorAll(to);
4639
+
} else if (to.closest) {
4640
+
let toEl = sourceEl.closest(to.closest);
4641
+
return toEl ? [toEl] : [];
4642
+
} else if (to.inner) {
4643
+
return sourceEl.querySelectorAll(to.inner);
4644
+
}
4645
+
};
4646
+
return to ? liveSocket2.jsQuerySelectorAll(sourceEl, to, defaultQuery) : [sourceEl];
4647
+
},
4648
+
defaultDisplay(el) {
4649
+
return { tr: "table-row", td: "table-cell" }[el.tagName.toLowerCase()] || "block";
4650
+
},
4651
+
transitionClasses(val) {
4652
+
if (!val) {
4653
+
return null;
4654
+
}
4655
+
let [trans, tStart, tEnd] = Array.isArray(val) ? val : [val.split(" "), [], []];
4656
+
trans = Array.isArray(trans) ? trans : trans.split(" ");
4657
+
tStart = Array.isArray(tStart) ? tStart : tStart.split(" ");
4658
+
tEnd = Array.isArray(tEnd) ? tEnd : tEnd.split(" ");
4659
+
return [trans, tStart, tEnd];
4660
+
}
4661
+
};
4662
+
var js_default = JS;
4663
+
var HOOK_ID = "hookId";
4664
+
var viewHookID = 1;
4665
+
var ViewHook = class {
4666
+
static makeID() {
4667
+
return viewHookID++;
4668
+
}
4669
+
static elementID(el) {
4670
+
return dom_default.private(el, HOOK_ID);
4671
+
}
4672
+
constructor(view, el, callbacks) {
4673
+
this.el = el;
4674
+
this.__attachView(view);
4675
+
this.__callbacks = callbacks;
4676
+
this.__listeners = /* @__PURE__ */ new Set();
4677
+
this.__isDisconnected = false;
4678
+
dom_default.putPrivate(this.el, HOOK_ID, this.constructor.makeID());
4679
+
for (let key in this.__callbacks) {
4680
+
this[key] = this.__callbacks[key];
4681
+
}
4682
+
}
4683
+
__attachView(view) {
4684
+
if (view) {
4685
+
this.__view = () => view;
4686
+
this.liveSocket = view.liveSocket;
4687
+
} else {
4688
+
this.__view = () => {
4689
+
throw new Error(`hook not yet attached to a live view: ${this.el.outerHTML}`);
4690
+
};
4691
+
this.liveSocket = null;
4692
+
}
4693
+
}
4694
+
__mounted() {
4695
+
this.mounted && this.mounted();
4696
+
}
4697
+
__updated() {
4698
+
this.updated && this.updated();
4699
+
}
4700
+
__beforeUpdate() {
4701
+
this.beforeUpdate && this.beforeUpdate();
4702
+
}
4703
+
__destroyed() {
4704
+
this.destroyed && this.destroyed();
4705
+
dom_default.deletePrivate(this.el, HOOK_ID);
4706
+
}
4707
+
__reconnected() {
4708
+
if (this.__isDisconnected) {
4709
+
this.__isDisconnected = false;
4710
+
this.reconnected && this.reconnected();
4711
+
}
4712
+
}
4713
+
__disconnected() {
4714
+
this.__isDisconnected = true;
4715
+
this.disconnected && this.disconnected();
4716
+
}
4717
+
/**
4718
+
* Binds the hook to JS commands.
4719
+
*
4720
+
* @param {ViewHook} hook - The ViewHook instance to bind.
4721
+
*
4722
+
* @returns {Object} An object with methods to manipulate the DOM and execute JavaScript.
4723
+
*/
4724
+
js() {
4725
+
let hook = this;
4726
+
return {
4727
+
/**
4728
+
* Executes encoded JavaScript in the context of the hook element.
4729
+
*
4730
+
* @param {string} encodedJS - The encoded JavaScript string to execute.
4731
+
*/
4732
+
exec(encodedJS) {
4733
+
hook.__view().liveSocket.execJS(hook.el, encodedJS, "hook");
4734
+
},
4735
+
/**
4736
+
* Shows an element.
4737
+
*
4738
+
* @param {HTMLElement} el - The element to show.
4739
+
* @param {Object} [opts={}] - Optional settings.
4740
+
* @param {string} [opts.display] - The CSS display value to set. Defaults "block".
4741
+
* @param {string} [opts.transition] - The CSS transition classes to set when showing.
4742
+
* @param {number} [opts.time] - The transition duration in milliseconds. Defaults 200.
4743
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4744
+
* Defaults `true`.
4745
+
*/
4746
+
show(el, opts = {}) {
4747
+
let owner = hook.__view().liveSocket.owner(el);
4748
+
js_default.show("hook", owner, el, opts.display, opts.transition, opts.time, opts.blocking);
4749
+
},
4750
+
/**
4751
+
* Hides an element.
4752
+
*
4753
+
* @param {HTMLElement} el - The element to hide.
4754
+
* @param {Object} [opts={}] - Optional settings.
4755
+
* @param {string} [opts.transition] - The CSS transition classes to set when hiding.
4756
+
* @param {number} [opts.time] - The transition duration in milliseconds. Defaults 200.
4757
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4758
+
* Defaults `true`.
4759
+
*/
4760
+
hide(el, opts = {}) {
4761
+
let owner = hook.__view().liveSocket.owner(el);
4762
+
js_default.hide("hook", owner, el, null, opts.transition, opts.time, opts.blocking);
4763
+
},
4764
+
/**
4765
+
* Toggles the visibility of an element.
4766
+
*
4767
+
* @param {HTMLElement} el - The element to toggle.
4768
+
* @param {Object} [opts={}] - Optional settings.
4769
+
* @param {string} [opts.display] - The CSS display value to set. Defaults "block".
4770
+
* @param {string} [opts.in] - The CSS transition classes for showing.
4771
+
* Accepts either the string of classes to apply when toggling in, or
4772
+
* a 3-tuple containing the transition class, the class to apply
4773
+
* to start the transition, and the ending transition class, such as:
4774
+
*
4775
+
* ["ease-out duration-300", "opacity-0", "opacity-100"]
4776
+
*
4777
+
* @param {string} [opts.out] - The CSS transition classes for hiding.
4778
+
* Accepts either string of classes to apply when toggling out, or
4779
+
* a 3-tuple containing the transition class, the class to apply
4780
+
* to start the transition, and the ending transition class, such as:
4781
+
*
4782
+
* ["ease-out duration-300", "opacity-100", "opacity-0"]
4783
+
*
4784
+
* @param {number} [opts.time] - The transition duration in milliseconds.
4785
+
*
4786
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4787
+
* Defaults `true`.
4788
+
*/
4789
+
toggle(el, opts = {}) {
4790
+
let owner = hook.__view().liveSocket.owner(el);
4791
+
opts.in = js_default.transitionClasses(opts.in);
4792
+
opts.out = js_default.transitionClasses(opts.out);
4793
+
js_default.toggle("hook", owner, el, opts.display, opts.in, opts.out, opts.time, opts.blocking);
4794
+
},
4795
+
/**
4796
+
* Adds CSS classes to an element.
4797
+
*
4798
+
* @param {HTMLElement} el - The element to add classes to.
4799
+
* @param {string|string[]} names - The class name(s) to add.
4800
+
* @param {Object} [opts={}] - Optional settings.
4801
+
* @param {string} [opts.transition] - The CSS transition property to set.
4802
+
* Accepts a string of classes to apply when adding classes or
4803
+
* a 3-tuple containing the transition class, the class to apply
4804
+
* to start the transition, and the ending transition class, such as:
4805
+
*
4806
+
* ["ease-out duration-300", "opacity-0", "opacity-100"]
4807
+
*
4808
+
* @param {number} [opts.time] - The transition duration in milliseconds.
4809
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4810
+
* Defaults `true`.
4811
+
*/
4812
+
addClass(el, names, opts = {}) {
4813
+
names = Array.isArray(names) ? names : names.split(" ");
4814
+
let owner = hook.__view().liveSocket.owner(el);
4815
+
js_default.addOrRemoveClasses(el, names, [], opts.transition, opts.time, owner, opts.blocking);
4816
+
},
4817
+
/**
4818
+
* Removes CSS classes from an element.
4819
+
*
4820
+
* @param {HTMLElement} el - The element to remove classes from.
4821
+
* @param {string|string[]} names - The class name(s) to remove.
4822
+
* @param {Object} [opts={}] - Optional settings.
4823
+
* @param {string} [opts.transition] - The CSS transition classes to set.
4824
+
* Accepts a string of classes to apply when removing classes or
4825
+
* a 3-tuple containing the transition class, the class to apply
4826
+
* to start the transition, and the ending transition class, such as:
4827
+
*
4828
+
* ["ease-out duration-300", "opacity-100", "opacity-0"]
4829
+
*
4830
+
* @param {number} [opts.time] - The transition duration in milliseconds.
4831
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4832
+
* Defaults `true`.
4833
+
*/
4834
+
removeClass(el, names, opts = {}) {
4835
+
opts.transition = js_default.transitionClasses(opts.transition);
4836
+
names = Array.isArray(names) ? names : names.split(" ");
4837
+
let owner = hook.__view().liveSocket.owner(el);
4838
+
js_default.addOrRemoveClasses(el, [], names, opts.transition, opts.time, owner, opts.blocking);
4839
+
},
4840
+
/**
4841
+
* Toggles CSS classes on an element.
4842
+
*
4843
+
* @param {HTMLElement} el - The element to toggle classes on.
4844
+
* @param {string|string[]} names - The class name(s) to toggle.
4845
+
* @param {Object} [opts={}] - Optional settings.
4846
+
* @param {string} [opts.transition] - The CSS transition classes to set.
4847
+
* Accepts a string of classes to apply when toggling classes or
4848
+
* a 3-tuple containing the transition class, the class to apply
4849
+
* to start the transition, and the ending transition class, such as:
4850
+
*
4851
+
* ["ease-out duration-300", "opacity-100", "opacity-0"]
4852
+
*
4853
+
* @param {number} [opts.time] - The transition duration in milliseconds.
4854
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4855
+
* Defaults `true`.
4856
+
*/
4857
+
toggleClass(el, names, opts = {}) {
4858
+
opts.transition = js_default.transitionClasses(opts.transition);
4859
+
names = Array.isArray(names) ? names : names.split(" ");
4860
+
let owner = hook.__view().liveSocket.owner(el);
4861
+
js_default.toggleClasses(el, names, opts.transition, opts.time, owner, opts.blocking);
4862
+
},
4863
+
/**
4864
+
* Applies a CSS transition to an element.
4865
+
*
4866
+
* @param {HTMLElement} el - The element to apply the transition to.
4867
+
* @param {string|string[]} transition - The transition class(es) to apply.
4868
+
* Accepts a string of classes to apply when transitioning or
4869
+
* a 3-tuple containing the transition class, the class to apply
4870
+
* to start the transition, and the ending transition class, such as:
4871
+
*
4872
+
* ["ease-out duration-300", "opacity-100", "opacity-0"]
4873
+
*
4874
+
* @param {Object} [opts={}] - Optional settings.
4875
+
* @param {number} [opts.time] - The transition duration in milliseconds.
4876
+
* @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.
4877
+
* Defaults `true`.
4878
+
*/
4879
+
transition(el, transition, opts = {}) {
4880
+
let owner = hook.__view().liveSocket.owner(el);
4881
+
js_default.addOrRemoveClasses(el, [], [], js_default.transitionClasses(transition), opts.time, owner, opts.blocking);
4882
+
},
4883
+
/**
4884
+
* Sets an attribute on an element.
4885
+
*
4886
+
* @param {HTMLElement} el - The element to set the attribute on.
4887
+
* @param {string} attr - The attribute name to set.
4888
+
* @param {string} val - The value to set for the attribute.
4889
+
*/
4890
+
setAttribute(el, attr, val) {
4891
+
js_default.setOrRemoveAttrs(el, [[attr, val]], []);
4892
+
},
4893
+
/**
4894
+
* Removes an attribute from an element.
4895
+
*
4896
+
* @param {HTMLElement} el - The element to remove the attribute from.
4897
+
* @param {string} attr - The attribute name to remove.
4898
+
*/
4899
+
removeAttribute(el, attr) {
4900
+
js_default.setOrRemoveAttrs(el, [], [attr]);
4901
+
},
4902
+
/**
4903
+
* Toggles an attribute on an element between two values.
4904
+
*
4905
+
* @param {HTMLElement} el - The element to toggle the attribute on.
4906
+
* @param {string} attr - The attribute name to toggle.
4907
+
* @param {string} val1 - The first value to toggle between.
4908
+
* @param {string} val2 - The second value to toggle between.
4909
+
*/
4910
+
toggleAttribute(el, attr, val1, val2) {
4911
+
js_default.toggleAttr(el, attr, val1, val2);
4912
+
}
4913
+
};
4914
+
}
4915
+
pushEvent(event, payload = {}, onReply) {
4916
+
if (onReply === void 0) {
4917
+
return new Promise((resolve, reject) => {
4918
+
try {
4919
+
const ref = this.__view().pushHookEvent(this.el, null, event, payload, (reply, _ref) => resolve(reply));
4920
+
if (ref === false) {
4921
+
reject(new Error("unable to push hook event. LiveView not connected"));
4922
+
}
4923
+
} catch (error) {
4924
+
reject(error);
4925
+
}
4926
+
});
4927
+
}
4928
+
return this.__view().pushHookEvent(this.el, null, event, payload, onReply);
4929
+
}
4930
+
pushEventTo(phxTarget, event, payload = {}, onReply) {
4931
+
if (onReply === void 0) {
4932
+
return new Promise((resolve, reject) => {
4933
+
try {
4934
+
this.__view().withinTargets(phxTarget, (view, targetCtx) => {
4935
+
const ref = view.pushHookEvent(this.el, targetCtx, event, payload, (reply, _ref) => resolve(reply));
4936
+
if (ref === false) {
4937
+
reject(new Error("unable to push hook event. LiveView not connected"));
4938
+
}
4939
+
});
4940
+
} catch (error) {
4941
+
reject(error);
4942
+
}
4943
+
});
4944
+
}
4945
+
return this.__view().withinTargets(phxTarget, (view, targetCtx) => {
4946
+
return view.pushHookEvent(this.el, targetCtx, event, payload, onReply);
4947
+
});
4948
+
}
4949
+
handleEvent(event, callback) {
4950
+
let callbackRef = (customEvent, bypass) => bypass ? event : callback(customEvent.detail);
4951
+
window.addEventListener(`phx:${event}`, callbackRef);
4952
+
this.__listeners.add(callbackRef);
4953
+
return callbackRef;
4954
+
}
4955
+
removeHandleEvent(callbackRef) {
4956
+
let event = callbackRef(null, true);
4957
+
window.removeEventListener(`phx:${event}`, callbackRef);
4958
+
this.__listeners.delete(callbackRef);
4959
+
}
4960
+
upload(name, files) {
4961
+
return this.__view().dispatchUploads(null, name, files);
4962
+
}
4963
+
uploadTo(phxTarget, name, files) {
4964
+
return this.__view().withinTargets(phxTarget, (view, targetCtx) => {
4965
+
view.dispatchUploads(targetCtx, name, files);
4966
+
});
4967
+
}
4968
+
__cleanup__() {
4969
+
this.__listeners.forEach((callbackRef) => this.removeHandleEvent(callbackRef));
4970
+
}
4971
+
};
4972
+
var prependFormDataKey = (key, prefix) => {
4973
+
let isArray = key.endsWith("[]");
4974
+
let baseKey = isArray ? key.slice(0, -2) : key;
4975
+
baseKey = baseKey.replace(/([^\[\]]+)(\]?$)/, `${prefix}$1$2`);
4976
+
if (isArray) {
4977
+
baseKey += "[]";
4978
+
}
4979
+
return baseKey;
4980
+
};
4981
+
var serializeForm = (form, opts, onlyNames = []) => {
4982
+
const { submitter } = opts;
4983
+
let injectedElement;
4984
+
if (submitter && submitter.name) {
4985
+
const input = document.createElement("input");
4986
+
input.type = "hidden";
4987
+
const formId = submitter.getAttribute("form");
4988
+
if (formId) {
4989
+
input.setAttribute("form", formId);
4990
+
}
4991
+
input.name = submitter.name;
4992
+
input.value = submitter.value;
4993
+
submitter.parentElement.insertBefore(input, submitter);
4994
+
injectedElement = input;
4995
+
}
4996
+
const formData = new FormData(form);
4997
+
const toRemove = [];
4998
+
formData.forEach((val, key, _index) => {
4999
+
if (val instanceof File) {
5000
+
toRemove.push(key);
5001
+
}
5002
+
});
5003
+
toRemove.forEach((key) => formData.delete(key));
5004
+
const params = new URLSearchParams();
5005
+
const { inputsUnused, onlyHiddenInputs } = Array.from(form.elements).reduce((acc, input) => {
5006
+
const { inputsUnused: inputsUnused2, onlyHiddenInputs: onlyHiddenInputs2 } = acc;
5007
+
const key = input.name;
5008
+
if (!key) {
5009
+
return acc;
5010
+
}
5011
+
if (inputsUnused2[key] === void 0) {
5012
+
inputsUnused2[key] = true;
5013
+
}
5014
+
if (onlyHiddenInputs2[key] === void 0) {
5015
+
onlyHiddenInputs2[key] = true;
5016
+
}
5017
+
const isUsed = dom_default.private(input, PHX_HAS_FOCUSED) || dom_default.private(input, PHX_HAS_SUBMITTED);
5018
+
const isHidden = input.type === "hidden";
5019
+
inputsUnused2[key] = inputsUnused2[key] && !isUsed;
5020
+
onlyHiddenInputs2[key] = onlyHiddenInputs2[key] && isHidden;
5021
+
return acc;
5022
+
}, { inputsUnused: {}, onlyHiddenInputs: {} });
5023
+
for (let [key, val] of formData.entries()) {
5024
+
if (onlyNames.length === 0 || onlyNames.indexOf(key) >= 0) {
5025
+
let isUnused = inputsUnused[key];
5026
+
let hidden = onlyHiddenInputs[key];
5027
+
if (isUnused && !(submitter && submitter.name == key) && !hidden) {
5028
+
params.append(prependFormDataKey(key, "_unused_"), "");
5029
+
}
5030
+
params.append(key, val);
5031
+
}
5032
+
}
5033
+
if (submitter && injectedElement) {
5034
+
submitter.parentElement.removeChild(injectedElement);
5035
+
}
5036
+
return params.toString();
5037
+
};
5038
+
var View = class _View {
5039
+
static closestView(el) {
5040
+
let liveViewEl = el.closest(PHX_VIEW_SELECTOR);
5041
+
return liveViewEl ? dom_default.private(liveViewEl, "view") : null;
5042
+
}
5043
+
constructor(el, liveSocket2, parentView, flash, liveReferer) {
5044
+
this.isDead = false;
5045
+
this.liveSocket = liveSocket2;
5046
+
this.flash = flash;
5047
+
this.parent = parentView;
5048
+
this.root = parentView ? parentView.root : this;
5049
+
this.el = el;
5050
+
dom_default.putPrivate(this.el, "view", this);
5051
+
this.id = this.el.id;
5052
+
this.ref = 0;
5053
+
this.lastAckRef = null;
5054
+
this.childJoins = 0;
5055
+
this.loaderTimer = null;
5056
+
this.disconnectedTimer = null;
5057
+
this.pendingDiffs = [];
5058
+
this.pendingForms = /* @__PURE__ */ new Set();
5059
+
this.redirect = false;
5060
+
this.href = null;
5061
+
this.joinCount = this.parent ? this.parent.joinCount - 1 : 0;
5062
+
this.joinAttempts = 0;
5063
+
this.joinPending = true;
5064
+
this.destroyed = false;
5065
+
this.joinCallback = function(onDone) {
5066
+
onDone && onDone();
5067
+
};
5068
+
this.stopCallback = function() {
5069
+
};
5070
+
this.pendingJoinOps = this.parent ? null : [];
5071
+
this.viewHooks = {};
5072
+
this.formSubmits = [];
5073
+
this.children = this.parent ? null : {};
5074
+
this.root.children[this.id] = {};
5075
+
this.formsForRecovery = {};
5076
+
this.channel = this.liveSocket.channel(`lv:${this.id}`, () => {
5077
+
let url = this.href && this.expandURL(this.href);
5078
+
return {
5079
+
redirect: this.redirect ? url : void 0,
5080
+
url: this.redirect ? void 0 : url || void 0,
5081
+
params: this.connectParams(liveReferer),
5082
+
session: this.getSession(),
5083
+
static: this.getStatic(),
5084
+
flash: this.flash,
5085
+
sticky: this.el.hasAttribute(PHX_STICKY)
5086
+
};
5087
+
});
5088
+
}
5089
+
setHref(href) {
5090
+
this.href = href;
5091
+
}
5092
+
setRedirect(href) {
5093
+
this.redirect = true;
5094
+
this.href = href;
5095
+
}
5096
+
isMain() {
5097
+
return this.el.hasAttribute(PHX_MAIN);
5098
+
}
5099
+
connectParams(liveReferer) {
5100
+
let params = this.liveSocket.params(this.el);
5101
+
let manifest = dom_default.all(document, `[${this.binding(PHX_TRACK_STATIC)}]`).map((node) => node.src || node.href).filter((url) => typeof url === "string");
5102
+
if (manifest.length > 0) {
5103
+
params["_track_static"] = manifest;
5104
+
}
5105
+
params["_mounts"] = this.joinCount;
5106
+
params["_mount_attempts"] = this.joinAttempts;
5107
+
params["_live_referer"] = liveReferer;
5108
+
this.joinAttempts++;
5109
+
return params;
5110
+
}
5111
+
isConnected() {
5112
+
return this.channel.canPush();
5113
+
}
5114
+
getSession() {
5115
+
return this.el.getAttribute(PHX_SESSION);
5116
+
}
5117
+
getStatic() {
5118
+
let val = this.el.getAttribute(PHX_STATIC);
5119
+
return val === "" ? null : val;
5120
+
}
5121
+
destroy(callback = function() {
5122
+
}) {
5123
+
this.destroyAllChildren();
5124
+
this.destroyed = true;
5125
+
delete this.root.children[this.id];
5126
+
if (this.parent) {
5127
+
delete this.root.children[this.parent.id][this.id];
5128
+
}
5129
+
clearTimeout(this.loaderTimer);
5130
+
let onFinished = () => {
5131
+
callback();
5132
+
for (let id in this.viewHooks) {
5133
+
this.destroyHook(this.viewHooks[id]);
5134
+
}
5135
+
};
5136
+
dom_default.markPhxChildDestroyed(this.el);
5137
+
this.log("destroyed", () => ["the child has been removed from the parent"]);
5138
+
this.channel.leave().receive("ok", onFinished).receive("error", onFinished).receive("timeout", onFinished);
5139
+
}
5140
+
setContainerClasses(...classes) {
5141
+
this.el.classList.remove(
5142
+
PHX_CONNECTED_CLASS,
5143
+
PHX_LOADING_CLASS,
5144
+
PHX_ERROR_CLASS,
5145
+
PHX_CLIENT_ERROR_CLASS,
5146
+
PHX_SERVER_ERROR_CLASS
5147
+
);
5148
+
this.el.classList.add(...classes);
5149
+
}
5150
+
showLoader(timeout) {
5151
+
clearTimeout(this.loaderTimer);
5152
+
if (timeout) {
5153
+
this.loaderTimer = setTimeout(() => this.showLoader(), timeout);
5154
+
} else {
5155
+
for (let id in this.viewHooks) {
5156
+
this.viewHooks[id].__disconnected();
5157
+
}
5158
+
this.setContainerClasses(PHX_LOADING_CLASS);
5159
+
}
5160
+
}
5161
+
execAll(binding) {
5162
+
dom_default.all(this.el, `[${binding}]`, (el) => this.liveSocket.execJS(el, el.getAttribute(binding)));
5163
+
}
5164
+
hideLoader() {
5165
+
clearTimeout(this.loaderTimer);
5166
+
clearTimeout(this.disconnectedTimer);
5167
+
this.setContainerClasses(PHX_CONNECTED_CLASS);
5168
+
this.execAll(this.binding("connected"));
5169
+
}
5170
+
triggerReconnected() {
5171
+
for (let id in this.viewHooks) {
5172
+
this.viewHooks[id].__reconnected();
5173
+
}
5174
+
}
5175
+
log(kind, msgCallback) {
5176
+
this.liveSocket.log(this, kind, msgCallback);
5177
+
}
5178
+
transition(time, onStart, onDone = function() {
5179
+
}) {
5180
+
this.liveSocket.transition(time, onStart, onDone);
5181
+
}
5182
+
// calls the callback with the view and target element for the given phxTarget
5183
+
// targets can be:
5184
+
// * an element itself, then it is simply passed to liveSocket.owner;
5185
+
// * a CID (Component ID), then we first search the component's element in the DOM
5186
+
// * a selector, then we search the selector in the DOM and call the callback
5187
+
// for each element found with the corresponding owner view
5188
+
withinTargets(phxTarget, callback, dom = document, viewEl) {
5189
+
if (phxTarget instanceof HTMLElement || phxTarget instanceof SVGElement) {
5190
+
return this.liveSocket.owner(phxTarget, (view) => callback(view, phxTarget));
5191
+
}
5192
+
if (isCid(phxTarget)) {
5193
+
let targets = dom_default.findComponentNodeList(viewEl || this.el, phxTarget);
5194
+
if (targets.length === 0) {
5195
+
logError(`no component found matching phx-target of ${phxTarget}`);
5196
+
} else {
5197
+
callback(this, parseInt(phxTarget));
5198
+
}
5199
+
} else {
5200
+
let targets = Array.from(dom.querySelectorAll(phxTarget));
5201
+
if (targets.length === 0) {
5202
+
logError(`nothing found matching the phx-target selector "${phxTarget}"`);
5203
+
}
5204
+
targets.forEach((target) => this.liveSocket.owner(target, (view) => callback(view, target)));
5205
+
}
5206
+
}
5207
+
applyDiff(type, rawDiff, callback) {
5208
+
this.log(type, () => ["", clone(rawDiff)]);
5209
+
let { diff, reply, events, title } = Rendered.extract(rawDiff);
5210
+
callback({ diff, reply, events });
5211
+
if (typeof title === "string" || type == "mount") {
5212
+
window.requestAnimationFrame(() => dom_default.putTitle(title));
5213
+
}
5214
+
}
5215
+
onJoin(resp) {
5216
+
let { rendered, container, liveview_version } = resp;
5217
+
if (container) {
5218
+
let [tag, attrs] = container;
5219
+
this.el = dom_default.replaceRootContainer(this.el, tag, attrs);
5220
+
}
5221
+
this.childJoins = 0;
5222
+
this.joinPending = true;
5223
+
this.flash = null;
5224
+
if (this.root === this) {
5225
+
this.formsForRecovery = this.getFormsForRecovery();
5226
+
}
5227
+
if (this.isMain() && window.history.state === null) {
5228
+
browser_default.pushState("replace", {
5229
+
type: "patch",
5230
+
id: this.id,
5231
+
position: this.liveSocket.currentHistoryPosition
5232
+
});
5233
+
}
5234
+
if (liveview_version !== this.liveSocket.version()) {
5235
+
console.error(`LiveView asset version mismatch. JavaScript version ${this.liveSocket.version()} vs. server ${liveview_version}. To avoid issues, please ensure that your assets use the same version as the server.`);
5236
+
}
5237
+
browser_default.dropLocal(this.liveSocket.localStorage, window.location.pathname, CONSECUTIVE_RELOADS);
5238
+
this.applyDiff("mount", rendered, ({ diff, events }) => {
5239
+
this.rendered = new Rendered(this.id, diff);
5240
+
let [html, streams] = this.renderContainer(null, "join");
5241
+
this.dropPendingRefs();
5242
+
this.joinCount++;
5243
+
this.joinAttempts = 0;
5244
+
this.maybeRecoverForms(html, () => {
5245
+
this.onJoinComplete(resp, html, streams, events);
5246
+
});
5247
+
});
5248
+
}
5249
+
dropPendingRefs() {
5250
+
dom_default.all(document, `[${PHX_REF_SRC}="${this.refSrc()}"]`, (el) => {
5251
+
el.removeAttribute(PHX_REF_LOADING);
5252
+
el.removeAttribute(PHX_REF_SRC);
5253
+
el.removeAttribute(PHX_REF_LOCK);
5254
+
});
5255
+
}
5256
+
onJoinComplete({ live_patch }, html, streams, events) {
5257
+
if (this.joinCount > 1 || this.parent && !this.parent.isJoinPending()) {
5258
+
return this.applyJoinPatch(live_patch, html, streams, events);
5259
+
}
5260
+
let newChildren = dom_default.findPhxChildrenInFragment(html, this.id).filter((toEl) => {
5261
+
let fromEl = toEl.id && this.el.querySelector(`[id="${toEl.id}"]`);
5262
+
let phxStatic = fromEl && fromEl.getAttribute(PHX_STATIC);
5263
+
if (phxStatic) {
5264
+
toEl.setAttribute(PHX_STATIC, phxStatic);
5265
+
}
5266
+
if (fromEl) {
5267
+
fromEl.setAttribute(PHX_ROOT_ID, this.root.id);
5268
+
}
5269
+
return this.joinChild(toEl);
5270
+
});
5271
+
if (newChildren.length === 0) {
5272
+
if (this.parent) {
5273
+
this.root.pendingJoinOps.push([this, () => this.applyJoinPatch(live_patch, html, streams, events)]);
5274
+
this.parent.ackJoin(this);
5275
+
} else {
5276
+
this.onAllChildJoinsComplete();
5277
+
this.applyJoinPatch(live_patch, html, streams, events);
5278
+
}
5279
+
} else {
5280
+
this.root.pendingJoinOps.push([this, () => this.applyJoinPatch(live_patch, html, streams, events)]);
5281
+
}
5282
+
}
5283
+
attachTrueDocEl() {
5284
+
this.el = dom_default.byId(this.id);
5285
+
this.el.setAttribute(PHX_ROOT_ID, this.root.id);
5286
+
}
5287
+
// this is invoked for dead and live views, so we must filter by
5288
+
// by owner to ensure we aren't duplicating hooks across disconnect
5289
+
// and connected states. This also handles cases where hooks exist
5290
+
// in a root layout with a LV in the body
5291
+
execNewMounted(parent = this.el) {
5292
+
let phxViewportTop = this.binding(PHX_VIEWPORT_TOP);
5293
+
let phxViewportBottom = this.binding(PHX_VIEWPORT_BOTTOM);
5294
+
dom_default.all(parent, `[${phxViewportTop}], [${phxViewportBottom}]`, (hookEl) => {
5295
+
if (this.ownsElement(hookEl)) {
5296
+
dom_default.maintainPrivateHooks(hookEl, hookEl, phxViewportTop, phxViewportBottom);
5297
+
this.maybeAddNewHook(hookEl);
5298
+
}
5299
+
});
5300
+
dom_default.all(parent, `[${this.binding(PHX_HOOK)}], [data-phx-${PHX_HOOK}]`, (hookEl) => {
5301
+
if (this.ownsElement(hookEl)) {
5302
+
this.maybeAddNewHook(hookEl);
5303
+
}
5304
+
});
5305
+
dom_default.all(parent, `[${this.binding(PHX_MOUNTED)}]`, (el) => {
5306
+
if (this.ownsElement(el)) {
5307
+
this.maybeMounted(el);
5308
+
}
5309
+
});
5310
+
}
5311
+
applyJoinPatch(live_patch, html, streams, events) {
5312
+
this.attachTrueDocEl();
5313
+
let patch = new DOMPatch(this, this.el, this.id, html, streams, null);
5314
+
patch.markPrunableContentForRemoval();
5315
+
this.performPatch(patch, false, true);
5316
+
this.joinNewChildren();
5317
+
this.execNewMounted();
5318
+
this.joinPending = false;
5319
+
this.liveSocket.dispatchEvents(events);
5320
+
this.applyPendingUpdates();
5321
+
if (live_patch) {
5322
+
let { kind, to } = live_patch;
5323
+
this.liveSocket.historyPatch(to, kind);
5324
+
}
5325
+
this.hideLoader();
5326
+
if (this.joinCount > 1) {
5327
+
this.triggerReconnected();
5328
+
}
5329
+
this.stopCallback();
5330
+
}
5331
+
triggerBeforeUpdateHook(fromEl, toEl) {
5332
+
this.liveSocket.triggerDOM("onBeforeElUpdated", [fromEl, toEl]);
5333
+
let hook = this.getHook(fromEl);
5334
+
let isIgnored = hook && dom_default.isIgnored(fromEl, this.binding(PHX_UPDATE));
5335
+
if (hook && !fromEl.isEqualNode(toEl) && !(isIgnored && isEqualObj(fromEl.dataset, toEl.dataset))) {
5336
+
hook.__beforeUpdate();
5337
+
return hook;
5338
+
}
5339
+
}
5340
+
maybeMounted(el) {
5341
+
let phxMounted = el.getAttribute(this.binding(PHX_MOUNTED));
5342
+
let hasBeenInvoked = phxMounted && dom_default.private(el, "mounted");
5343
+
if (phxMounted && !hasBeenInvoked) {
5344
+
this.liveSocket.execJS(el, phxMounted);
5345
+
dom_default.putPrivate(el, "mounted", true);
5346
+
}
5347
+
}
5348
+
maybeAddNewHook(el) {
5349
+
let newHook = this.addHook(el);
5350
+
if (newHook) {
5351
+
newHook.__mounted();
5352
+
}
5353
+
}
5354
+
performPatch(patch, pruneCids, isJoinPatch = false) {
5355
+
let removedEls = [];
5356
+
let phxChildrenAdded = false;
5357
+
let updatedHookIds = /* @__PURE__ */ new Set();
5358
+
this.liveSocket.triggerDOM("onPatchStart", [patch.targetContainer]);
5359
+
patch.after("added", (el) => {
5360
+
this.liveSocket.triggerDOM("onNodeAdded", [el]);
5361
+
let phxViewportTop = this.binding(PHX_VIEWPORT_TOP);
5362
+
let phxViewportBottom = this.binding(PHX_VIEWPORT_BOTTOM);
5363
+
dom_default.maintainPrivateHooks(el, el, phxViewportTop, phxViewportBottom);
5364
+
this.maybeAddNewHook(el);
5365
+
if (el.getAttribute) {
5366
+
this.maybeMounted(el);
5367
+
}
5368
+
});
5369
+
patch.after("phxChildAdded", (el) => {
5370
+
if (dom_default.isPhxSticky(el)) {
5371
+
this.liveSocket.joinRootViews();
5372
+
} else {
5373
+
phxChildrenAdded = true;
5374
+
}
5375
+
});
5376
+
patch.before("updated", (fromEl, toEl) => {
5377
+
let hook = this.triggerBeforeUpdateHook(fromEl, toEl);
5378
+
if (hook) {
5379
+
updatedHookIds.add(fromEl.id);
5380
+
}
5381
+
});
5382
+
patch.after("updated", (el) => {
5383
+
if (updatedHookIds.has(el.id)) {
5384
+
this.getHook(el).__updated();
5385
+
}
5386
+
});
5387
+
patch.after("discarded", (el) => {
5388
+
if (el.nodeType === Node.ELEMENT_NODE) {
5389
+
removedEls.push(el);
5390
+
}
5391
+
});
5392
+
patch.after("transitionsDiscarded", (els) => this.afterElementsRemoved(els, pruneCids));
5393
+
patch.perform(isJoinPatch);
5394
+
this.afterElementsRemoved(removedEls, pruneCids);
5395
+
this.liveSocket.triggerDOM("onPatchEnd", [patch.targetContainer]);
5396
+
return phxChildrenAdded;
5397
+
}
5398
+
afterElementsRemoved(elements, pruneCids) {
5399
+
let destroyedCIDs = [];
5400
+
elements.forEach((parent) => {
5401
+
let components = dom_default.all(parent, `[${PHX_COMPONENT}]`);
5402
+
let hooks = dom_default.all(parent, `[${this.binding(PHX_HOOK)}], [data-phx-hook]`);
5403
+
components.concat(parent).forEach((el) => {
5404
+
let cid = this.componentID(el);
5405
+
if (isCid(cid) && destroyedCIDs.indexOf(cid) === -1) {
5406
+
destroyedCIDs.push(cid);
5407
+
}
5408
+
});
5409
+
hooks.concat(parent).forEach((hookEl) => {
5410
+
let hook = this.getHook(hookEl);
5411
+
hook && this.destroyHook(hook);
5412
+
});
5413
+
});
5414
+
if (pruneCids) {
5415
+
this.maybePushComponentsDestroyed(destroyedCIDs);
5416
+
}
5417
+
}
5418
+
joinNewChildren() {
5419
+
dom_default.findPhxChildren(this.el, this.id).forEach((el) => this.joinChild(el));
5420
+
}
5421
+
maybeRecoverForms(html, callback) {
5422
+
const phxChange = this.binding("change");
5423
+
const oldForms = this.root.formsForRecovery;
5424
+
let template = document.createElement("template");
5425
+
template.innerHTML = html;
5426
+
const rootEl = template.content.firstElementChild;
5427
+
rootEl.id = this.id;
5428
+
rootEl.setAttribute(PHX_ROOT_ID, this.root.id);
5429
+
rootEl.setAttribute(PHX_SESSION, this.getSession());
5430
+
rootEl.setAttribute(PHX_STATIC, this.getStatic());
5431
+
rootEl.setAttribute(PHX_PARENT_ID, this.parent ? this.parent.id : null);
5432
+
const formsToRecover = (
5433
+
// we go over all forms in the new DOM; because this is only the HTML for the current
5434
+
// view, we can be sure that all forms are owned by this view:
5435
+
dom_default.all(template.content, "form").filter((newForm) => newForm.id && oldForms[newForm.id]).filter((newForm) => !this.pendingForms.has(newForm.id)).filter((newForm) => oldForms[newForm.id].getAttribute(phxChange) === newForm.getAttribute(phxChange)).map((newForm) => {
5436
+
return [oldForms[newForm.id], newForm];
5437
+
})
5438
+
);
5439
+
if (formsToRecover.length === 0) {
5440
+
return callback();
5441
+
}
5442
+
formsToRecover.forEach(([oldForm, newForm], i) => {
5443
+
this.pendingForms.add(newForm.id);
5444
+
this.pushFormRecovery(oldForm, newForm, template.content.firstElementChild, () => {
5445
+
this.pendingForms.delete(newForm.id);
5446
+
if (i === formsToRecover.length - 1) {
5447
+
callback();
5448
+
}
5449
+
});
5450
+
});
5451
+
}
5452
+
getChildById(id) {
5453
+
return this.root.children[this.id][id];
5454
+
}
5455
+
getDescendentByEl(el) {
5456
+
var _a;
5457
+
if (el.id === this.id) {
5458
+
return this;
5459
+
} else {
5460
+
return (_a = this.children[el.getAttribute(PHX_PARENT_ID)]) == null ? void 0 : _a[el.id];
5461
+
}
5462
+
}
5463
+
destroyDescendent(id) {
5464
+
for (let parentId in this.root.children) {
5465
+
for (let childId in this.root.children[parentId]) {
5466
+
if (childId === id) {
5467
+
return this.root.children[parentId][childId].destroy();
5468
+
}
5469
+
}
5470
+
}
5471
+
}
5472
+
joinChild(el) {
5473
+
let child = this.getChildById(el.id);
5474
+
if (!child) {
5475
+
let view = new _View(el, this.liveSocket, this);
5476
+
this.root.children[this.id][view.id] = view;
5477
+
view.join();
5478
+
this.childJoins++;
5479
+
return true;
5480
+
}
5481
+
}
5482
+
isJoinPending() {
5483
+
return this.joinPending;
5484
+
}
5485
+
ackJoin(_child) {
5486
+
this.childJoins--;
5487
+
if (this.childJoins === 0) {
5488
+
if (this.parent) {
5489
+
this.parent.ackJoin(this);
5490
+
} else {
5491
+
this.onAllChildJoinsComplete();
5492
+
}
5493
+
}
5494
+
}
5495
+
onAllChildJoinsComplete() {
5496
+
this.pendingForms.clear();
5497
+
this.formsForRecovery = {};
5498
+
this.joinCallback(() => {
5499
+
this.pendingJoinOps.forEach(([view, op]) => {
5500
+
if (!view.isDestroyed()) {
5501
+
op();
5502
+
}
5503
+
});
5504
+
this.pendingJoinOps = [];
5505
+
});
5506
+
}
5507
+
update(diff, events) {
5508
+
if (this.isJoinPending() || this.liveSocket.hasPendingLink() && this.root.isMain()) {
5509
+
return this.pendingDiffs.push({ diff, events });
5510
+
}
5511
+
this.rendered.mergeDiff(diff);
5512
+
let phxChildrenAdded = false;
5513
+
if (this.rendered.isComponentOnlyDiff(diff)) {
5514
+
this.liveSocket.time("component patch complete", () => {
5515
+
let parentCids = dom_default.findExistingParentCIDs(this.el, this.rendered.componentCIDs(diff));
5516
+
parentCids.forEach((parentCID) => {
5517
+
if (this.componentPatch(this.rendered.getComponent(diff, parentCID), parentCID)) {
5518
+
phxChildrenAdded = true;
5519
+
}
5520
+
});
5521
+
});
5522
+
} else if (!isEmpty(diff)) {
5523
+
this.liveSocket.time("full patch complete", () => {
5524
+
let [html, streams] = this.renderContainer(diff, "update");
5525
+
let patch = new DOMPatch(this, this.el, this.id, html, streams, null);
5526
+
phxChildrenAdded = this.performPatch(patch, true);
5527
+
});
5528
+
}
5529
+
this.liveSocket.dispatchEvents(events);
5530
+
if (phxChildrenAdded) {
5531
+
this.joinNewChildren();
5532
+
}
5533
+
}
5534
+
renderContainer(diff, kind) {
5535
+
return this.liveSocket.time(`toString diff (${kind})`, () => {
5536
+
let tag = this.el.tagName;
5537
+
let cids = diff ? this.rendered.componentCIDs(diff) : null;
5538
+
let [html, streams] = this.rendered.toString(cids);
5539
+
return [`<${tag}>${html}</${tag}>`, streams];
5540
+
});
5541
+
}
5542
+
componentPatch(diff, cid) {
5543
+
if (isEmpty(diff))
5544
+
return false;
5545
+
let [html, streams] = this.rendered.componentToString(cid);
5546
+
let patch = new DOMPatch(this, this.el, this.id, html, streams, cid);
5547
+
let childrenAdded = this.performPatch(patch, true);
5548
+
return childrenAdded;
5549
+
}
5550
+
getHook(el) {
5551
+
return this.viewHooks[ViewHook.elementID(el)];
5552
+
}
5553
+
addHook(el) {
5554
+
let hookElId = ViewHook.elementID(el);
5555
+
if (el.getAttribute && !this.ownsElement(el)) {
5556
+
return;
5557
+
}
5558
+
if (hookElId && !this.viewHooks[hookElId]) {
5559
+
let hook = dom_default.getCustomElHook(el) || logError(`no hook found for custom element: ${el.id}`);
5560
+
this.viewHooks[hookElId] = hook;
5561
+
hook.__attachView(this);
5562
+
return hook;
5563
+
} else if (hookElId || !el.getAttribute) {
5564
+
return;
5565
+
} else {
5566
+
let hookName = el.getAttribute(`data-phx-${PHX_HOOK}`) || el.getAttribute(this.binding(PHX_HOOK));
5567
+
let callbacks = this.liveSocket.getHookCallbacks(hookName);
5568
+
if (callbacks) {
5569
+
if (!el.id) {
5570
+
logError(`no DOM ID for hook "${hookName}". Hooks require a unique ID on each element.`, el);
5571
+
}
5572
+
let hook = new ViewHook(this, el, callbacks);
5573
+
this.viewHooks[ViewHook.elementID(hook.el)] = hook;
5574
+
return hook;
5575
+
} else if (hookName !== null) {
5576
+
logError(`unknown hook found for "${hookName}"`, el);
5577
+
}
5578
+
}
5579
+
}
5580
+
destroyHook(hook) {
5581
+
const hookId = ViewHook.elementID(hook.el);
5582
+
hook.__destroyed();
5583
+
hook.__cleanup__();
5584
+
delete this.viewHooks[hookId];
5585
+
}
5586
+
applyPendingUpdates() {
5587
+
if (this.liveSocket.hasPendingLink() && this.root.isMain()) {
5588
+
return;
5589
+
}
5590
+
this.pendingDiffs.forEach(({ diff, events }) => this.update(diff, events));
5591
+
this.pendingDiffs = [];
5592
+
this.eachChild((child) => child.applyPendingUpdates());
5593
+
}
5594
+
eachChild(callback) {
5595
+
let children = this.root.children[this.id] || {};
5596
+
for (let id in children) {
5597
+
callback(this.getChildById(id));
5598
+
}
5599
+
}
5600
+
onChannel(event, cb) {
5601
+
this.liveSocket.onChannel(this.channel, event, (resp) => {
5602
+
if (this.isJoinPending()) {
5603
+
this.root.pendingJoinOps.push([this, () => cb(resp)]);
5604
+
} else {
5605
+
this.liveSocket.requestDOMUpdate(() => cb(resp));
5606
+
}
5607
+
});
5608
+
}
5609
+
bindChannel() {
5610
+
this.liveSocket.onChannel(this.channel, "diff", (rawDiff) => {
5611
+
this.liveSocket.requestDOMUpdate(() => {
5612
+
this.applyDiff("update", rawDiff, ({ diff, events }) => this.update(diff, events));
5613
+
});
5614
+
});
5615
+
this.onChannel("redirect", ({ to, flash }) => this.onRedirect({ to, flash }));
5616
+
this.onChannel("live_patch", (redir) => this.onLivePatch(redir));
5617
+
this.onChannel("live_redirect", (redir) => this.onLiveRedirect(redir));
5618
+
this.channel.onError((reason) => this.onError(reason));
5619
+
this.channel.onClose((reason) => this.onClose(reason));
5620
+
}
5621
+
destroyAllChildren() {
5622
+
this.eachChild((child) => child.destroy());
5623
+
}
5624
+
onLiveRedirect(redir) {
5625
+
let { to, kind, flash } = redir;
5626
+
let url = this.expandURL(to);
5627
+
let e = new CustomEvent("phx:server-navigate", { detail: { to, kind, flash } });
5628
+
this.liveSocket.historyRedirect(e, url, kind, flash);
5629
+
}
5630
+
onLivePatch(redir) {
5631
+
let { to, kind } = redir;
5632
+
this.href = this.expandURL(to);
5633
+
this.liveSocket.historyPatch(to, kind);
5634
+
}
5635
+
expandURL(to) {
5636
+
return to.startsWith("/") ? `${window.location.protocol}//${window.location.host}${to}` : to;
5637
+
}
5638
+
onRedirect({ to, flash, reloadToken }) {
5639
+
this.liveSocket.redirect(to, flash, reloadToken);
5640
+
}
5641
+
isDestroyed() {
5642
+
return this.destroyed;
5643
+
}
5644
+
joinDead() {
5645
+
this.isDead = true;
5646
+
}
5647
+
joinPush() {
5648
+
this.joinPush = this.joinPush || this.channel.join();
5649
+
return this.joinPush;
5650
+
}
5651
+
join(callback) {
5652
+
this.showLoader(this.liveSocket.loaderTimeout);
5653
+
this.bindChannel();
5654
+
if (this.isMain()) {
5655
+
this.stopCallback = this.liveSocket.withPageLoading({ to: this.href, kind: "initial" });
5656
+
}
5657
+
this.joinCallback = (onDone) => {
5658
+
onDone = onDone || function() {
5659
+
};
5660
+
callback ? callback(this.joinCount, onDone) : onDone();
5661
+
};
5662
+
this.wrapPush(() => this.channel.join(), {
5663
+
ok: (resp) => this.liveSocket.requestDOMUpdate(() => this.onJoin(resp)),
5664
+
error: (error) => this.onJoinError(error),
5665
+
timeout: () => this.onJoinError({ reason: "timeout" })
5666
+
});
5667
+
}
5668
+
onJoinError(resp) {
5669
+
if (resp.reason === "reload") {
5670
+
this.log("error", () => [`failed mount with ${resp.status}. Falling back to page reload`, resp]);
5671
+
this.onRedirect({ to: this.root.href, reloadToken: resp.token });
5672
+
return;
5673
+
} else if (resp.reason === "unauthorized" || resp.reason === "stale") {
5674
+
this.log("error", () => ["unauthorized live_redirect. Falling back to page request", resp]);
5675
+
this.onRedirect({ to: this.root.href, flash: this.flash });
5676
+
return;
5677
+
}
5678
+
if (resp.redirect || resp.live_redirect) {
5679
+
this.joinPending = false;
5680
+
this.channel.leave();
5681
+
}
5682
+
if (resp.redirect) {
5683
+
return this.onRedirect(resp.redirect);
5684
+
}
5685
+
if (resp.live_redirect) {
5686
+
return this.onLiveRedirect(resp.live_redirect);
5687
+
}
5688
+
this.log("error", () => ["unable to join", resp]);
5689
+
if (this.isMain()) {
5690
+
this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS]);
5691
+
if (this.liveSocket.isConnected()) {
5692
+
this.liveSocket.reloadWithJitter(this);
5693
+
}
5694
+
} else {
5695
+
if (this.joinAttempts >= MAX_CHILD_JOIN_ATTEMPTS) {
5696
+
this.root.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS]);
5697
+
this.log("error", () => [`giving up trying to mount after ${MAX_CHILD_JOIN_ATTEMPTS} tries`, resp]);
5698
+
this.destroy();
5699
+
}
5700
+
let trueChildEl = dom_default.byId(this.el.id);
5701
+
if (trueChildEl) {
5702
+
dom_default.mergeAttrs(trueChildEl, this.el);
5703
+
this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS]);
5704
+
this.el = trueChildEl;
5705
+
} else {
5706
+
this.destroy();
5707
+
}
5708
+
}
5709
+
}
5710
+
onClose(reason) {
5711
+
if (this.isDestroyed()) {
5712
+
return;
5713
+
}
5714
+
if (this.isMain() && this.liveSocket.hasPendingLink() && reason !== "leave") {
5715
+
return this.liveSocket.reloadWithJitter(this);
5716
+
}
5717
+
this.destroyAllChildren();
5718
+
this.liveSocket.dropActiveElement(this);
5719
+
if (document.activeElement) {
5720
+
document.activeElement.blur();
5721
+
}
5722
+
if (this.liveSocket.isUnloaded()) {
5723
+
this.showLoader(BEFORE_UNLOAD_LOADER_TIMEOUT);
5724
+
}
5725
+
}
5726
+
onError(reason) {
5727
+
this.onClose(reason);
5728
+
if (this.liveSocket.isConnected()) {
5729
+
this.log("error", () => ["view crashed", reason]);
5730
+
}
5731
+
if (!this.liveSocket.isUnloaded()) {
5732
+
if (this.liveSocket.isConnected()) {
5733
+
this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS]);
5734
+
} else {
5735
+
this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_CLIENT_ERROR_CLASS]);
5736
+
}
5737
+
}
5738
+
}
5739
+
displayError(classes) {
5740
+
if (this.isMain()) {
5741
+
dom_default.dispatchEvent(window, "phx:page-loading-start", { detail: { to: this.href, kind: "error" } });
5742
+
}
5743
+
this.showLoader();
5744
+
this.setContainerClasses(...classes);
5745
+
this.delayedDisconnected();
5746
+
}
5747
+
delayedDisconnected() {
5748
+
this.disconnectedTimer = setTimeout(() => {
5749
+
this.execAll(this.binding("disconnected"));
5750
+
}, this.liveSocket.disconnectedTimeout);
5751
+
}
5752
+
wrapPush(callerPush, receives) {
5753
+
let latency = this.liveSocket.getLatencySim();
5754
+
let withLatency = latency ? (cb) => setTimeout(() => !this.isDestroyed() && cb(), latency) : (cb) => !this.isDestroyed() && cb();
5755
+
withLatency(() => {
5756
+
callerPush().receive("ok", (resp) => withLatency(() => receives.ok && receives.ok(resp))).receive("error", (reason) => withLatency(() => receives.error && receives.error(reason))).receive("timeout", () => withLatency(() => receives.timeout && receives.timeout()));
5757
+
});
5758
+
}
5759
+
pushWithReply(refGenerator, event, payload) {
5760
+
if (!this.isConnected()) {
5761
+
return Promise.reject({ error: "noconnection" });
5762
+
}
5763
+
let [ref, [el], opts] = refGenerator ? refGenerator() : [null, [], {}];
5764
+
let oldJoinCount = this.joinCount;
5765
+
let onLoadingDone = function() {
5766
+
};
5767
+
if (opts.page_loading) {
5768
+
onLoadingDone = this.liveSocket.withPageLoading({ kind: "element", target: el });
5769
+
}
5770
+
if (typeof payload.cid !== "number") {
5771
+
delete payload.cid;
5772
+
}
5773
+
return new Promise((resolve, reject) => {
5774
+
this.wrapPush(() => this.channel.push(event, payload, PUSH_TIMEOUT), {
5775
+
ok: (resp) => {
5776
+
if (ref !== null) {
5777
+
this.lastAckRef = ref;
5778
+
}
5779
+
let finish = (hookReply) => {
5780
+
if (resp.redirect) {
5781
+
this.onRedirect(resp.redirect);
5782
+
}
5783
+
if (resp.live_patch) {
5784
+
this.onLivePatch(resp.live_patch);
5785
+
}
5786
+
if (resp.live_redirect) {
5787
+
this.onLiveRedirect(resp.live_redirect);
5788
+
}
5789
+
onLoadingDone();
5790
+
resolve({ resp, reply: hookReply });
5791
+
};
5792
+
if (resp.diff) {
5793
+
this.liveSocket.requestDOMUpdate(() => {
5794
+
this.applyDiff("update", resp.diff, ({ diff, reply, events }) => {
5795
+
if (ref !== null) {
5796
+
this.undoRefs(ref, payload.event);
5797
+
}
5798
+
this.update(diff, events);
5799
+
finish(reply);
5800
+
});
5801
+
});
5802
+
} else {
5803
+
if (ref !== null) {
5804
+
this.undoRefs(ref, payload.event);
5805
+
}
5806
+
finish(null);
5807
+
}
5808
+
},
5809
+
error: (reason) => reject({ error: reason }),
5810
+
timeout: () => {
5811
+
reject({ timeout: true });
5812
+
if (this.joinCount === oldJoinCount) {
5813
+
this.liveSocket.reloadWithJitter(this, () => {
5814
+
this.log("timeout", () => ["received timeout while communicating with server. Falling back to hard refresh for recovery"]);
5815
+
});
5816
+
}
5817
+
}
5818
+
});
5819
+
});
5820
+
}
5821
+
undoRefs(ref, phxEvent, onlyEls) {
5822
+
if (!this.isConnected()) {
5823
+
return;
5824
+
}
5825
+
let selector = `[${PHX_REF_SRC}="${this.refSrc()}"]`;
5826
+
if (onlyEls) {
5827
+
onlyEls = new Set(onlyEls);
5828
+
dom_default.all(document, selector, (parent) => {
5829
+
if (onlyEls && !onlyEls.has(parent)) {
5830
+
return;
5831
+
}
5832
+
dom_default.all(parent, selector, (child) => this.undoElRef(child, ref, phxEvent));
5833
+
this.undoElRef(parent, ref, phxEvent);
5834
+
});
5835
+
} else {
5836
+
dom_default.all(document, selector, (el) => this.undoElRef(el, ref, phxEvent));
5837
+
}
5838
+
}
5839
+
undoElRef(el, ref, phxEvent) {
5840
+
let elRef = new ElementRef(el);
5841
+
elRef.maybeUndo(ref, phxEvent, (clonedTree) => {
5842
+
let patch = new DOMPatch(this, el, this.id, clonedTree, [], null, { undoRef: ref });
5843
+
const phxChildrenAdded = this.performPatch(patch, true);
5844
+
dom_default.all(el, `[${PHX_REF_SRC}="${this.refSrc()}"]`, (child) => this.undoElRef(child, ref, phxEvent));
5845
+
if (phxChildrenAdded) {
5846
+
this.joinNewChildren();
5847
+
}
5848
+
});
5849
+
}
5850
+
refSrc() {
5851
+
return this.el.id;
5852
+
}
5853
+
putRef(elements, phxEvent, eventType, opts = {}) {
5854
+
let newRef = this.ref++;
5855
+
let disableWith = this.binding(PHX_DISABLE_WITH);
5856
+
if (opts.loading) {
5857
+
let loadingEls = dom_default.all(document, opts.loading).map((el) => {
5858
+
return { el, lock: true, loading: true };
5859
+
});
5860
+
elements = elements.concat(loadingEls);
5861
+
}
5862
+
for (let { el, lock, loading } of elements) {
5863
+
if (!lock && !loading) {
5864
+
throw new Error("putRef requires lock or loading");
5865
+
}
5866
+
el.setAttribute(PHX_REF_SRC, this.refSrc());
5867
+
if (loading) {
5868
+
el.setAttribute(PHX_REF_LOADING, newRef);
5869
+
}
5870
+
if (lock) {
5871
+
el.setAttribute(PHX_REF_LOCK, newRef);
5872
+
}
5873
+
if (!loading || opts.submitter && !(el === opts.submitter || el === opts.form)) {
5874
+
continue;
5875
+
}
5876
+
let lockCompletePromise = new Promise((resolve) => {
5877
+
el.addEventListener(`phx:undo-lock:${newRef}`, () => resolve(detail), { once: true });
5878
+
});
5879
+
let loadingCompletePromise = new Promise((resolve) => {
5880
+
el.addEventListener(`phx:undo-loading:${newRef}`, () => resolve(detail), { once: true });
5881
+
});
5882
+
el.classList.add(`phx-${eventType}-loading`);
5883
+
let disableText = el.getAttribute(disableWith);
5884
+
if (disableText !== null) {
5885
+
if (!el.getAttribute(PHX_DISABLE_WITH_RESTORE)) {
5886
+
el.setAttribute(PHX_DISABLE_WITH_RESTORE, el.innerText);
5887
+
}
5888
+
if (disableText !== "") {
5889
+
el.innerText = disableText;
5890
+
}
5891
+
el.setAttribute(PHX_DISABLED, el.getAttribute(PHX_DISABLED) || el.disabled);
5892
+
el.setAttribute("disabled", "");
5893
+
}
5894
+
let detail = {
5895
+
event: phxEvent,
5896
+
eventType,
5897
+
ref: newRef,
5898
+
isLoading: loading,
5899
+
isLocked: lock,
5900
+
lockElements: elements.filter(({ lock: lock2 }) => lock2).map(({ el: el2 }) => el2),
5901
+
loadingElements: elements.filter(({ loading: loading2 }) => loading2).map(({ el: el2 }) => el2),
5902
+
unlock: (els) => {
5903
+
els = Array.isArray(els) ? els : [els];
5904
+
this.undoRefs(newRef, phxEvent, els);
5905
+
},
5906
+
lockComplete: lockCompletePromise,
5907
+
loadingComplete: loadingCompletePromise,
5908
+
lock: (lockEl) => {
5909
+
return new Promise((resolve) => {
5910
+
if (this.isAcked(newRef)) {
5911
+
return resolve(detail);
5912
+
}
5913
+
lockEl.setAttribute(PHX_REF_LOCK, newRef);
5914
+
lockEl.setAttribute(PHX_REF_SRC, this.refSrc());
5915
+
lockEl.addEventListener(`phx:lock-stop:${newRef}`, () => resolve(detail), { once: true });
5916
+
});
5917
+
}
5918
+
};
5919
+
el.dispatchEvent(new CustomEvent("phx:push", {
5920
+
detail,
5921
+
bubbles: true,
5922
+
cancelable: false
5923
+
}));
5924
+
if (phxEvent) {
5925
+
el.dispatchEvent(new CustomEvent(`phx:push:${phxEvent}`, {
5926
+
detail,
5927
+
bubbles: true,
5928
+
cancelable: false
5929
+
}));
5930
+
}
5931
+
}
5932
+
return [newRef, elements.map(({ el }) => el), opts];
5933
+
}
5934
+
isAcked(ref) {
5935
+
return this.lastAckRef !== null && this.lastAckRef >= ref;
5936
+
}
5937
+
componentID(el) {
5938
+
let cid = el.getAttribute && el.getAttribute(PHX_COMPONENT);
5939
+
return cid ? parseInt(cid) : null;
5940
+
}
5941
+
targetComponentID(target, targetCtx, opts = {}) {
5942
+
if (isCid(targetCtx)) {
5943
+
return targetCtx;
5944
+
}
5945
+
let cidOrSelector = opts.target || target.getAttribute(this.binding("target"));
5946
+
if (isCid(cidOrSelector)) {
5947
+
return parseInt(cidOrSelector);
5948
+
} else if (targetCtx && (cidOrSelector !== null || opts.target)) {
5949
+
return this.closestComponentID(targetCtx);
5950
+
} else {
5951
+
return null;
5952
+
}
5953
+
}
5954
+
closestComponentID(targetCtx) {
5955
+
if (isCid(targetCtx)) {
5956
+
return targetCtx;
5957
+
} else if (targetCtx) {
5958
+
return maybe(targetCtx.closest(`[${PHX_COMPONENT}]`), (el) => this.ownsElement(el) && this.componentID(el));
5959
+
} else {
5960
+
return null;
5961
+
}
5962
+
}
5963
+
pushHookEvent(el, targetCtx, event, payload, onReply) {
5964
+
if (!this.isConnected()) {
5965
+
this.log("hook", () => ["unable to push hook event. LiveView not connected", event, payload]);
5966
+
return false;
5967
+
}
5968
+
let [ref, els, opts] = this.putRef([{ el, loading: true, lock: true }], event, "hook");
5969
+
this.pushWithReply(() => [ref, els, opts], "event", {
5970
+
type: "hook",
5971
+
event,
5972
+
value: payload,
5973
+
cid: this.closestComponentID(targetCtx)
5974
+
}).then(({ resp: _resp, reply: hookReply }) => onReply(hookReply, ref));
5975
+
return ref;
5976
+
}
5977
+
extractMeta(el, meta, value) {
5978
+
let prefix = this.binding("value-");
5979
+
for (let i = 0; i < el.attributes.length; i++) {
5980
+
if (!meta) {
5981
+
meta = {};
5982
+
}
5983
+
let name = el.attributes[i].name;
5984
+
if (name.startsWith(prefix)) {
5985
+
meta[name.replace(prefix, "")] = el.getAttribute(name);
5986
+
}
5987
+
}
5988
+
if (el.value !== void 0 && !(el instanceof HTMLFormElement)) {
5989
+
if (!meta) {
5990
+
meta = {};
5991
+
}
5992
+
meta.value = el.value;
5993
+
if (el.tagName === "INPUT" && CHECKABLE_INPUTS.indexOf(el.type) >= 0 && !el.checked) {
5994
+
delete meta.value;
5995
+
}
5996
+
}
5997
+
if (value) {
5998
+
if (!meta) {
5999
+
meta = {};
6000
+
}
6001
+
for (let key in value) {
6002
+
meta[key] = value[key];
6003
+
}
6004
+
}
6005
+
return meta;
6006
+
}
6007
+
pushEvent(type, el, targetCtx, phxEvent, meta, opts = {}, onReply) {
6008
+
this.pushWithReply(() => this.putRef([{ el, loading: true, lock: true }], phxEvent, type, opts), "event", {
6009
+
type,
6010
+
event: phxEvent,
6011
+
value: this.extractMeta(el, meta, opts.value),
6012
+
cid: this.targetComponentID(el, targetCtx, opts)
6013
+
}).then(({ reply }) => onReply && onReply(reply)).catch((error) => logError("Failed to push event", error));
6014
+
}
6015
+
pushFileProgress(fileEl, entryRef, progress, onReply = function() {
6016
+
}) {
6017
+
this.liveSocket.withinOwners(fileEl.form, (view, targetCtx) => {
6018
+
view.pushWithReply(null, "progress", {
6019
+
event: fileEl.getAttribute(view.binding(PHX_PROGRESS)),
6020
+
ref: fileEl.getAttribute(PHX_UPLOAD_REF),
6021
+
entry_ref: entryRef,
6022
+
progress,
6023
+
cid: view.targetComponentID(fileEl.form, targetCtx)
6024
+
}).then(({ resp }) => onReply(resp)).catch((error) => logError("Failed to push file progress", error));
6025
+
});
6026
+
}
6027
+
pushInput(inputEl, targetCtx, forceCid, phxEvent, opts, callback) {
6028
+
if (!inputEl.form) {
6029
+
throw new Error("form events require the input to be inside a form");
6030
+
}
6031
+
let uploads;
6032
+
let cid = isCid(forceCid) ? forceCid : this.targetComponentID(inputEl.form, targetCtx, opts);
6033
+
let refGenerator = () => {
6034
+
return this.putRef([
6035
+
{ el: inputEl, loading: true, lock: true },
6036
+
{ el: inputEl.form, loading: true, lock: true }
6037
+
], phxEvent, "change", opts);
6038
+
};
6039
+
let formData;
6040
+
let meta = this.extractMeta(inputEl.form, {}, opts.value);
6041
+
let serializeOpts = {};
6042
+
if (inputEl instanceof HTMLButtonElement) {
6043
+
serializeOpts.submitter = inputEl;
6044
+
}
6045
+
if (inputEl.getAttribute(this.binding("change"))) {
6046
+
formData = serializeForm(inputEl.form, serializeOpts, [inputEl.name]);
6047
+
} else {
6048
+
formData = serializeForm(inputEl.form, serializeOpts);
6049
+
}
6050
+
if (dom_default.isUploadInput(inputEl) && inputEl.files && inputEl.files.length > 0) {
6051
+
LiveUploader.trackFiles(inputEl, Array.from(inputEl.files));
6052
+
}
6053
+
uploads = LiveUploader.serializeUploads(inputEl);
6054
+
let event = {
6055
+
type: "form",
6056
+
event: phxEvent,
6057
+
value: formData,
6058
+
meta: __spreadValues({
6059
+
// no target was implicitly sent as "undefined" in LV <= 1.0.5, therefore
6060
+
// we have to keep it. In 1.0.6 we switched from passing meta as URL encoded data
6061
+
// to passing it directly in the event, but the JSON encode would drop keys with
6062
+
// undefined values.
6063
+
_target: opts._target || "undefined"
6064
+
}, meta),
6065
+
uploads,
6066
+
cid
6067
+
};
6068
+
this.pushWithReply(refGenerator, "event", event).then(({ resp }) => {
6069
+
if (dom_default.isUploadInput(inputEl) && dom_default.isAutoUpload(inputEl)) {
6070
+
ElementRef.onUnlock(inputEl, () => {
6071
+
if (LiveUploader.filesAwaitingPreflight(inputEl).length > 0) {
6072
+
let [ref, _els] = refGenerator();
6073
+
this.undoRefs(ref, phxEvent, [inputEl.form]);
6074
+
this.uploadFiles(inputEl.form, phxEvent, targetCtx, ref, cid, (_uploads) => {
6075
+
callback && callback(resp);
6076
+
this.triggerAwaitingSubmit(inputEl.form, phxEvent);
6077
+
this.undoRefs(ref, phxEvent);
6078
+
});
6079
+
}
6080
+
});
6081
+
} else {
6082
+
callback && callback(resp);
6083
+
}
6084
+
}).catch((error) => logError("Failed to push input event", error));
6085
+
}
6086
+
triggerAwaitingSubmit(formEl, phxEvent) {
6087
+
let awaitingSubmit = this.getScheduledSubmit(formEl);
6088
+
if (awaitingSubmit) {
6089
+
let [_el, _ref, _opts, callback] = awaitingSubmit;
6090
+
this.cancelSubmit(formEl, phxEvent);
6091
+
callback();
6092
+
}
6093
+
}
6094
+
getScheduledSubmit(formEl) {
6095
+
return this.formSubmits.find(([el, _ref, _opts, _callback]) => el.isSameNode(formEl));
6096
+
}
6097
+
scheduleSubmit(formEl, ref, opts, callback) {
6098
+
if (this.getScheduledSubmit(formEl)) {
6099
+
return true;
6100
+
}
6101
+
this.formSubmits.push([formEl, ref, opts, callback]);
6102
+
}
6103
+
cancelSubmit(formEl, phxEvent) {
6104
+
this.formSubmits = this.formSubmits.filter(([el, ref, _opts, _callback]) => {
6105
+
if (el.isSameNode(formEl)) {
6106
+
this.undoRefs(ref, phxEvent);
6107
+
return false;
6108
+
} else {
6109
+
return true;
6110
+
}
6111
+
});
6112
+
}
6113
+
disableForm(formEl, phxEvent, opts = {}) {
6114
+
let filterIgnored = (el) => {
6115
+
let userIgnored = closestPhxBinding(el, `${this.binding(PHX_UPDATE)}=ignore`, el.form);
6116
+
return !(userIgnored || closestPhxBinding(el, "data-phx-update=ignore", el.form));
6117
+
};
6118
+
let filterDisables = (el) => {
6119
+
return el.hasAttribute(this.binding(PHX_DISABLE_WITH));
6120
+
};
6121
+
let filterButton = (el) => el.tagName == "BUTTON";
6122
+
let filterInput = (el) => ["INPUT", "TEXTAREA", "SELECT"].includes(el.tagName);
6123
+
let formElements = Array.from(formEl.elements);
6124
+
let disables = formElements.filter(filterDisables);
6125
+
let buttons = formElements.filter(filterButton).filter(filterIgnored);
6126
+
let inputs = formElements.filter(filterInput).filter(filterIgnored);
6127
+
buttons.forEach((button) => {
6128
+
button.setAttribute(PHX_DISABLED, button.disabled);
6129
+
button.disabled = true;
6130
+
});
6131
+
inputs.forEach((input) => {
6132
+
input.setAttribute(PHX_READONLY, input.readOnly);
6133
+
input.readOnly = true;
6134
+
if (input.files) {
6135
+
input.setAttribute(PHX_DISABLED, input.disabled);
6136
+
input.disabled = true;
6137
+
}
6138
+
});
6139
+
let formEls = disables.concat(buttons).concat(inputs).map((el) => {
6140
+
return { el, loading: true, lock: true };
6141
+
});
6142
+
let els = [{ el: formEl, loading: true, lock: false }].concat(formEls).reverse();
6143
+
return this.putRef(els, phxEvent, "submit", opts);
6144
+
}
6145
+
pushFormSubmit(formEl, targetCtx, phxEvent, submitter, opts, onReply) {
6146
+
let refGenerator = () => this.disableForm(formEl, phxEvent, __spreadProps(__spreadValues({}, opts), {
6147
+
form: formEl,
6148
+
submitter
6149
+
}));
6150
+
dom_default.putPrivate(formEl, "submitter", submitter);
6151
+
let cid = this.targetComponentID(formEl, targetCtx);
6152
+
if (LiveUploader.hasUploadsInProgress(formEl)) {
6153
+
let [ref, _els] = refGenerator();
6154
+
let push = () => this.pushFormSubmit(formEl, targetCtx, phxEvent, submitter, opts, onReply);
6155
+
return this.scheduleSubmit(formEl, ref, opts, push);
6156
+
} else if (LiveUploader.inputsAwaitingPreflight(formEl).length > 0) {
6157
+
let [ref, els] = refGenerator();
6158
+
let proxyRefGen = () => [ref, els, opts];
6159
+
this.uploadFiles(formEl, phxEvent, targetCtx, ref, cid, (_uploads) => {
6160
+
if (LiveUploader.inputsAwaitingPreflight(formEl).length > 0) {
6161
+
return this.undoRefs(ref, phxEvent);
6162
+
}
6163
+
let meta = this.extractMeta(formEl, {}, opts.value);
6164
+
let formData = serializeForm(formEl, { submitter });
6165
+
this.pushWithReply(proxyRefGen, "event", {
6166
+
type: "form",
6167
+
event: phxEvent,
6168
+
value: formData,
6169
+
meta,
6170
+
cid
6171
+
}).then(({ resp }) => onReply(resp)).catch((error) => logError("Failed to push form submit", error));
6172
+
});
6173
+
} else if (!(formEl.hasAttribute(PHX_REF_SRC) && formEl.classList.contains("phx-submit-loading"))) {
6174
+
let meta = this.extractMeta(formEl, {}, opts.value);
6175
+
let formData = serializeForm(formEl, { submitter });
6176
+
this.pushWithReply(refGenerator, "event", {
6177
+
type: "form",
6178
+
event: phxEvent,
6179
+
value: formData,
6180
+
meta,
6181
+
cid
6182
+
}).then(({ resp }) => onReply(resp)).catch((error) => logError("Failed to push form submit", error));
6183
+
}
6184
+
}
6185
+
uploadFiles(formEl, phxEvent, targetCtx, ref, cid, onComplete) {
6186
+
let joinCountAtUpload = this.joinCount;
6187
+
let inputEls = LiveUploader.activeFileInputs(formEl);
6188
+
let numFileInputsInProgress = inputEls.length;
6189
+
inputEls.forEach((inputEl) => {
6190
+
let uploader = new LiveUploader(inputEl, this, () => {
6191
+
numFileInputsInProgress--;
6192
+
if (numFileInputsInProgress === 0) {
6193
+
onComplete();
6194
+
}
6195
+
});
6196
+
let entries = uploader.entries().map((entry) => entry.toPreflightPayload());
6197
+
if (entries.length === 0) {
6198
+
numFileInputsInProgress--;
6199
+
return;
6200
+
}
6201
+
let payload = {
6202
+
ref: inputEl.getAttribute(PHX_UPLOAD_REF),
6203
+
entries,
6204
+
cid: this.targetComponentID(inputEl.form, targetCtx)
6205
+
};
6206
+
this.log("upload", () => ["sending preflight request", payload]);
6207
+
this.pushWithReply(null, "allow_upload", payload).then(({ resp }) => {
6208
+
this.log("upload", () => ["got preflight response", resp]);
6209
+
uploader.entries().forEach((entry) => {
6210
+
if (resp.entries && !resp.entries[entry.ref]) {
6211
+
this.handleFailedEntryPreflight(entry.ref, "failed preflight", uploader);
6212
+
}
6213
+
});
6214
+
if (resp.error || Object.keys(resp.entries).length === 0) {
6215
+
this.undoRefs(ref, phxEvent);
6216
+
let errors = resp.error || [];
6217
+
errors.map(([entry_ref, reason]) => {
6218
+
this.handleFailedEntryPreflight(entry_ref, reason, uploader);
6219
+
});
6220
+
} else {
6221
+
let onError = (callback) => {
6222
+
this.channel.onError(() => {
6223
+
if (this.joinCount === joinCountAtUpload) {
6224
+
callback();
6225
+
}
6226
+
});
6227
+
};
6228
+
uploader.initAdapterUpload(resp, onError, this.liveSocket);
6229
+
}
6230
+
}).catch((error) => logError("Failed to push upload", error));
6231
+
});
6232
+
}
6233
+
handleFailedEntryPreflight(uploadRef, reason, uploader) {
6234
+
if (uploader.isAutoUpload()) {
6235
+
let entry = uploader.entries().find((entry2) => entry2.ref === uploadRef.toString());
6236
+
if (entry) {
6237
+
entry.cancel();
6238
+
}
6239
+
} else {
6240
+
uploader.entries().map((entry) => entry.cancel());
6241
+
}
6242
+
this.log("upload", () => [`error for entry ${uploadRef}`, reason]);
6243
+
}
6244
+
dispatchUploads(targetCtx, name, filesOrBlobs) {
6245
+
let targetElement = this.targetCtxElement(targetCtx) || this.el;
6246
+
let inputs = dom_default.findUploadInputs(targetElement).filter((el) => el.name === name);
6247
+
if (inputs.length === 0) {
6248
+
logError(`no live file inputs found matching the name "${name}"`);
6249
+
} else if (inputs.length > 1) {
6250
+
logError(`duplicate live file inputs found matching the name "${name}"`);
6251
+
} else {
6252
+
dom_default.dispatchEvent(inputs[0], PHX_TRACK_UPLOADS, { detail: { files: filesOrBlobs } });
6253
+
}
6254
+
}
6255
+
targetCtxElement(targetCtx) {
6256
+
if (isCid(targetCtx)) {
6257
+
let [target] = dom_default.findComponentNodeList(this.el, targetCtx);
6258
+
return target;
6259
+
} else if (targetCtx) {
6260
+
return targetCtx;
6261
+
} else {
6262
+
return null;
6263
+
}
6264
+
}
6265
+
pushFormRecovery(oldForm, newForm, templateDom, callback) {
6266
+
const phxChange = this.binding("change");
6267
+
const phxTarget = newForm.getAttribute(this.binding("target")) || newForm;
6268
+
const phxEvent = newForm.getAttribute(this.binding(PHX_AUTO_RECOVER)) || newForm.getAttribute(this.binding("change"));
6269
+
const inputs = Array.from(oldForm.elements).filter((el) => dom_default.isFormInput(el) && el.name && !el.hasAttribute(phxChange));
6270
+
if (inputs.length === 0) {
6271
+
callback();
6272
+
return;
6273
+
}
6274
+
inputs.forEach((input2) => input2.hasAttribute(PHX_UPLOAD_REF) && LiveUploader.clearFiles(input2));
6275
+
let input = inputs.find((el) => el.type !== "hidden") || inputs[0];
6276
+
let pending = 0;
6277
+
this.withinTargets(phxTarget, (targetView, targetCtx) => {
6278
+
const cid = this.targetComponentID(newForm, targetCtx);
6279
+
pending++;
6280
+
let e = new CustomEvent("phx:form-recovery", { detail: { sourceElement: oldForm } });
6281
+
js_default.exec(e, "change", phxEvent, this, input, ["push", {
6282
+
_target: input.name,
6283
+
targetView,
6284
+
targetCtx,
6285
+
newCid: cid,
6286
+
callback: () => {
6287
+
pending--;
6288
+
if (pending === 0) {
6289
+
callback();
6290
+
}
6291
+
}
6292
+
}]);
6293
+
}, templateDom, templateDom);
6294
+
}
6295
+
pushLinkPatch(e, href, targetEl, callback) {
6296
+
let linkRef = this.liveSocket.setPendingLink(href);
6297
+
let loading = e.isTrusted && e.type !== "popstate";
6298
+
let refGen = targetEl ? () => this.putRef([{ el: targetEl, loading, lock: true }], null, "click") : null;
6299
+
let fallback = () => this.liveSocket.redirect(window.location.href);
6300
+
let url = href.startsWith("/") ? `${location.protocol}//${location.host}${href}` : href;
6301
+
this.pushWithReply(refGen, "live_patch", { url }).then(
6302
+
({ resp }) => {
6303
+
this.liveSocket.requestDOMUpdate(() => {
6304
+
if (resp.link_redirect) {
6305
+
this.liveSocket.replaceMain(href, null, callback, linkRef);
6306
+
} else {
6307
+
if (this.liveSocket.commitPendingLink(linkRef)) {
6308
+
this.href = href;
6309
+
}
6310
+
this.applyPendingUpdates();
6311
+
callback && callback(linkRef);
6312
+
}
6313
+
});
6314
+
},
6315
+
({ error: _error, timeout: _timeout }) => fallback()
6316
+
);
6317
+
}
6318
+
getFormsForRecovery() {
6319
+
if (this.joinCount === 0) {
6320
+
return {};
6321
+
}
6322
+
let phxChange = this.binding("change");
6323
+
return dom_default.all(this.el, `form[${phxChange}]`).filter((form) => form.id).filter((form) => form.elements.length > 0).filter((form) => form.getAttribute(this.binding(PHX_AUTO_RECOVER)) !== "ignore").map((form) => {
6324
+
const clonedForm = form.cloneNode(false);
6325
+
dom_default.copyPrivates(clonedForm, form);
6326
+
Array.from(form.elements).forEach((el) => {
6327
+
const clonedEl = el.cloneNode(true);
6328
+
morphdom_esm_default(clonedEl, el);
6329
+
dom_default.copyPrivates(clonedEl, el);
6330
+
clonedForm.appendChild(clonedEl);
6331
+
});
6332
+
return clonedForm;
6333
+
}).reduce((acc, form) => {
6334
+
acc[form.id] = form;
6335
+
return acc;
6336
+
}, {});
6337
+
}
6338
+
maybePushComponentsDestroyed(destroyedCIDs) {
6339
+
let willDestroyCIDs = destroyedCIDs.filter((cid) => {
6340
+
return dom_default.findComponentNodeList(this.el, cid).length === 0;
6341
+
});
6342
+
if (willDestroyCIDs.length > 0) {
6343
+
willDestroyCIDs.forEach((cid) => this.rendered.resetRender(cid));
6344
+
this.pushWithReply(null, "cids_will_destroy", { cids: willDestroyCIDs }).then(() => {
6345
+
this.liveSocket.requestDOMUpdate(() => {
6346
+
let completelyDestroyCIDs = willDestroyCIDs.filter((cid) => {
6347
+
return dom_default.findComponentNodeList(this.el, cid).length === 0;
6348
+
});
6349
+
if (completelyDestroyCIDs.length > 0) {
6350
+
this.pushWithReply(null, "cids_destroyed", { cids: completelyDestroyCIDs }).then(({ resp }) => {
6351
+
this.rendered.pruneCIDs(resp.cids);
6352
+
}).catch((error) => logError("Failed to push components destroyed", error));
6353
+
}
6354
+
});
6355
+
}).catch((error) => logError("Failed to push components destroyed", error));
6356
+
}
6357
+
}
6358
+
ownsElement(el) {
6359
+
let parentViewEl = el.closest(PHX_VIEW_SELECTOR);
6360
+
return el.getAttribute(PHX_PARENT_ID) === this.id || parentViewEl && parentViewEl.id === this.id || !parentViewEl && this.isDead;
6361
+
}
6362
+
submitForm(form, targetCtx, phxEvent, submitter, opts = {}) {
6363
+
dom_default.putPrivate(form, PHX_HAS_SUBMITTED, true);
6364
+
const inputs = Array.from(form.elements);
6365
+
inputs.forEach((input) => dom_default.putPrivate(input, PHX_HAS_SUBMITTED, true));
6366
+
this.liveSocket.blurActiveElement(this);
6367
+
this.pushFormSubmit(form, targetCtx, phxEvent, submitter, opts, () => {
6368
+
this.liveSocket.restorePreviouslyActiveFocus();
6369
+
});
6370
+
}
6371
+
binding(kind) {
6372
+
return this.liveSocket.binding(kind);
6373
+
}
6374
+
};
6375
+
var LiveSocket = class {
6376
+
constructor(url, phxSocket, opts = {}) {
6377
+
this.unloaded = false;
6378
+
if (!phxSocket || phxSocket.constructor.name === "Object") {
6379
+
throw new Error(`
6380
+
a phoenix Socket must be provided as the second argument to the LiveSocket constructor. For example:
6381
+
6382
+
import {Socket} from "phoenix"
6383
+
import {LiveSocket} from "phoenix_live_view"
6384
+
let liveSocket = new LiveSocket("/live", Socket, {...})
6385
+
`);
6386
+
}
6387
+
this.socket = new phxSocket(url, opts);
6388
+
this.bindingPrefix = opts.bindingPrefix || BINDING_PREFIX;
6389
+
this.opts = opts;
6390
+
this.params = closure2(opts.params || {});
6391
+
this.viewLogger = opts.viewLogger;
6392
+
this.metadataCallbacks = opts.metadata || {};
6393
+
this.defaults = Object.assign(clone(DEFAULTS), opts.defaults || {});
6394
+
this.activeElement = null;
6395
+
this.prevActive = null;
6396
+
this.silenced = false;
6397
+
this.main = null;
6398
+
this.outgoingMainEl = null;
6399
+
this.clickStartedAtTarget = null;
6400
+
this.linkRef = 1;
6401
+
this.roots = {};
6402
+
this.href = window.location.href;
6403
+
this.pendingLink = null;
6404
+
this.currentLocation = clone(window.location);
6405
+
this.hooks = opts.hooks || {};
6406
+
this.uploaders = opts.uploaders || {};
6407
+
this.loaderTimeout = opts.loaderTimeout || LOADER_TIMEOUT;
6408
+
this.disconnectedTimeout = opts.disconnectedTimeout || DISCONNECTED_TIMEOUT;
6409
+
this.reloadWithJitterTimer = null;
6410
+
this.maxReloads = opts.maxReloads || MAX_RELOADS;
6411
+
this.reloadJitterMin = opts.reloadJitterMin || RELOAD_JITTER_MIN;
6412
+
this.reloadJitterMax = opts.reloadJitterMax || RELOAD_JITTER_MAX;
6413
+
this.failsafeJitter = opts.failsafeJitter || FAILSAFE_JITTER;
6414
+
this.localStorage = opts.localStorage || window.localStorage;
6415
+
this.sessionStorage = opts.sessionStorage || window.sessionStorage;
6416
+
this.boundTopLevelEvents = false;
6417
+
this.boundEventNames = /* @__PURE__ */ new Set();
6418
+
this.serverCloseRef = null;
6419
+
this.domCallbacks = Object.assign(
6420
+
{
6421
+
jsQuerySelectorAll: null,
6422
+
onPatchStart: closure2(),
6423
+
onPatchEnd: closure2(),
6424
+
onNodeAdded: closure2(),
6425
+
onBeforeElUpdated: closure2()
6426
+
},
6427
+
opts.dom || {}
6428
+
);
6429
+
this.transitions = new TransitionSet();
6430
+
this.currentHistoryPosition = parseInt(this.sessionStorage.getItem(PHX_LV_HISTORY_POSITION)) || 0;
6431
+
window.addEventListener("pagehide", (_e) => {
6432
+
this.unloaded = true;
6433
+
});
6434
+
this.socket.onOpen(() => {
6435
+
if (this.isUnloaded()) {
6436
+
window.location.reload();
6437
+
}
6438
+
});
6439
+
}
6440
+
// public
6441
+
version() {
6442
+
return "1.0.17";
6443
+
}
6444
+
isProfileEnabled() {
6445
+
return this.sessionStorage.getItem(PHX_LV_PROFILE) === "true";
6446
+
}
6447
+
isDebugEnabled() {
6448
+
return this.sessionStorage.getItem(PHX_LV_DEBUG) === "true";
6449
+
}
6450
+
isDebugDisabled() {
6451
+
return this.sessionStorage.getItem(PHX_LV_DEBUG) === "false";
6452
+
}
6453
+
enableDebug() {
6454
+
this.sessionStorage.setItem(PHX_LV_DEBUG, "true");
6455
+
}
6456
+
enableProfiling() {
6457
+
this.sessionStorage.setItem(PHX_LV_PROFILE, "true");
6458
+
}
6459
+
disableDebug() {
6460
+
this.sessionStorage.setItem(PHX_LV_DEBUG, "false");
6461
+
}
6462
+
disableProfiling() {
6463
+
this.sessionStorage.removeItem(PHX_LV_PROFILE);
6464
+
}
6465
+
enableLatencySim(upperBoundMs) {
6466
+
this.enableDebug();
6467
+
console.log("latency simulator enabled for the duration of this browser session. Call disableLatencySim() to disable");
6468
+
this.sessionStorage.setItem(PHX_LV_LATENCY_SIM, upperBoundMs);
6469
+
}
6470
+
disableLatencySim() {
6471
+
this.sessionStorage.removeItem(PHX_LV_LATENCY_SIM);
6472
+
}
6473
+
getLatencySim() {
6474
+
let str = this.sessionStorage.getItem(PHX_LV_LATENCY_SIM);
6475
+
return str ? parseInt(str) : null;
6476
+
}
6477
+
getSocket() {
6478
+
return this.socket;
6479
+
}
6480
+
connect() {
6481
+
if (window.location.hostname === "localhost" && !this.isDebugDisabled()) {
6482
+
this.enableDebug();
6483
+
}
6484
+
let doConnect = () => {
6485
+
this.resetReloadStatus();
6486
+
if (this.joinRootViews()) {
6487
+
this.bindTopLevelEvents();
6488
+
this.socket.connect();
6489
+
} else if (this.main) {
6490
+
this.socket.connect();
6491
+
} else {
6492
+
this.bindTopLevelEvents({ dead: true });
6493
+
}
6494
+
this.joinDeadView();
6495
+
};
6496
+
if (["complete", "loaded", "interactive"].indexOf(document.readyState) >= 0) {
6497
+
doConnect();
6498
+
} else {
6499
+
document.addEventListener("DOMContentLoaded", () => doConnect());
6500
+
}
6501
+
}
6502
+
disconnect(callback) {
6503
+
clearTimeout(this.reloadWithJitterTimer);
6504
+
if (this.serverCloseRef) {
6505
+
this.socket.off(this.serverCloseRef);
6506
+
this.serverCloseRef = null;
6507
+
}
6508
+
this.socket.disconnect(callback);
6509
+
}
6510
+
replaceTransport(transport) {
6511
+
clearTimeout(this.reloadWithJitterTimer);
6512
+
this.socket.replaceTransport(transport);
6513
+
this.connect();
6514
+
}
6515
+
execJS(el, encodedJS, eventType = null) {
6516
+
let e = new CustomEvent("phx:exec", { detail: { sourceElement: el } });
6517
+
this.owner(el, (view) => js_default.exec(e, eventType, encodedJS, view, el));
6518
+
}
6519
+
// private
6520
+
execJSHookPush(el, phxEvent, data, callback) {
6521
+
this.withinOwners(el, (view) => {
6522
+
let e = new CustomEvent("phx:exec", { detail: { sourceElement: el } });
6523
+
js_default.exec(e, "hook", phxEvent, view, el, ["push", { data, callback }]);
6524
+
});
6525
+
}
6526
+
unload() {
6527
+
if (this.unloaded) {
6528
+
return;
6529
+
}
6530
+
if (this.main && this.isConnected()) {
6531
+
this.log(this.main, "socket", () => ["disconnect for page nav"]);
6532
+
}
6533
+
this.unloaded = true;
6534
+
this.destroyAllViews();
6535
+
this.disconnect();
6536
+
}
6537
+
triggerDOM(kind, args) {
6538
+
this.domCallbacks[kind](...args);
6539
+
}
6540
+
time(name, func) {
6541
+
if (!this.isProfileEnabled() || !console.time) {
6542
+
return func();
6543
+
}
6544
+
console.time(name);
6545
+
let result = func();
6546
+
console.timeEnd(name);
6547
+
return result;
6548
+
}
6549
+
log(view, kind, msgCallback) {
6550
+
if (this.viewLogger) {
6551
+
let [msg, obj] = msgCallback();
6552
+
this.viewLogger(view, kind, msg, obj);
6553
+
} else if (this.isDebugEnabled()) {
6554
+
let [msg, obj] = msgCallback();
6555
+
debug(view, kind, msg, obj);
6556
+
}
6557
+
}
6558
+
requestDOMUpdate(callback) {
6559
+
this.transitions.after(callback);
6560
+
}
6561
+
transition(time, onStart, onDone = function() {
6562
+
}) {
6563
+
this.transitions.addTransition(time, onStart, onDone);
6564
+
}
6565
+
onChannel(channel, event, cb) {
6566
+
channel.on(event, (data) => {
6567
+
let latency = this.getLatencySim();
6568
+
if (!latency) {
6569
+
cb(data);
6570
+
} else {
6571
+
setTimeout(() => cb(data), latency);
6572
+
}
6573
+
});
6574
+
}
6575
+
reloadWithJitter(view, log) {
6576
+
clearTimeout(this.reloadWithJitterTimer);
6577
+
this.disconnect();
6578
+
let minMs = this.reloadJitterMin;
6579
+
let maxMs = this.reloadJitterMax;
6580
+
let afterMs = Math.floor(Math.random() * (maxMs - minMs + 1)) + minMs;
6581
+
let tries = browser_default.updateLocal(this.localStorage, window.location.pathname, CONSECUTIVE_RELOADS, 0, (count) => count + 1);
6582
+
if (tries >= this.maxReloads) {
6583
+
afterMs = this.failsafeJitter;
6584
+
}
6585
+
this.reloadWithJitterTimer = setTimeout(() => {
6586
+
if (view.isDestroyed() || view.isConnected()) {
6587
+
return;
6588
+
}
6589
+
view.destroy();
6590
+
log ? log() : this.log(view, "join", () => [`encountered ${tries} consecutive reloads`]);
6591
+
if (tries >= this.maxReloads) {
6592
+
this.log(view, "join", () => [`exceeded ${this.maxReloads} consecutive reloads. Entering failsafe mode`]);
6593
+
}
6594
+
if (this.hasPendingLink()) {
6595
+
window.location = this.pendingLink;
6596
+
} else {
6597
+
window.location.reload();
6598
+
}
6599
+
}, afterMs);
6600
+
}
6601
+
getHookCallbacks(name) {
6602
+
return name && name.startsWith("Phoenix.") ? hooks_default[name.split(".")[1]] : this.hooks[name];
6603
+
}
6604
+
isUnloaded() {
6605
+
return this.unloaded;
6606
+
}
6607
+
isConnected() {
6608
+
return this.socket.isConnected();
6609
+
}
6610
+
getBindingPrefix() {
6611
+
return this.bindingPrefix;
6612
+
}
6613
+
binding(kind) {
6614
+
return `${this.getBindingPrefix()}${kind}`;
6615
+
}
6616
+
channel(topic, params) {
6617
+
return this.socket.channel(topic, params);
6618
+
}
6619
+
joinDeadView() {
6620
+
let body = document.body;
6621
+
if (body && !this.isPhxView(body) && !this.isPhxView(document.firstElementChild)) {
6622
+
let view = this.newRootView(body);
6623
+
view.setHref(this.getHref());
6624
+
view.joinDead();
6625
+
if (!this.main) {
6626
+
this.main = view;
6627
+
}
6628
+
window.requestAnimationFrame(() => {
6629
+
var _a;
6630
+
view.execNewMounted();
6631
+
this.maybeScroll((_a = history.state) == null ? void 0 : _a.scroll);
6632
+
});
6633
+
}
6634
+
}
6635
+
joinRootViews() {
6636
+
let rootsFound = false;
6637
+
dom_default.all(document, `${PHX_VIEW_SELECTOR}:not([${PHX_PARENT_ID}])`, (rootEl) => {
6638
+
if (!this.getRootById(rootEl.id)) {
6639
+
let view = this.newRootView(rootEl);
6640
+
if (!dom_default.isPhxSticky(rootEl)) {
6641
+
view.setHref(this.getHref());
6642
+
}
6643
+
view.join();
6644
+
if (rootEl.hasAttribute(PHX_MAIN)) {
6645
+
this.main = view;
6646
+
}
6647
+
}
6648
+
rootsFound = true;
6649
+
});
6650
+
return rootsFound;
6651
+
}
6652
+
redirect(to, flash, reloadToken) {
6653
+
if (reloadToken) {
6654
+
browser_default.setCookie(PHX_RELOAD_STATUS, reloadToken, 60);
6655
+
}
6656
+
this.unload();
6657
+
browser_default.redirect(to, flash);
6658
+
}
6659
+
replaceMain(href, flash, callback = null, linkRef = this.setPendingLink(href)) {
6660
+
const liveReferer = this.currentLocation.href;
6661
+
this.outgoingMainEl = this.outgoingMainEl || this.main.el;
6662
+
const stickies = dom_default.findPhxSticky(document) || [];
6663
+
const removeEls = dom_default.all(this.outgoingMainEl, `[${this.binding("remove")}]`).filter((el) => !dom_default.isChildOfAny(el, stickies));
6664
+
const newMainEl = dom_default.cloneNode(this.outgoingMainEl, "");
6665
+
this.main.showLoader(this.loaderTimeout);
6666
+
this.main.destroy();
6667
+
this.main = this.newRootView(newMainEl, flash, liveReferer);
6668
+
this.main.setRedirect(href);
6669
+
this.transitionRemoves(removeEls);
6670
+
this.main.join((joinCount, onDone) => {
6671
+
if (joinCount === 1 && this.commitPendingLink(linkRef)) {
6672
+
this.requestDOMUpdate(() => {
6673
+
removeEls.forEach((el) => el.remove());
6674
+
stickies.forEach((el) => newMainEl.appendChild(el));
6675
+
this.outgoingMainEl.replaceWith(newMainEl);
6676
+
this.outgoingMainEl = null;
6677
+
callback && callback(linkRef);
6678
+
onDone();
6679
+
});
6680
+
}
6681
+
});
6682
+
}
6683
+
transitionRemoves(elements, callback) {
6684
+
let removeAttr = this.binding("remove");
6685
+
let silenceEvents = (e) => {
6686
+
e.preventDefault();
6687
+
e.stopImmediatePropagation();
6688
+
};
6689
+
elements.forEach((el) => {
6690
+
for (let event of this.boundEventNames) {
6691
+
el.addEventListener(event, silenceEvents, true);
6692
+
}
6693
+
this.execJS(el, el.getAttribute(removeAttr), "remove");
6694
+
});
6695
+
this.requestDOMUpdate(() => {
6696
+
elements.forEach((el) => {
6697
+
for (let event of this.boundEventNames) {
6698
+
el.removeEventListener(event, silenceEvents, true);
6699
+
}
6700
+
});
6701
+
callback && callback();
6702
+
});
6703
+
}
6704
+
isPhxView(el) {
6705
+
return el.getAttribute && el.getAttribute(PHX_SESSION) !== null;
6706
+
}
6707
+
newRootView(el, flash, liveReferer) {
6708
+
let view = new View(el, this, null, flash, liveReferer);
6709
+
this.roots[view.id] = view;
6710
+
return view;
6711
+
}
6712
+
owner(childEl, callback) {
6713
+
let view;
6714
+
const closestViewEl = childEl.closest(PHX_VIEW_SELECTOR);
6715
+
if (closestViewEl) {
6716
+
view = this.getViewByEl(closestViewEl);
6717
+
} else {
6718
+
view = this.main;
6719
+
}
6720
+
return view && callback ? callback(view) : view;
6721
+
}
6722
+
withinOwners(childEl, callback) {
6723
+
this.owner(childEl, (view) => callback(view, childEl));
6724
+
}
6725
+
getViewByEl(el) {
6726
+
let rootId = el.getAttribute(PHX_ROOT_ID);
6727
+
return maybe(this.getRootById(rootId), (root) => root.getDescendentByEl(el));
6728
+
}
6729
+
getRootById(id) {
6730
+
return this.roots[id];
6731
+
}
6732
+
destroyAllViews() {
6733
+
for (let id in this.roots) {
6734
+
this.roots[id].destroy();
6735
+
delete this.roots[id];
6736
+
}
6737
+
this.main = null;
6738
+
}
6739
+
destroyViewByEl(el) {
6740
+
let root = this.getRootById(el.getAttribute(PHX_ROOT_ID));
6741
+
if (root && root.id === el.id) {
6742
+
root.destroy();
6743
+
delete this.roots[root.id];
6744
+
} else if (root) {
6745
+
root.destroyDescendent(el.id);
6746
+
}
6747
+
}
6748
+
getActiveElement() {
6749
+
return document.activeElement;
6750
+
}
6751
+
dropActiveElement(view) {
6752
+
if (this.prevActive && view.ownsElement(this.prevActive)) {
6753
+
this.prevActive = null;
6754
+
}
6755
+
}
6756
+
restorePreviouslyActiveFocus() {
6757
+
if (this.prevActive && this.prevActive !== document.body) {
6758
+
this.prevActive.focus();
6759
+
}
6760
+
}
6761
+
blurActiveElement() {
6762
+
this.prevActive = this.getActiveElement();
6763
+
if (this.prevActive !== document.body) {
6764
+
this.prevActive.blur();
6765
+
}
6766
+
}
6767
+
bindTopLevelEvents({ dead } = {}) {
6768
+
if (this.boundTopLevelEvents) {
6769
+
return;
6770
+
}
6771
+
this.boundTopLevelEvents = true;
6772
+
this.serverCloseRef = this.socket.onClose((event) => {
6773
+
if (event && event.code === 1e3 && this.main) {
6774
+
return this.reloadWithJitter(this.main);
6775
+
}
6776
+
});
6777
+
document.body.addEventListener("click", function() {
6778
+
});
6779
+
window.addEventListener("pageshow", (e) => {
6780
+
if (e.persisted) {
6781
+
this.getSocket().disconnect();
6782
+
this.withPageLoading({ to: window.location.href, kind: "redirect" });
6783
+
window.location.reload();
6784
+
}
6785
+
}, true);
6786
+
if (!dead) {
6787
+
this.bindNav();
6788
+
}
6789
+
this.bindClicks();
6790
+
if (!dead) {
6791
+
this.bindForms();
6792
+
}
6793
+
this.bind({ keyup: "keyup", keydown: "keydown" }, (e, type, view, targetEl, phxEvent, _phxTarget) => {
6794
+
let matchKey = targetEl.getAttribute(this.binding(PHX_KEY));
6795
+
let pressedKey = e.key && e.key.toLowerCase();
6796
+
if (matchKey && matchKey.toLowerCase() !== pressedKey) {
6797
+
return;
6798
+
}
6799
+
let data = __spreadValues({ key: e.key }, this.eventMeta(type, e, targetEl));
6800
+
js_default.exec(e, type, phxEvent, view, targetEl, ["push", { data }]);
6801
+
});
6802
+
this.bind({ blur: "focusout", focus: "focusin" }, (e, type, view, targetEl, phxEvent, phxTarget) => {
6803
+
if (!phxTarget) {
6804
+
let data = __spreadValues({ key: e.key }, this.eventMeta(type, e, targetEl));
6805
+
js_default.exec(e, type, phxEvent, view, targetEl, ["push", { data }]);
6806
+
}
6807
+
});
6808
+
this.bind({ blur: "blur", focus: "focus" }, (e, type, view, targetEl, phxEvent, phxTarget) => {
6809
+
if (phxTarget === "window") {
6810
+
let data = this.eventMeta(type, e, targetEl);
6811
+
js_default.exec(e, type, phxEvent, view, targetEl, ["push", { data }]);
6812
+
}
6813
+
});
6814
+
this.on("dragover", (e) => e.preventDefault());
6815
+
this.on("drop", (e) => {
6816
+
e.preventDefault();
6817
+
let dropTargetId = maybe(closestPhxBinding(e.target, this.binding(PHX_DROP_TARGET)), (trueTarget) => {
6818
+
return trueTarget.getAttribute(this.binding(PHX_DROP_TARGET));
6819
+
});
6820
+
let dropTarget = dropTargetId && document.getElementById(dropTargetId);
6821
+
let files = Array.from(e.dataTransfer.files || []);
6822
+
if (!dropTarget || dropTarget.disabled || files.length === 0 || !(dropTarget.files instanceof FileList)) {
6823
+
return;
6824
+
}
6825
+
LiveUploader.trackFiles(dropTarget, files, e.dataTransfer);
6826
+
dropTarget.dispatchEvent(new Event("input", { bubbles: true }));
6827
+
});
6828
+
this.on(PHX_TRACK_UPLOADS, (e) => {
6829
+
let uploadTarget = e.target;
6830
+
if (!dom_default.isUploadInput(uploadTarget)) {
6831
+
return;
6832
+
}
6833
+
let files = Array.from(e.detail.files || []).filter((f) => f instanceof File || f instanceof Blob);
6834
+
LiveUploader.trackFiles(uploadTarget, files);
6835
+
uploadTarget.dispatchEvent(new Event("input", { bubbles: true }));
6836
+
});
6837
+
}
6838
+
eventMeta(eventName, e, targetEl) {
6839
+
let callback = this.metadataCallbacks[eventName];
6840
+
return callback ? callback(e, targetEl) : {};
6841
+
}
6842
+
setPendingLink(href) {
6843
+
this.linkRef++;
6844
+
this.pendingLink = href;
6845
+
this.resetReloadStatus();
6846
+
return this.linkRef;
6847
+
}
6848
+
// anytime we are navigating or connecting, drop reload cookie in case
6849
+
// we issue the cookie but the next request was interrupted and the server never dropped it
6850
+
resetReloadStatus() {
6851
+
browser_default.deleteCookie(PHX_RELOAD_STATUS);
6852
+
}
6853
+
commitPendingLink(linkRef) {
6854
+
if (this.linkRef !== linkRef) {
6855
+
return false;
6856
+
} else {
6857
+
this.href = this.pendingLink;
6858
+
this.pendingLink = null;
6859
+
return true;
6860
+
}
6861
+
}
6862
+
getHref() {
6863
+
return this.href;
6864
+
}
6865
+
hasPendingLink() {
6866
+
return !!this.pendingLink;
6867
+
}
6868
+
bind(events, callback) {
6869
+
for (let event in events) {
6870
+
let browserEventName = events[event];
6871
+
this.on(browserEventName, (e) => {
6872
+
let binding = this.binding(event);
6873
+
let windowBinding = this.binding(`window-${event}`);
6874
+
let targetPhxEvent = e.target.getAttribute && e.target.getAttribute(binding);
6875
+
if (targetPhxEvent) {
6876
+
this.debounce(e.target, e, browserEventName, () => {
6877
+
this.withinOwners(e.target, (view) => {
6878
+
callback(e, event, view, e.target, targetPhxEvent, null);
6879
+
});
6880
+
});
6881
+
} else {
6882
+
dom_default.all(document, `[${windowBinding}]`, (el) => {
6883
+
let phxEvent = el.getAttribute(windowBinding);
6884
+
this.debounce(el, e, browserEventName, () => {
6885
+
this.withinOwners(el, (view) => {
6886
+
callback(e, event, view, el, phxEvent, "window");
6887
+
});
6888
+
});
6889
+
});
6890
+
}
6891
+
});
6892
+
}
6893
+
}
6894
+
bindClicks() {
6895
+
this.on("mousedown", (e) => this.clickStartedAtTarget = e.target);
6896
+
this.bindClick("click", "click");
6897
+
}
6898
+
bindClick(eventName, bindingName) {
6899
+
let click = this.binding(bindingName);
6900
+
window.addEventListener(eventName, (e) => {
6901
+
let target = null;
6902
+
if (e.detail === 0)
6903
+
this.clickStartedAtTarget = e.target;
6904
+
let clickStartedAtTarget = this.clickStartedAtTarget || e.target;
6905
+
target = closestPhxBinding(e.target, click);
6906
+
this.dispatchClickAway(e, clickStartedAtTarget);
6907
+
this.clickStartedAtTarget = null;
6908
+
let phxEvent = target && target.getAttribute(click);
6909
+
if (!phxEvent) {
6910
+
if (dom_default.isNewPageClick(e, window.location)) {
6911
+
this.unload();
6912
+
}
6913
+
return;
6914
+
}
6915
+
if (target.getAttribute("href") === "#") {
6916
+
e.preventDefault();
6917
+
}
6918
+
if (target.hasAttribute(PHX_REF_SRC)) {
6919
+
return;
6920
+
}
6921
+
this.debounce(target, e, "click", () => {
6922
+
this.withinOwners(target, (view) => {
6923
+
js_default.exec(e, "click", phxEvent, view, target, ["push", { data: this.eventMeta("click", e, target) }]);
6924
+
});
6925
+
});
6926
+
}, false);
6927
+
}
6928
+
dispatchClickAway(e, clickStartedAt) {
6929
+
let phxClickAway = this.binding("click-away");
6930
+
dom_default.all(document, `[${phxClickAway}]`, (el) => {
6931
+
if (!(el.isSameNode(clickStartedAt) || el.contains(clickStartedAt))) {
6932
+
this.withinOwners(el, (view) => {
6933
+
let phxEvent = el.getAttribute(phxClickAway);
6934
+
if (js_default.isVisible(el) && js_default.isInViewport(el)) {
6935
+
js_default.exec(e, "click", phxEvent, view, el, ["push", { data: this.eventMeta("click", e, e.target) }]);
6936
+
}
6937
+
});
6938
+
}
6939
+
});
6940
+
}
6941
+
bindNav() {
6942
+
if (!browser_default.canPushState()) {
6943
+
return;
6944
+
}
6945
+
if (history.scrollRestoration) {
6946
+
history.scrollRestoration = "manual";
6947
+
}
6948
+
let scrollTimer = null;
6949
+
window.addEventListener("scroll", (_e) => {
6950
+
clearTimeout(scrollTimer);
6951
+
scrollTimer = setTimeout(() => {
6952
+
browser_default.updateCurrentState((state) => Object.assign(state, { scroll: window.scrollY }));
6953
+
}, 100);
6954
+
});
6955
+
window.addEventListener("popstate", (event) => {
6956
+
if (!this.registerNewLocation(window.location)) {
6957
+
return;
6958
+
}
6959
+
let { type, backType, id, scroll, position } = event.state || {};
6960
+
let href = window.location.href;
6961
+
let isForward = position > this.currentHistoryPosition;
6962
+
type = isForward ? type : backType || type;
6963
+
this.currentHistoryPosition = position || 0;
6964
+
this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString());
6965
+
dom_default.dispatchEvent(window, "phx:navigate", { detail: { href, patch: type === "patch", pop: true, direction: isForward ? "forward" : "backward" } });
6966
+
this.requestDOMUpdate(() => {
6967
+
const callback = () => {
6968
+
this.maybeScroll(scroll);
6969
+
};
6970
+
if (this.main.isConnected() && (type === "patch" && id === this.main.id)) {
6971
+
this.main.pushLinkPatch(event, href, null, callback);
6972
+
} else {
6973
+
this.replaceMain(href, null, callback);
6974
+
}
6975
+
});
6976
+
}, false);
6977
+
window.addEventListener("click", (e) => {
6978
+
let target = closestPhxBinding(e.target, PHX_LIVE_LINK);
6979
+
let type = target && target.getAttribute(PHX_LIVE_LINK);
6980
+
if (!type || !this.isConnected() || !this.main || dom_default.wantsNewTab(e)) {
6981
+
return;
6982
+
}
6983
+
let href = target.href instanceof SVGAnimatedString ? target.href.baseVal : target.href;
6984
+
let linkState = target.getAttribute(PHX_LINK_STATE);
6985
+
e.preventDefault();
6986
+
e.stopImmediatePropagation();
6987
+
if (this.pendingLink === href) {
6988
+
return;
6989
+
}
6990
+
this.requestDOMUpdate(() => {
6991
+
if (type === "patch") {
6992
+
this.pushHistoryPatch(e, href, linkState, target);
6993
+
} else if (type === "redirect") {
6994
+
this.historyRedirect(e, href, linkState, null, target);
6995
+
} else {
6996
+
throw new Error(`expected ${PHX_LIVE_LINK} to be "patch" or "redirect", got: ${type}`);
6997
+
}
6998
+
let phxClick = target.getAttribute(this.binding("click"));
6999
+
if (phxClick) {
7000
+
this.requestDOMUpdate(() => this.execJS(target, phxClick, "click"));
7001
+
}
7002
+
});
7003
+
}, false);
7004
+
}
7005
+
maybeScroll(scroll) {
7006
+
if (typeof scroll === "number") {
7007
+
requestAnimationFrame(() => {
7008
+
window.scrollTo(0, scroll);
7009
+
});
7010
+
}
7011
+
}
7012
+
dispatchEvent(event, payload = {}) {
7013
+
dom_default.dispatchEvent(window, `phx:${event}`, { detail: payload });
7014
+
}
7015
+
dispatchEvents(events) {
7016
+
events.forEach(([event, payload]) => this.dispatchEvent(event, payload));
7017
+
}
7018
+
withPageLoading(info, callback) {
7019
+
dom_default.dispatchEvent(window, "phx:page-loading-start", { detail: info });
7020
+
let done = () => dom_default.dispatchEvent(window, "phx:page-loading-stop", { detail: info });
7021
+
return callback ? callback(done) : done;
7022
+
}
7023
+
pushHistoryPatch(e, href, linkState, targetEl) {
7024
+
if (!this.isConnected() || !this.main.isMain()) {
7025
+
return browser_default.redirect(href);
7026
+
}
7027
+
this.withPageLoading({ to: href, kind: "patch" }, (done) => {
7028
+
this.main.pushLinkPatch(e, href, targetEl, (linkRef) => {
7029
+
this.historyPatch(href, linkState, linkRef);
7030
+
done();
7031
+
});
7032
+
});
7033
+
}
7034
+
historyPatch(href, linkState, linkRef = this.setPendingLink(href)) {
7035
+
if (!this.commitPendingLink(linkRef)) {
7036
+
return;
7037
+
}
7038
+
this.currentHistoryPosition++;
7039
+
this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString());
7040
+
browser_default.updateCurrentState((state) => __spreadProps(__spreadValues({}, state), { backType: "patch" }));
7041
+
browser_default.pushState(linkState, {
7042
+
type: "patch",
7043
+
id: this.main.id,
7044
+
position: this.currentHistoryPosition
7045
+
}, href);
7046
+
dom_default.dispatchEvent(window, "phx:navigate", { detail: { patch: true, href, pop: false, direction: "forward" } });
7047
+
this.registerNewLocation(window.location);
7048
+
}
7049
+
historyRedirect(e, href, linkState, flash, targetEl) {
7050
+
const clickLoading = targetEl && e.isTrusted && e.type !== "popstate";
7051
+
if (clickLoading) {
7052
+
targetEl.classList.add("phx-click-loading");
7053
+
}
7054
+
if (!this.isConnected() || !this.main.isMain()) {
7055
+
return browser_default.redirect(href, flash);
7056
+
}
7057
+
if (/^\/$|^\/[^\/]+.*$/.test(href)) {
7058
+
let { protocol, host } = window.location;
7059
+
href = `${protocol}//${host}${href}`;
7060
+
}
7061
+
let scroll = window.scrollY;
7062
+
this.withPageLoading({ to: href, kind: "redirect" }, (done) => {
7063
+
this.replaceMain(href, flash, (linkRef) => {
7064
+
if (linkRef === this.linkRef) {
7065
+
this.currentHistoryPosition++;
7066
+
this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString());
7067
+
browser_default.updateCurrentState((state) => __spreadProps(__spreadValues({}, state), { backType: "redirect" }));
7068
+
browser_default.pushState(linkState, {
7069
+
type: "redirect",
7070
+
id: this.main.id,
7071
+
scroll,
7072
+
position: this.currentHistoryPosition
7073
+
}, href);
7074
+
dom_default.dispatchEvent(window, "phx:navigate", { detail: { href, patch: false, pop: false, direction: "forward" } });
7075
+
this.registerNewLocation(window.location);
7076
+
}
7077
+
if (clickLoading) {
7078
+
targetEl.classList.remove("phx-click-loading");
7079
+
}
7080
+
done();
7081
+
});
7082
+
});
7083
+
}
7084
+
registerNewLocation(newLocation) {
7085
+
let { pathname, search } = this.currentLocation;
7086
+
if (pathname + search === newLocation.pathname + newLocation.search) {
7087
+
return false;
7088
+
} else {
7089
+
this.currentLocation = clone(newLocation);
7090
+
return true;
7091
+
}
7092
+
}
7093
+
bindForms() {
7094
+
let iterations = 0;
7095
+
let externalFormSubmitted = false;
7096
+
this.on("submit", (e) => {
7097
+
let phxSubmit = e.target.getAttribute(this.binding("submit"));
7098
+
let phxChange = e.target.getAttribute(this.binding("change"));
7099
+
if (!externalFormSubmitted && phxChange && !phxSubmit) {
7100
+
externalFormSubmitted = true;
7101
+
e.preventDefault();
7102
+
this.withinOwners(e.target, (view) => {
7103
+
view.disableForm(e.target);
7104
+
window.requestAnimationFrame(() => {
7105
+
if (dom_default.isUnloadableFormSubmit(e)) {
7106
+
this.unload();
7107
+
}
7108
+
e.target.submit();
7109
+
});
7110
+
});
7111
+
}
7112
+
});
7113
+
this.on("submit", (e) => {
7114
+
let phxEvent = e.target.getAttribute(this.binding("submit"));
7115
+
if (!phxEvent) {
7116
+
if (dom_default.isUnloadableFormSubmit(e)) {
7117
+
this.unload();
7118
+
}
7119
+
return;
7120
+
}
7121
+
e.preventDefault();
7122
+
e.target.disabled = true;
7123
+
this.withinOwners(e.target, (view) => {
7124
+
js_default.exec(e, "submit", phxEvent, view, e.target, ["push", { submitter: e.submitter }]);
7125
+
});
7126
+
});
7127
+
for (let type of ["change", "input"]) {
7128
+
this.on(type, (e) => {
7129
+
if (e instanceof CustomEvent && e.target.form === void 0) {
7130
+
if (e.detail && e.detail.dispatcher) {
7131
+
throw new Error(`dispatching a custom ${type} event is only supported on input elements inside a form`);
7132
+
}
7133
+
return;
7134
+
}
7135
+
let phxChange = this.binding("change");
7136
+
let input = e.target;
7137
+
if (e.isComposing) {
7138
+
const key = `composition-listener-${type}`;
7139
+
if (!dom_default.private(input, key)) {
7140
+
dom_default.putPrivate(input, key, true);
7141
+
input.addEventListener("compositionend", () => {
7142
+
input.dispatchEvent(new Event(type, { bubbles: true }));
7143
+
dom_default.deletePrivate(input, key);
7144
+
}, { once: true });
7145
+
}
7146
+
return;
7147
+
}
7148
+
let inputEvent = input.getAttribute(phxChange);
7149
+
let formEvent = input.form && input.form.getAttribute(phxChange);
7150
+
let phxEvent = inputEvent || formEvent;
7151
+
if (!phxEvent) {
7152
+
return;
7153
+
}
7154
+
if (input.type === "number" && input.validity && input.validity.badInput) {
7155
+
return;
7156
+
}
7157
+
let dispatcher = inputEvent ? input : input.form;
7158
+
let currentIterations = iterations;
7159
+
iterations++;
7160
+
let { at, type: lastType } = dom_default.private(input, "prev-iteration") || {};
7161
+
if (at === currentIterations - 1 && type === "change" && lastType === "input") {
7162
+
return;
7163
+
}
7164
+
dom_default.putPrivate(input, "prev-iteration", { at: currentIterations, type });
7165
+
this.debounce(input, e, type, () => {
7166
+
this.withinOwners(dispatcher, (view) => {
7167
+
dom_default.putPrivate(input, PHX_HAS_FOCUSED, true);
7168
+
js_default.exec(e, "change", phxEvent, view, input, ["push", { _target: e.target.name, dispatcher }]);
7169
+
});
7170
+
});
7171
+
});
7172
+
}
7173
+
this.on("reset", (e) => {
7174
+
let form = e.target;
7175
+
dom_default.resetForm(form);
7176
+
let input = Array.from(form.elements).find((el) => el.type === "reset");
7177
+
if (input) {
7178
+
window.requestAnimationFrame(() => {
7179
+
input.dispatchEvent(new Event("input", { bubbles: true, cancelable: false }));
7180
+
});
7181
+
}
7182
+
});
7183
+
}
7184
+
debounce(el, event, eventType, callback) {
7185
+
if (eventType === "blur" || eventType === "focusout") {
7186
+
return callback();
7187
+
}
7188
+
let phxDebounce = this.binding(PHX_DEBOUNCE);
7189
+
let phxThrottle = this.binding(PHX_THROTTLE);
7190
+
let defaultDebounce = this.defaults.debounce.toString();
7191
+
let defaultThrottle = this.defaults.throttle.toString();
7192
+
this.withinOwners(el, (view) => {
7193
+
let asyncFilter = () => !view.isDestroyed() && document.body.contains(el);
7194
+
dom_default.debounce(el, event, phxDebounce, defaultDebounce, phxThrottle, defaultThrottle, asyncFilter, () => {
7195
+
callback();
7196
+
});
7197
+
});
7198
+
}
7199
+
silenceEvents(callback) {
7200
+
this.silenced = true;
7201
+
callback();
7202
+
this.silenced = false;
7203
+
}
7204
+
on(event, callback) {
7205
+
this.boundEventNames.add(event);
7206
+
window.addEventListener(event, (e) => {
7207
+
if (!this.silenced) {
7208
+
callback(e);
7209
+
}
7210
+
});
7211
+
}
7212
+
jsQuerySelectorAll(sourceEl, query, defaultQuery) {
7213
+
let all = this.domCallbacks.jsQuerySelectorAll;
7214
+
return all ? all(sourceEl, query, defaultQuery) : defaultQuery();
7215
+
}
7216
+
};
7217
+
var TransitionSet = class {
7218
+
constructor() {
7219
+
this.transitions = /* @__PURE__ */ new Set();
7220
+
this.pendingOps = [];
7221
+
}
7222
+
reset() {
7223
+
this.transitions.forEach((timer) => {
7224
+
clearTimeout(timer);
7225
+
this.transitions.delete(timer);
7226
+
});
7227
+
this.flushPendingOps();
7228
+
}
7229
+
after(callback) {
7230
+
if (this.size() === 0) {
7231
+
callback();
7232
+
} else {
7233
+
this.pushPendingOp(callback);
7234
+
}
7235
+
}
7236
+
addTransition(time, onStart, onDone) {
7237
+
onStart();
7238
+
let timer = setTimeout(() => {
7239
+
this.transitions.delete(timer);
7240
+
onDone();
7241
+
this.flushPendingOps();
7242
+
}, time);
7243
+
this.transitions.add(timer);
7244
+
}
7245
+
pushPendingOp(op) {
7246
+
this.pendingOps.push(op);
7247
+
}
7248
+
size() {
7249
+
return this.transitions.size;
7250
+
}
7251
+
flushPendingOps() {
7252
+
if (this.size() > 0) {
7253
+
return;
7254
+
}
7255
+
let op = this.pendingOps.shift();
7256
+
if (op) {
7257
+
op();
7258
+
this.flushPendingOps();
7259
+
}
7260
+
}
7261
+
};
7262
+
7263
+
// js/app.js
7264
+
var import_topbar = __toESM(require_topbar());
7265
+
var csrfToken = document.querySelector("meta[name='csrf-token']").getAttribute("content");
7266
+
var liveSocket = new LiveSocket("/live", Socket, {
7267
+
longPollFallbackMs: 2500,
7268
+
params: { _csrf_token: csrfToken }
7269
+
});
7270
+
import_topbar.default.config({ barColors: { 0: "#29d" }, shadowColor: "rgba(0, 0, 0, .3)" });
7271
+
window.addEventListener("phx:page-loading-start", (_info) => import_topbar.default.show(300));
7272
+
window.addEventListener("phx:page-loading-stop", (_info) => import_topbar.default.hide());
7273
+
liveSocket.connect();
7274
+
window.liveSocket = liveSocket;
7275
+
})();
7276
+
/**
7277
+
* @license MIT
7278
+
* topbar 2.0.0, 2023-02-04
7279
+
* https://buunguyen.github.io/topbar
7280
+
* Copyright (c) 2021 Buu Nguyen
7281
+
*/
7282
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../assets/vendor/topbar.js", "../../../deps/phoenix_html/priv/static/phoenix_html.js", "../../../deps/phoenix/assets/js/phoenix/utils.js", "../../../deps/phoenix/assets/js/phoenix/constants.js", "../../../deps/phoenix/assets/js/phoenix/push.js", "../../../deps/phoenix/assets/js/phoenix/timer.js", "../../../deps/phoenix/assets/js/phoenix/channel.js", "../../../deps/phoenix/assets/js/phoenix/ajax.js", "../../../deps/phoenix/assets/js/phoenix/longpoll.js", "../../../deps/phoenix/assets/js/phoenix/presence.js", "../../../deps/phoenix/assets/js/phoenix/serializer.js", "../../../deps/phoenix/assets/js/phoenix/socket.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/constants.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/entry_uploader.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/utils.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/browser.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/dom.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/upload_entry.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/live_uploader.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/aria.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/hooks.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/element_ref.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/dom_post_morph_restorer.js", "../../../deps/phoenix_live_view/node_modules/morphdom/dist/morphdom-esm.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/dom_patch.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/rendered.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/js.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/view_hook.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/view.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/live_socket.js", "../../../deps/phoenix_live_view/assets/js/phoenix_live_view/index.js", "../../../assets/js/app.js"],
  "sourcesContent": ["/**\n * @license MIT\n * topbar 2.0.0, 2023-02-04\n * https://buunguyen.github.io/topbar\n * Copyright (c) 2021 Buu Nguyen\n */\n(function (window, document) {\n  \"use strict\";\n\n  // https://gist.github.com/paulirish/1579671\n  (function () {\n    var lastTime = 0;\n    var vendors = [\"ms\", \"moz\", \"webkit\", \"o\"];\n    for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {\n      window.requestAnimationFrame =\n        window[vendors[x] + \"RequestAnimationFrame\"];\n      window.cancelAnimationFrame =\n        window[vendors[x] + \"CancelAnimationFrame\"] ||\n        window[vendors[x] + \"CancelRequestAnimationFrame\"];\n    }\n    if (!window.requestAnimationFrame)\n      window.requestAnimationFrame = function (callback, element) {\n        var currTime = new Date().getTime();\n        var timeToCall = Math.max(0, 16 - (currTime - lastTime));\n        var id = window.setTimeout(function () {\n          callback(currTime + timeToCall);\n        }, timeToCall);\n        lastTime = currTime + timeToCall;\n        return id;\n      };\n    if (!window.cancelAnimationFrame)\n      window.cancelAnimationFrame = function (id) {\n        clearTimeout(id);\n      };\n  })();\n\n  var canvas,\n    currentProgress,\n    showing,\n    progressTimerId = null,\n    fadeTimerId = null,\n    delayTimerId = null,\n    addEvent = function (elem, type, handler) {\n      if (elem.addEventListener) elem.addEventListener(type, handler, false);\n      else if (elem.attachEvent) elem.attachEvent(\"on\" + type, handler);\n      else elem[\"on\" + type] = handler;\n    },\n    options = {\n      autoRun: true,\n      barThickness: 3,\n      barColors: {\n        0: \"rgba(26,  188, 156, .9)\",\n        \".25\": \"rgba(52,  152, 219, .9)\",\n        \".50\": \"rgba(241, 196, 15,  .9)\",\n        \".75\": \"rgba(230, 126, 34,  .9)\",\n        \"1.0\": \"rgba(211, 84,  0,   .9)\",\n      },\n      shadowBlur: 10,\n      shadowColor: \"rgba(0,   0,   0,   .6)\",\n      className: null,\n    },\n    repaint = function () {\n      canvas.width = window.innerWidth;\n      canvas.height = options.barThickness * 5; // need space for shadow\n\n      var ctx = canvas.getContext(\"2d\");\n      ctx.shadowBlur = options.shadowBlur;\n      ctx.shadowColor = options.shadowColor;\n\n      var lineGradient = ctx.createLinearGradient(0, 0, canvas.width, 0);\n      for (var stop in options.barColors)\n        lineGradient.addColorStop(stop, options.barColors[stop]);\n      ctx.lineWidth = options.barThickness;\n      ctx.beginPath();\n      ctx.moveTo(0, options.barThickness / 2);\n      ctx.lineTo(\n        Math.ceil(currentProgress * canvas.width),\n        options.barThickness / 2\n      );\n      ctx.strokeStyle = lineGradient;\n      ctx.stroke();\n    },\n    createCanvas = function () {\n      canvas = document.createElement(\"canvas\");\n      var style = canvas.style;\n      style.position = \"fixed\";\n      style.top = style.left = style.right = style.margin = style.padding = 0;\n      style.zIndex = 100001;\n      style.display = \"none\";\n      if (options.className) canvas.classList.add(options.className);\n      document.body.appendChild(canvas);\n      addEvent(window, \"resize\", repaint);\n    },\n    topbar = {\n      config: function (opts) {\n        for (var key in opts)\n          if (options.hasOwnProperty(key)) options[key] = opts[key];\n      },\n      show: function (delay) {\n        if (showing) return;\n        if (delay) {\n          if (delayTimerId) return;\n          delayTimerId = setTimeout(() => topbar.show(), delay);\n        } else  {\n          showing = true;\n          if (fadeTimerId !== null) window.cancelAnimationFrame(fadeTimerId);\n          if (!canvas) createCanvas();\n          canvas.style.opacity = 1;\n          canvas.style.display = \"block\";\n          topbar.progress(0);\n          if (options.autoRun) {\n            (function loop() {\n              progressTimerId = window.requestAnimationFrame(loop);\n              topbar.progress(\n                \"+\" + 0.05 * Math.pow(1 - Math.sqrt(currentProgress), 2)\n              );\n            })();\n          }\n        }\n      },\n      progress: function (to) {\n        if (typeof to === \"undefined\") return currentProgress;\n        if (typeof to === \"string\") {\n          to =\n            (to.indexOf(\"+\") >= 0 || to.indexOf(\"-\") >= 0\n              ? currentProgress\n              : 0) + parseFloat(to);\n        }\n        currentProgress = to > 1 ? 1 : to;\n        repaint();\n        return currentProgress;\n      },\n      hide: function () {\n        clearTimeout(delayTimerId);\n        delayTimerId = null;\n        if (!showing) return;\n        showing = false;\n        if (progressTimerId != null) {\n          window.cancelAnimationFrame(progressTimerId);\n          progressTimerId = null;\n        }\n        (function loop() {\n          if (topbar.progress(\"+.1\") >= 1) {\n            canvas.style.opacity -= 0.05;\n            if (canvas.style.opacity <= 0.05) {\n              canvas.style.display = \"none\";\n              fadeTimerId = null;\n              return;\n            }\n          }\n          fadeTimerId = window.requestAnimationFrame(loop);\n        })();\n      },\n    };\n\n  if (typeof module === \"object\" && typeof module.exports === \"object\") {\n    module.exports = topbar;\n  } else if (typeof define === \"function\" && define.amd) {\n    define(function () {\n      return topbar;\n    });\n  } else {\n    this.topbar = topbar;\n  }\n}.call(this, window, document));\n", "\"use strict\";\n\n(function() {\n  var PolyfillEvent = eventConstructor();\n\n  function eventConstructor() {\n    if (typeof window.CustomEvent === \"function\") return window.CustomEvent;\n    // IE<=9 Support\n    function CustomEvent(event, params) {\n      params = params || {bubbles: false, cancelable: false, detail: undefined};\n      var evt = document.createEvent('CustomEvent');\n      evt.initCustomEvent(event, params.bubbles, params.cancelable, params.detail);\n      return evt;\n    }\n    CustomEvent.prototype = window.Event.prototype;\n    return CustomEvent;\n  }\n\n  function buildHiddenInput(name, value) {\n    var input = document.createElement(\"input\");\n    input.type = \"hidden\";\n    input.name = name;\n    input.value = value;\n    return input;\n  }\n\n  function handleClick(element, targetModifierKey) {\n    var to = element.getAttribute(\"data-to\"),\n        method = buildHiddenInput(\"_method\", element.getAttribute(\"data-method\")),\n        csrf = buildHiddenInput(\"_csrf_token\", element.getAttribute(\"data-csrf\")),\n        form = document.createElement(\"form\"),\n        submit = document.createElement(\"input\"),\n        target = element.getAttribute(\"target\");\n\n    form.method = (element.getAttribute(\"data-method\") === \"get\") ? \"get\" : \"post\";\n    form.action = to;\n    form.style.display = \"none\";\n\n    if (target) form.target = target;\n    else if (targetModifierKey) form.target = \"_blank\";\n\n    form.appendChild(csrf);\n    form.appendChild(method);\n    document.body.appendChild(form);\n\n    // Insert a button and click it instead of using `form.submit`\n    // because the `submit` function does not emit a `submit` event.\n    submit.type = \"submit\";\n    form.appendChild(submit);\n    submit.click();\n  }\n\n  window.addEventListener(\"click\", function(e) {\n    var element = e.target;\n    if (e.defaultPrevented) return;\n\n    while (element && element.getAttribute) {\n      var phoenixLinkEvent = new PolyfillEvent('phoenix.link.click', {\n        \"bubbles\": true, \"cancelable\": true\n      });\n\n      if (!element.dispatchEvent(phoenixLinkEvent)) {\n        e.preventDefault();\n        e.stopImmediatePropagation();\n        return false;\n      }\n\n      if (element.getAttribute(\"data-method\") && element.getAttribute(\"data-to\")) {\n        handleClick(element, e.metaKey || e.shiftKey);\n        e.preventDefault();\n        return false;\n      } else {\n        element = element.parentNode;\n      }\n    }\n  }, false);\n\n  window.addEventListener('phoenix.link.click', function (e) {\n    var message = e.target.getAttribute(\"data-confirm\");\n    if(message && !window.confirm(message)) {\n      e.preventDefault();\n    }\n  }, false);\n})();\n", "// wraps value in closure or returns closure\nexport let closure = (value) => {\n  if(typeof value === \"function\"){\n    return value\n  } else {\n    let closure = function (){ return value }\n    return closure\n  }\n}\n", "export const globalSelf = typeof self !== \"undefined\" ? self : null\nexport const phxWindow = typeof window !== \"undefined\" ? window : null\nexport const global = globalSelf || phxWindow || global\nexport const DEFAULT_VSN = \"2.0.0\"\nexport const SOCKET_STATES = {connecting: 0, open: 1, closing: 2, closed: 3}\nexport const DEFAULT_TIMEOUT = 10000\nexport const WS_CLOSE_NORMAL = 1000\nexport const CHANNEL_STATES = {\n  closed: \"closed\",\n  errored: \"errored\",\n  joined: \"joined\",\n  joining: \"joining\",\n  leaving: \"leaving\",\n}\nexport const CHANNEL_EVENTS = {\n  close: \"phx_close\",\n  error: \"phx_error\",\n  join: \"phx_join\",\n  reply: \"phx_reply\",\n  leave: \"phx_leave\"\n}\n\nexport const TRANSPORTS = {\n  longpoll: \"longpoll\",\n  websocket: \"websocket\"\n}\nexport const XHR_STATES = {\n  complete: 4\n}\n", "/**\n * Initializes the Push\n * @param {Channel} channel - The Channel\n * @param {string} event - The event, for example `\"phx_join\"`\n * @param {Object} payload - The payload, for example `{user_id: 123}`\n * @param {number} timeout - The push timeout in milliseconds\n */\nexport default class Push {\n  constructor(channel, event, payload, timeout){\n    this.channel = channel\n    this.event = event\n    this.payload = payload || function (){ return {} }\n    this.receivedResp = null\n    this.timeout = timeout\n    this.timeoutTimer = null\n    this.recHooks = []\n    this.sent = false\n  }\n\n  /**\n   *\n   * @param {number} timeout\n   */\n  resend(timeout){\n    this.timeout = timeout\n    this.reset()\n    this.send()\n  }\n\n  /**\n   *\n   */\n  send(){\n    if(this.hasReceived(\"timeout\")){ return }\n    this.startTimeout()\n    this.sent = true\n    this.channel.socket.push({\n      topic: this.channel.topic,\n      event: this.event,\n      payload: this.payload(),\n      ref: this.ref,\n      join_ref: this.channel.joinRef()\n    })\n  }\n\n  /**\n   *\n   * @param {*} status\n   * @param {*} callback\n   */\n  receive(status, callback){\n    if(this.hasReceived(status)){\n      callback(this.receivedResp.response)\n    }\n\n    this.recHooks.push({status, callback})\n    return this\n  }\n\n  /**\n   * @private\n   */\n  reset(){\n    this.cancelRefEvent()\n    this.ref = null\n    this.refEvent = null\n    this.receivedResp = null\n    this.sent = false\n  }\n\n  /**\n   * @private\n   */\n  matchReceive({status, response, _ref}){\n    this.recHooks.filter(h => h.status === status)\n      .forEach(h => h.callback(response))\n  }\n\n  /**\n   * @private\n   */\n  cancelRefEvent(){\n    if(!this.refEvent){ return }\n    this.channel.off(this.refEvent)\n  }\n\n  /**\n   * @private\n   */\n  cancelTimeout(){\n    clearTimeout(this.timeoutTimer)\n    this.timeoutTimer = null\n  }\n\n  /**\n   * @private\n   */\n  startTimeout(){\n    if(this.timeoutTimer){ this.cancelTimeout() }\n    this.ref = this.channel.socket.makeRef()\n    this.refEvent = this.channel.replyEventName(this.ref)\n\n    this.channel.on(this.refEvent, payload => {\n      this.cancelRefEvent()\n      this.cancelTimeout()\n      this.receivedResp = payload\n      this.matchReceive(payload)\n    })\n\n    this.timeoutTimer = setTimeout(() => {\n      this.trigger(\"timeout\", {})\n    }, this.timeout)\n  }\n\n  /**\n   * @private\n   */\n  hasReceived(status){\n    return this.receivedResp && this.receivedResp.status === status\n  }\n\n  /**\n   * @private\n   */\n  trigger(status, response){\n    this.channel.trigger(this.refEvent, {status, response})\n  }\n}\n", "/**\n *\n * Creates a timer that accepts a `timerCalc` function to perform\n * calculated timeout retries, such as exponential backoff.\n *\n * @example\n * let reconnectTimer = new Timer(() => this.connect(), function(tries){\n *   return [1000, 5000, 10000][tries - 1] || 10000\n * })\n * reconnectTimer.scheduleTimeout() // fires after 1000\n * reconnectTimer.scheduleTimeout() // fires after 5000\n * reconnectTimer.reset()\n * reconnectTimer.scheduleTimeout() // fires after 1000\n *\n * @param {Function} callback\n * @param {Function} timerCalc\n */\nexport default class Timer {\n  constructor(callback, timerCalc){\n    this.callback = callback\n    this.timerCalc = timerCalc\n    this.timer = null\n    this.tries = 0\n  }\n\n  reset(){\n    this.tries = 0\n    clearTimeout(this.timer)\n  }\n\n  /**\n   * Cancels any previous scheduleTimeout and schedules callback\n   */\n  scheduleTimeout(){\n    clearTimeout(this.timer)\n\n    this.timer = setTimeout(() => {\n      this.tries = this.tries + 1\n      this.callback()\n    }, this.timerCalc(this.tries + 1))\n  }\n}\n", "import {closure} from \"./utils\"\nimport {\n  CHANNEL_EVENTS,\n  CHANNEL_STATES,\n} from \"./constants\"\n\nimport Push from \"./push\"\nimport Timer from \"./timer\"\n\n/**\n *\n * @param {string} topic\n * @param {(Object|function)} params\n * @param {Socket} socket\n */\nexport default class Channel {\n  constructor(topic, params, socket){\n    this.state = CHANNEL_STATES.closed\n    this.topic = topic\n    this.params = closure(params || {})\n    this.socket = socket\n    this.bindings = []\n    this.bindingRef = 0\n    this.timeout = this.socket.timeout\n    this.joinedOnce = false\n    this.joinPush = new Push(this, CHANNEL_EVENTS.join, this.params, this.timeout)\n    this.pushBuffer = []\n    this.stateChangeRefs = []\n\n    this.rejoinTimer = new Timer(() => {\n      if(this.socket.isConnected()){ this.rejoin() }\n    }, this.socket.rejoinAfterMs)\n    this.stateChangeRefs.push(this.socket.onError(() => this.rejoinTimer.reset()))\n    this.stateChangeRefs.push(this.socket.onOpen(() => {\n      this.rejoinTimer.reset()\n      if(this.isErrored()){ this.rejoin() }\n    })\n    )\n    this.joinPush.receive(\"ok\", () => {\n      this.state = CHANNEL_STATES.joined\n      this.rejoinTimer.reset()\n      this.pushBuffer.forEach(pushEvent => pushEvent.send())\n      this.pushBuffer = []\n    })\n    this.joinPush.receive(\"error\", () => {\n      this.state = CHANNEL_STATES.errored\n      if(this.socket.isConnected()){ this.rejoinTimer.scheduleTimeout() }\n    })\n    this.onClose(() => {\n      this.rejoinTimer.reset()\n      if(this.socket.hasLogger()) this.socket.log(\"channel\", `close ${this.topic} ${this.joinRef()}`)\n      this.state = CHANNEL_STATES.closed\n      this.socket.remove(this)\n    })\n    this.onError(reason => {\n      if(this.socket.hasLogger()) this.socket.log(\"channel\", `error ${this.topic}`, reason)\n      if(this.isJoining()){ this.joinPush.reset() }\n      this.state = CHANNEL_STATES.errored\n      if(this.socket.isConnected()){ this.rejoinTimer.scheduleTimeout() }\n    })\n    this.joinPush.receive(\"timeout\", () => {\n      if(this.socket.hasLogger()) this.socket.log(\"channel\", `timeout ${this.topic} (${this.joinRef()})`, this.joinPush.timeout)\n      let leavePush = new Push(this, CHANNEL_EVENTS.leave, closure({}), this.timeout)\n      leavePush.send()\n      this.state = CHANNEL_STATES.errored\n      this.joinPush.reset()\n      if(this.socket.isConnected()){ this.rejoinTimer.scheduleTimeout() }\n    })\n    this.on(CHANNEL_EVENTS.reply, (payload, ref) => {\n      this.trigger(this.replyEventName(ref), payload)\n    })\n  }\n\n  /**\n   * Join the channel\n   * @param {integer} timeout\n   * @returns {Push}\n   */\n  join(timeout = this.timeout){\n    if(this.joinedOnce){\n      throw new Error(\"tried to join multiple times. 'join' can only be called a single time per channel instance\")\n    } else {\n      this.timeout = timeout\n      this.joinedOnce = true\n      this.rejoin()\n      return this.joinPush\n    }\n  }\n\n  /**\n   * Hook into channel close\n   * @param {Function} callback\n   */\n  onClose(callback){\n    this.on(CHANNEL_EVENTS.close, callback)\n  }\n\n  /**\n   * Hook into channel errors\n   * @param {Function} callback\n   */\n  onError(callback){\n    return this.on(CHANNEL_EVENTS.error, reason => callback(reason))\n  }\n\n  /**\n   * Subscribes on channel events\n   *\n   * Subscription returns a ref counter, which can be used later to\n   * unsubscribe the exact event listener\n   *\n   * @example\n   * const ref1 = channel.on(\"event\", do_stuff)\n   * const ref2 = channel.on(\"event\", do_other_stuff)\n   * channel.off(\"event\", ref1)\n   * // Since unsubscription, do_stuff won't fire,\n   * // while do_other_stuff will keep firing on the \"event\"\n   *\n   * @param {string} event\n   * @param {Function} callback\n   * @returns {integer} ref\n   */\n  on(event, callback){\n    let ref = this.bindingRef++\n    this.bindings.push({event, ref, callback})\n    return ref\n  }\n\n  /**\n   * Unsubscribes off of channel events\n   *\n   * Use the ref returned from a channel.on() to unsubscribe one\n   * handler, or pass nothing for the ref to unsubscribe all\n   * handlers for the given event.\n   *\n   * @example\n   * // Unsubscribe the do_stuff handler\n   * const ref1 = channel.on(\"event\", do_stuff)\n   * channel.off(\"event\", ref1)\n   *\n   * // Unsubscribe all handlers from event\n   * channel.off(\"event\")\n   *\n   * @param {string} event\n   * @param {integer} ref\n   */\n  off(event, ref){\n    this.bindings = this.bindings.filter((bind) => {\n      return !(bind.event === event && (typeof ref === \"undefined\" || ref === bind.ref))\n    })\n  }\n\n  /**\n   * @private\n   */\n  canPush(){ return this.socket.isConnected() && this.isJoined() }\n\n  /**\n   * Sends a message `event` to phoenix with the payload `payload`.\n   * Phoenix receives this in the `handle_in(event, payload, socket)`\n   * function. if phoenix replies or it times out (default 10000ms),\n   * then optionally the reply can be received.\n   *\n   * @example\n   * channel.push(\"event\")\n   *   .receive(\"ok\", payload => console.log(\"phoenix replied:\", payload))\n   *   .receive(\"error\", err => console.log(\"phoenix errored\", err))\n   *   .receive(\"timeout\", () => console.log(\"timed out pushing\"))\n   * @param {string} event\n   * @param {Object} payload\n   * @param {number} [timeout]\n   * @returns {Push}\n   */\n  push(event, payload, timeout = this.timeout){\n    payload = payload || {}\n    if(!this.joinedOnce){\n      throw new Error(`tried to push '${event}' to '${this.topic}' before joining. Use channel.join() before pushing events`)\n    }\n    let pushEvent = new Push(this, event, function (){ return payload }, timeout)\n    if(this.canPush()){\n      pushEvent.send()\n    } else {\n      pushEvent.startTimeout()\n      this.pushBuffer.push(pushEvent)\n    }\n\n    return pushEvent\n  }\n\n  /** Leaves the channel\n   *\n   * Unsubscribes from server events, and\n   * instructs channel to terminate on server\n   *\n   * Triggers onClose() hooks\n   *\n   * To receive leave acknowledgements, use the `receive`\n   * hook to bind to the server ack, ie:\n   *\n   * @example\n   * channel.leave().receive(\"ok\", () => alert(\"left!\") )\n   *\n   * @param {integer} timeout\n   * @returns {Push}\n   */\n  leave(timeout = this.timeout){\n    this.rejoinTimer.reset()\n    this.joinPush.cancelTimeout()\n\n    this.state = CHANNEL_STATES.leaving\n    let onClose = () => {\n      if(this.socket.hasLogger()) this.socket.log(\"channel\", `leave ${this.topic}`)\n      this.trigger(CHANNEL_EVENTS.close, \"leave\")\n    }\n    let leavePush = new Push(this, CHANNEL_EVENTS.leave, closure({}), timeout)\n    leavePush.receive(\"ok\", () => onClose())\n      .receive(\"timeout\", () => onClose())\n    leavePush.send()\n    if(!this.canPush()){ leavePush.trigger(\"ok\", {}) }\n\n    return leavePush\n  }\n\n  /**\n   * Overridable message hook\n   *\n   * Receives all events for specialized message handling\n   * before dispatching to the channel callbacks.\n   *\n   * Must return the payload, modified or unmodified\n   * @param {string} event\n   * @param {Object} payload\n   * @param {integer} ref\n   * @returns {Object}\n   */\n  onMessage(_event, payload, _ref){ return payload }\n\n  /**\n   * @private\n   */\n  isMember(topic, event, payload, joinRef){\n    if(this.topic !== topic){ return false }\n\n    if(joinRef && joinRef !== this.joinRef()){\n      if(this.socket.hasLogger()) this.socket.log(\"channel\", \"dropping outdated message\", {topic, event, payload, joinRef})\n      return false\n    } else {\n      return true\n    }\n  }\n\n  /**\n   * @private\n   */\n  joinRef(){ return this.joinPush.ref }\n\n  /**\n   * @private\n   */\n  rejoin(timeout = this.timeout){\n    if(this.isLeaving()){ return }\n    this.socket.leaveOpenTopic(this.topic)\n    this.state = CHANNEL_STATES.joining\n    this.joinPush.resend(timeout)\n  }\n\n  /**\n   * @private\n   */\n  trigger(event, payload, ref, joinRef){\n    let handledPayload = this.onMessage(event, payload, ref, joinRef)\n    if(payload && !handledPayload){ throw new Error(\"channel onMessage callbacks must return the payload, modified or unmodified\") }\n\n    let eventBindings = this.bindings.filter(bind => bind.event === event)\n\n    for(let i = 0; i < eventBindings.length; i++){\n      let bind = eventBindings[i]\n      bind.callback(handledPayload, ref, joinRef || this.joinRef())\n    }\n  }\n\n  /**\n   * @private\n   */\n  replyEventName(ref){ return `chan_reply_${ref}` }\n\n  /**\n   * @private\n   */\n  isClosed(){ return this.state === CHANNEL_STATES.closed }\n\n  /**\n   * @private\n   */\n  isErrored(){ return this.state === CHANNEL_STATES.errored }\n\n  /**\n   * @private\n   */\n  isJoined(){ return this.state === CHANNEL_STATES.joined }\n\n  /**\n   * @private\n   */\n  isJoining(){ return this.state === CHANNEL_STATES.joining }\n\n  /**\n   * @private\n   */\n  isLeaving(){ return this.state === CHANNEL_STATES.leaving }\n}\n", "import {\n  global,\n  XHR_STATES\n} from \"./constants\"\n\nexport default class Ajax {\n\n  static request(method, endPoint, accept, body, timeout, ontimeout, callback){\n    if(global.XDomainRequest){\n      let req = new global.XDomainRequest() // IE8, IE9\n      return this.xdomainRequest(req, method, endPoint, body, timeout, ontimeout, callback)\n    } else {\n      let req = new global.XMLHttpRequest() // IE7+, Firefox, Chrome, Opera, Safari\n      return this.xhrRequest(req, method, endPoint, accept, body, timeout, ontimeout, callback)\n    }\n  }\n\n  static xdomainRequest(req, method, endPoint, body, timeout, ontimeout, callback){\n    req.timeout = timeout\n    req.open(method, endPoint)\n    req.onload = () => {\n      let response = this.parseJSON(req.responseText)\n      callback && callback(response)\n    }\n    if(ontimeout){ req.ontimeout = ontimeout }\n\n    // Work around bug in IE9 that requires an attached onprogress handler\n    req.onprogress = () => { }\n\n    req.send(body)\n    return req\n  }\n\n  static xhrRequest(req, method, endPoint, accept, body, timeout, ontimeout, callback){\n    req.open(method, endPoint, true)\n    req.timeout = timeout\n    req.setRequestHeader(\"Content-Type\", accept)\n    req.onerror = () => callback && callback(null)\n    req.onreadystatechange = () => {\n      if(req.readyState === XHR_STATES.complete && callback){\n        let response = this.parseJSON(req.responseText)\n        callback(response)\n      }\n    }\n    if(ontimeout){ req.ontimeout = ontimeout }\n\n    req.send(body)\n    return req\n  }\n\n  static parseJSON(resp){\n    if(!resp || resp === \"\"){ return null }\n\n    try {\n      return JSON.parse(resp)\n    } catch (e){\n      console && console.log(\"failed to parse JSON response\", resp)\n      return null\n    }\n  }\n\n  static serialize(obj, parentKey){\n    let queryStr = []\n    for(var key in obj){\n      if(!Object.prototype.hasOwnProperty.call(obj, key)){ continue }\n      let paramKey = parentKey ? `${parentKey}[${key}]` : key\n      let paramVal = obj[key]\n      if(typeof paramVal === \"object\"){\n        queryStr.push(this.serialize(paramVal, paramKey))\n      } else {\n        queryStr.push(encodeURIComponent(paramKey) + \"=\" + encodeURIComponent(paramVal))\n      }\n    }\n    return queryStr.join(\"&\")\n  }\n\n  static appendParams(url, params){\n    if(Object.keys(params).length === 0){ return url }\n\n    let prefix = url.match(/\\?/) ? \"&\" : \"?\"\n    return `${url}${prefix}${this.serialize(params)}`\n  }\n}\n", "import {\n  SOCKET_STATES,\n  TRANSPORTS\n} from \"./constants\"\n\nimport Ajax from \"./ajax\"\n\nlet arrayBufferToBase64 = (buffer) => {\n  let binary = \"\"\n  let bytes = new Uint8Array(buffer)\n  let len = bytes.byteLength\n  for(let i = 0; i < len; i++){ binary += String.fromCharCode(bytes[i]) }\n  return btoa(binary)\n}\n\nexport default class LongPoll {\n\n  constructor(endPoint){\n    this.endPoint = null\n    this.token = null\n    this.skipHeartbeat = true\n    this.reqs = new Set()\n    this.awaitingBatchAck = false\n    this.currentBatch = null\n    this.currentBatchTimer = null\n    this.batchBuffer = []\n    this.onopen = function (){ } // noop\n    this.onerror = function (){ } // noop\n    this.onmessage = function (){ } // noop\n    this.onclose = function (){ } // noop\n    this.pollEndpoint = this.normalizeEndpoint(endPoint)\n    this.readyState = SOCKET_STATES.connecting\n    // we must wait for the caller to finish setting up our callbacks and timeout properties\n    setTimeout(() => this.poll(), 0)\n  }\n\n  normalizeEndpoint(endPoint){\n    return (endPoint\n      .replace(\"ws://\", \"http://\")\n      .replace(\"wss://\", \"https://\")\n      .replace(new RegExp(\"(.*)\\/\" + TRANSPORTS.websocket), \"$1/\" + TRANSPORTS.longpoll))\n  }\n\n  endpointURL(){\n    return Ajax.appendParams(this.pollEndpoint, {token: this.token})\n  }\n\n  closeAndRetry(code, reason, wasClean){\n    this.close(code, reason, wasClean)\n    this.readyState = SOCKET_STATES.connecting\n  }\n\n  ontimeout(){\n    this.onerror(\"timeout\")\n    this.closeAndRetry(1005, \"timeout\", false)\n  }\n\n  isActive(){ return this.readyState === SOCKET_STATES.open || this.readyState === SOCKET_STATES.connecting }\n\n  poll(){\n    this.ajax(\"GET\", \"application/json\", null, () => this.ontimeout(), resp => {\n      if(resp){\n        var {status, token, messages} = resp\n        this.token = token\n      } else {\n        status = 0\n      }\n\n      switch(status){\n        case 200:\n          messages.forEach(msg => {\n            // Tasks are what things like event handlers, setTimeout callbacks,\n            // promise resolves and more are run within.\n            // In modern browsers, there are two different kinds of tasks,\n            // microtasks and macrotasks.\n            // Microtasks are mainly used for Promises, while macrotasks are\n            // used for everything else.\n            // Microtasks always have priority over macrotasks. If the JS engine\n            // is looking for a task to run, it will always try to empty the\n            // microtask queue before attempting to run anything from the\n            // macrotask queue.\n            //\n            // For the WebSocket transport, messages always arrive in their own\n            // event. This means that if any promises are resolved from within,\n            // their callbacks will always finish execution by the time the\n            // next message event handler is run.\n            //\n            // In order to emulate this behaviour, we need to make sure each\n            // onmessage handler is run within its own macrotask.\n            setTimeout(() => this.onmessage({data: msg}), 0)\n          })\n          this.poll()\n          break\n        case 204:\n          this.poll()\n          break\n        case 410:\n          this.readyState = SOCKET_STATES.open\n          this.onopen({})\n          this.poll()\n          break\n        case 403:\n          this.onerror(403)\n          this.close(1008, \"forbidden\", false)\n          break\n        case 0:\n        case 500:\n          this.onerror(500)\n          this.closeAndRetry(1011, \"internal server error\", 500)\n          break\n        default: throw new Error(`unhandled poll status ${status}`)\n      }\n    })\n  }\n\n  // we collect all pushes within the current event loop by\n  // setTimeout 0, which optimizes back-to-back procedural\n  // pushes against an empty buffer\n\n  send(body){\n    if(typeof(body) !== \"string\"){ body = arrayBufferToBase64(body) }\n    if(this.currentBatch){\n      this.currentBatch.push(body)\n    } else if(this.awaitingBatchAck){\n      this.batchBuffer.push(body)\n    } else {\n      this.currentBatch = [body]\n      this.currentBatchTimer = setTimeout(() => {\n        this.batchSend(this.currentBatch)\n        this.currentBatch = null\n      }, 0)\n    }\n  }\n\n  batchSend(messages){\n    this.awaitingBatchAck = true\n    this.ajax(\"POST\", \"application/x-ndjson\", messages.join(\"\\n\"), () => this.onerror(\"timeout\"), resp => {\n      this.awaitingBatchAck = false\n      if(!resp || resp.status !== 200){\n        this.onerror(resp && resp.status)\n        this.closeAndRetry(1011, \"internal server error\", false)\n      } else if(this.batchBuffer.length > 0){\n        this.batchSend(this.batchBuffer)\n        this.batchBuffer = []\n      }\n    })\n  }\n\n  close(code, reason, wasClean){\n    for(let req of this.reqs){ req.abort() }\n    this.readyState = SOCKET_STATES.closed\n    let opts = Object.assign({code: 1000, reason: undefined, wasClean: true}, {code, reason, wasClean})\n    this.batchBuffer = []\n    clearTimeout(this.currentBatchTimer)\n    this.currentBatchTimer = null\n    if(typeof(CloseEvent) !== \"undefined\"){\n      this.onclose(new CloseEvent(\"close\", opts))\n    } else {\n      this.onclose(opts)\n    }\n  }\n\n  ajax(method, contentType, body, onCallerTimeout, callback){\n    let req\n    let ontimeout = () => {\n      this.reqs.delete(req)\n      onCallerTimeout()\n    }\n    req = Ajax.request(method, this.endpointURL(), contentType, body, this.timeout, ontimeout, resp => {\n      this.reqs.delete(req)\n      if(this.isActive()){ callback(resp) }\n    })\n    this.reqs.add(req)\n  }\n}\n", "/**\n * Initializes the Presence\n * @param {Channel} channel - The Channel\n * @param {Object} opts - The options,\n *        for example `{events: {state: \"state\", diff: \"diff\"}}`\n */\nexport default class Presence {\n\n  constructor(channel, opts = {}){\n    let events = opts.events || {state: \"presence_state\", diff: \"presence_diff\"}\n    this.state = {}\n    this.pendingDiffs = []\n    this.channel = channel\n    this.joinRef = null\n    this.caller = {\n      onJoin: function (){ },\n      onLeave: function (){ },\n      onSync: function (){ }\n    }\n\n    this.channel.on(events.state, newState => {\n      let {onJoin, onLeave, onSync} = this.caller\n\n      this.joinRef = this.channel.joinRef()\n      this.state = Presence.syncState(this.state, newState, onJoin, onLeave)\n\n      this.pendingDiffs.forEach(diff => {\n        this.state = Presence.syncDiff(this.state, diff, onJoin, onLeave)\n      })\n      this.pendingDiffs = []\n      onSync()\n    })\n\n    this.channel.on(events.diff, diff => {\n      let {onJoin, onLeave, onSync} = this.caller\n\n      if(this.inPendingSyncState()){\n        this.pendingDiffs.push(diff)\n      } else {\n        this.state = Presence.syncDiff(this.state, diff, onJoin, onLeave)\n        onSync()\n      }\n    })\n  }\n\n  onJoin(callback){ this.caller.onJoin = callback }\n\n  onLeave(callback){ this.caller.onLeave = callback }\n\n  onSync(callback){ this.caller.onSync = callback }\n\n  list(by){ return Presence.list(this.state, by) }\n\n  inPendingSyncState(){\n    return !this.joinRef || (this.joinRef !== this.channel.joinRef())\n  }\n\n  // lower-level public static API\n\n  /**\n   * Used to sync the list of presences on the server\n   * with the client's state. An optional `onJoin` and `onLeave` callback can\n   * be provided to react to changes in the client's local presences across\n   * disconnects and reconnects with the server.\n   *\n   * @returns {Presence}\n   */\n  static syncState(currentState, newState, onJoin, onLeave){\n    let state = this.clone(currentState)\n    let joins = {}\n    let leaves = {}\n\n    this.map(state, (key, presence) => {\n      if(!newState[key]){\n        leaves[key] = presence\n      }\n    })\n    this.map(newState, (key, newPresence) => {\n      let currentPresence = state[key]\n      if(currentPresence){\n        let newRefs = newPresence.metas.map(m => m.phx_ref)\n        let curRefs = currentPresence.metas.map(m => m.phx_ref)\n        let joinedMetas = newPresence.metas.filter(m => curRefs.indexOf(m.phx_ref) < 0)\n        let leftMetas = currentPresence.metas.filter(m => newRefs.indexOf(m.phx_ref) < 0)\n        if(joinedMetas.length > 0){\n          joins[key] = newPresence\n          joins[key].metas = joinedMetas\n        }\n        if(leftMetas.length > 0){\n          leaves[key] = this.clone(currentPresence)\n          leaves[key].metas = leftMetas\n        }\n      } else {\n        joins[key] = newPresence\n      }\n    })\n    return this.syncDiff(state, {joins: joins, leaves: leaves}, onJoin, onLeave)\n  }\n\n  /**\n   *\n   * Used to sync a diff of presence join and leave\n   * events from the server, as they happen. Like `syncState`, `syncDiff`\n   * accepts optional `onJoin` and `onLeave` callbacks to react to a user\n   * joining or leaving from a device.\n   *\n   * @returns {Presence}\n   */\n  static syncDiff(state, diff, onJoin, onLeave){\n    let {joins, leaves} = this.clone(diff)\n    if(!onJoin){ onJoin = function (){ } }\n    if(!onLeave){ onLeave = function (){ } }\n\n    this.map(joins, (key, newPresence) => {\n      let currentPresence = state[key]\n      state[key] = this.clone(newPresence)\n      if(currentPresence){\n        let joinedRefs = state[key].metas.map(m => m.phx_ref)\n        let curMetas = currentPresence.metas.filter(m => joinedRefs.indexOf(m.phx_ref) < 0)\n        state[key].metas.unshift(...curMetas)\n      }\n      onJoin(key, currentPresence, newPresence)\n    })\n    this.map(leaves, (key, leftPresence) => {\n      let currentPresence = state[key]\n      if(!currentPresence){ return }\n      let refsToRemove = leftPresence.metas.map(m => m.phx_ref)\n      currentPresence.metas = currentPresence.metas.filter(p => {\n        return refsToRemove.indexOf(p.phx_ref) < 0\n      })\n      onLeave(key, currentPresence, leftPresence)\n      if(currentPresence.metas.length === 0){\n        delete state[key]\n      }\n    })\n    return state\n  }\n\n  /**\n   * Returns the array of presences, with selected metadata.\n   *\n   * @param {Object} presences\n   * @param {Function} chooser\n   *\n   * @returns {Presence}\n   */\n  static list(presences, chooser){\n    if(!chooser){ chooser = function (key, pres){ return pres } }\n\n    return this.map(presences, (key, presence) => {\n      return chooser(key, presence)\n    })\n  }\n\n  // private\n\n  static map(obj, func){\n    return Object.getOwnPropertyNames(obj).map(key => func(key, obj[key]))\n  }\n\n  static clone(obj){ return JSON.parse(JSON.stringify(obj)) }\n}\n", "/* The default serializer for encoding and decoding messages */\nimport {\n  CHANNEL_EVENTS\n} from \"./constants\"\n\nexport default {\n  HEADER_LENGTH: 1,\n  META_LENGTH: 4,\n  KINDS: {push: 0, reply: 1, broadcast: 2},\n\n  encode(msg, callback){\n    if(msg.payload.constructor === ArrayBuffer){\n      return callback(this.binaryEncode(msg))\n    } else {\n      let payload = [msg.join_ref, msg.ref, msg.topic, msg.event, msg.payload]\n      return callback(JSON.stringify(payload))\n    }\n  },\n\n  decode(rawPayload, callback){\n    if(rawPayload.constructor === ArrayBuffer){\n      return callback(this.binaryDecode(rawPayload))\n    } else {\n      let [join_ref, ref, topic, event, payload] = JSON.parse(rawPayload)\n      return callback({join_ref, ref, topic, event, payload})\n    }\n  },\n\n  // private\n\n  binaryEncode(message){\n    let {join_ref, ref, event, topic, payload} = message\n    let metaLength = this.META_LENGTH + join_ref.length + ref.length + topic.length + event.length\n    let header = new ArrayBuffer(this.HEADER_LENGTH + metaLength)\n    let view = new DataView(header)\n    let offset = 0\n\n    view.setUint8(offset++, this.KINDS.push) // kind\n    view.setUint8(offset++, join_ref.length)\n    view.setUint8(offset++, ref.length)\n    view.setUint8(offset++, topic.length)\n    view.setUint8(offset++, event.length)\n    Array.from(join_ref, char => view.setUint8(offset++, char.charCodeAt(0)))\n    Array.from(ref, char => view.setUint8(offset++, char.charCodeAt(0)))\n    Array.from(topic, char => view.setUint8(offset++, char.charCodeAt(0)))\n    Array.from(event, char => view.setUint8(offset++, char.charCodeAt(0)))\n\n    var combined = new Uint8Array(header.byteLength + payload.byteLength)\n    combined.set(new Uint8Array(header), 0)\n    combined.set(new Uint8Array(payload), header.byteLength)\n\n    return combined.buffer\n  },\n\n  binaryDecode(buffer){\n    let view = new DataView(buffer)\n    let kind = view.getUint8(0)\n    let decoder = new TextDecoder()\n    switch(kind){\n      case this.KINDS.push: return this.decodePush(buffer, view, decoder)\n      case this.KINDS.reply: return this.decodeReply(buffer, view, decoder)\n      case this.KINDS.broadcast: return this.decodeBroadcast(buffer, view, decoder)\n    }\n  },\n\n  decodePush(buffer, view, decoder){\n    let joinRefSize = view.getUint8(1)\n    let topicSize = view.getUint8(2)\n    let eventSize = view.getUint8(3)\n    let offset = this.HEADER_LENGTH + this.META_LENGTH - 1 // pushes have no ref\n    let joinRef = decoder.decode(buffer.slice(offset, offset + joinRefSize))\n    offset = offset + joinRefSize\n    let topic = decoder.decode(buffer.slice(offset, offset + topicSize))\n    offset = offset + topicSize\n    let event = decoder.decode(buffer.slice(offset, offset + eventSize))\n    offset = offset + eventSize\n    let data = buffer.slice(offset, buffer.byteLength)\n    return {join_ref: joinRef, ref: null, topic: topic, event: event, payload: data}\n  },\n\n  decodeReply(buffer, view, decoder){\n    let joinRefSize = view.getUint8(1)\n    let refSize = view.getUint8(2)\n    let topicSize = view.getUint8(3)\n    let eventSize = view.getUint8(4)\n    let offset = this.HEADER_LENGTH + this.META_LENGTH\n    let joinRef = decoder.decode(buffer.slice(offset, offset + joinRefSize))\n    offset = offset + joinRefSize\n    let ref = decoder.decode(buffer.slice(offset, offset + refSize))\n    offset = offset + refSize\n    let topic = decoder.decode(buffer.slice(offset, offset + topicSize))\n    offset = offset + topicSize\n    let event = decoder.decode(buffer.slice(offset, offset + eventSize))\n    offset = offset + eventSize\n    let data = buffer.slice(offset, buffer.byteLength)\n    let payload = {status: event, response: data}\n    return {join_ref: joinRef, ref: ref, topic: topic, event: CHANNEL_EVENTS.reply, payload: payload}\n  },\n\n  decodeBroadcast(buffer, view, decoder){\n    let topicSize = view.getUint8(1)\n    let eventSize = view.getUint8(2)\n    let offset = this.HEADER_LENGTH + 2\n    let topic = decoder.decode(buffer.slice(offset, offset + topicSize))\n    offset = offset + topicSize\n    let event = decoder.decode(buffer.slice(offset, offset + eventSize))\n    offset = offset + eventSize\n    let data = buffer.slice(offset, buffer.byteLength)\n\n    return {join_ref: null, ref: null, topic: topic, event: event, payload: data}\n  }\n}\n", "import {\n  global,\n  phxWindow,\n  CHANNEL_EVENTS,\n  DEFAULT_TIMEOUT,\n  DEFAULT_VSN,\n  SOCKET_STATES,\n  TRANSPORTS,\n  WS_CLOSE_NORMAL\n} from \"./constants\"\n\nimport {\n  closure\n} from \"./utils\"\n\nimport Ajax from \"./ajax\"\nimport Channel from \"./channel\"\nimport LongPoll from \"./longpoll\"\nimport Serializer from \"./serializer\"\nimport Timer from \"./timer\"\n\n/** Initializes the Socket *\n *\n * For IE8 support use an ES5-shim (https://github.com/es-shims/es5-shim)\n *\n * @param {string} endPoint - The string WebSocket endpoint, ie, `\"ws://example.com/socket\"`,\n *                                               `\"wss://example.com\"`\n *                                               `\"/socket\"` (inherited host & protocol)\n * @param {Object} [opts] - Optional configuration\n * @param {Function} [opts.transport] - The Websocket Transport, for example WebSocket or Phoenix.LongPoll.\n *\n * Defaults to WebSocket with automatic LongPoll fallback if WebSocket is not defined.\n * To fallback to LongPoll when WebSocket attempts fail, use `longPollFallbackMs: 2500`.\n *\n * @param {Function} [opts.longPollFallbackMs] - The millisecond time to attempt the primary transport\n * before falling back to the LongPoll transport. Disabled by default.\n *\n * @param {Function} [opts.debug] - When true, enables debug logging. Default false.\n *\n * @param {Function} [opts.encode] - The function to encode outgoing messages.\n *\n * Defaults to JSON encoder.\n *\n * @param {Function} [opts.decode] - The function to decode incoming messages.\n *\n * Defaults to JSON:\n *\n * ```javascript\n * (payload, callback) => callback(JSON.parse(payload))\n * ```\n *\n * @param {number} [opts.timeout] - The default timeout in milliseconds to trigger push timeouts.\n *\n * Defaults `DEFAULT_TIMEOUT`\n * @param {number} [opts.heartbeatIntervalMs] - The millisec interval to send a heartbeat message\n * @param {number} [opts.reconnectAfterMs] - The optional function that returns the millisec\n * socket reconnect interval.\n *\n * Defaults to stepped backoff of:\n *\n * ```javascript\n * function(tries){\n *   return [10, 50, 100, 150, 200, 250, 500, 1000, 2000][tries - 1] || 5000\n * }\n * ````\n *\n * @param {number} [opts.rejoinAfterMs] - The optional function that returns the millisec\n * rejoin interval for individual channels.\n *\n * ```javascript\n * function(tries){\n *   return [1000, 2000, 5000][tries - 1] || 10000\n * }\n * ````\n *\n * @param {Function} [opts.logger] - The optional function for specialized logging, ie:\n *\n * ```javascript\n * function(kind, msg, data) {\n *   console.log(`${kind}: ${msg}`, data)\n * }\n * ```\n *\n * @param {number} [opts.longpollerTimeout] - The maximum timeout of a long poll AJAX request.\n *\n * Defaults to 20s (double the server long poll timer).\n *\n * @param {(Object|function)} [opts.params] - The optional params to pass when connecting\n * @param {string} [opts.binaryType] - The binary type to use for binary WebSocket frames.\n *\n * Defaults to \"arraybuffer\"\n *\n * @param {vsn} [opts.vsn] - The serializer's protocol version to send on connect.\n *\n * Defaults to DEFAULT_VSN.\n *\n * @param {Object} [opts.sessionStorage] - An optional Storage compatible object\n * Phoenix uses sessionStorage for longpoll fallback history. Overriding the store is\n * useful when Phoenix won't have access to `sessionStorage`. For example, This could\n * happen if a site loads a cross-domain channel in an iframe. Example usage:\n *\n *     class InMemoryStorage {\n *       constructor() { this.storage = {} }\n *       getItem(keyName) { return this.storage[keyName] || null }\n *       removeItem(keyName) { delete this.storage[keyName] }\n *       setItem(keyName, keyValue) { this.storage[keyName] = keyValue }\n *     }\n *\n*/\nexport default class Socket {\n  constructor(endPoint, opts = {}){\n    this.stateChangeCallbacks = {open: [], close: [], error: [], message: []}\n    this.channels = []\n    this.sendBuffer = []\n    this.ref = 0\n    this.timeout = opts.timeout || DEFAULT_TIMEOUT\n    this.transport = opts.transport || global.WebSocket || LongPoll\n    this.primaryPassedHealthCheck = false\n    this.longPollFallbackMs = opts.longPollFallbackMs\n    this.fallbackTimer = null\n    this.sessionStore = opts.sessionStorage || (global && global.sessionStorage)\n    this.establishedConnections = 0\n    this.defaultEncoder = Serializer.encode.bind(Serializer)\n    this.defaultDecoder = Serializer.decode.bind(Serializer)\n    this.closeWasClean = false\n    this.disconnecting = false\n    this.binaryType = opts.binaryType || \"arraybuffer\"\n    this.connectClock = 1\n    if(this.transport !== LongPoll){\n      this.encode = opts.encode || this.defaultEncoder\n      this.decode = opts.decode || this.defaultDecoder\n    } else {\n      this.encode = this.defaultEncoder\n      this.decode = this.defaultDecoder\n    }\n    let awaitingConnectionOnPageShow = null\n    if(phxWindow && phxWindow.addEventListener){\n      phxWindow.addEventListener(\"pagehide\", _e => {\n        if(this.conn){\n          this.disconnect()\n          awaitingConnectionOnPageShow = this.connectClock\n        }\n      })\n      phxWindow.addEventListener(\"pageshow\", _e => {\n        if(awaitingConnectionOnPageShow === this.connectClock){\n          awaitingConnectionOnPageShow = null\n          this.connect()\n        }\n      })\n    }\n    this.heartbeatIntervalMs = opts.heartbeatIntervalMs || 30000\n    this.rejoinAfterMs = (tries) => {\n      if(opts.rejoinAfterMs){\n        return opts.rejoinAfterMs(tries)\n      } else {\n        return [1000, 2000, 5000][tries - 1] || 10000\n      }\n    }\n    this.reconnectAfterMs = (tries) => {\n      if(opts.reconnectAfterMs){\n        return opts.reconnectAfterMs(tries)\n      } else {\n        return [10, 50, 100, 150, 200, 250, 500, 1000, 2000][tries - 1] || 5000\n      }\n    }\n    this.logger = opts.logger || null\n    if(!this.logger && opts.debug){\n      this.logger = (kind, msg, data) => { console.log(`${kind}: ${msg}`, data) }\n    }\n    this.longpollerTimeout = opts.longpollerTimeout || 20000\n    this.params = closure(opts.params || {})\n    this.endPoint = `${endPoint}/${TRANSPORTS.websocket}`\n    this.vsn = opts.vsn || DEFAULT_VSN\n    this.heartbeatTimeoutTimer = null\n    this.heartbeatTimer = null\n    this.pendingHeartbeatRef = null\n    this.reconnectTimer = new Timer(() => {\n      this.teardown(() => this.connect())\n    }, this.reconnectAfterMs)\n  }\n\n  /**\n   * Returns the LongPoll transport reference\n   */\n  getLongPollTransport(){ return LongPoll }\n\n  /**\n   * Disconnects and replaces the active transport\n   *\n   * @param {Function} newTransport - The new transport class to instantiate\n   *\n   */\n  replaceTransport(newTransport){\n    this.connectClock++\n    this.closeWasClean = true\n    clearTimeout(this.fallbackTimer)\n    this.reconnectTimer.reset()\n    if(this.conn){\n      this.conn.close()\n      this.conn = null\n    }\n    this.transport = newTransport\n  }\n\n  /**\n   * Returns the socket protocol\n   *\n   * @returns {string}\n   */\n  protocol(){ return location.protocol.match(/^https/) ? \"wss\" : \"ws\" }\n\n  /**\n   * The fully qualified socket url\n   *\n   * @returns {string}\n   */\n  endPointURL(){\n    let uri = Ajax.appendParams(\n      Ajax.appendParams(this.endPoint, this.params()), {vsn: this.vsn})\n    if(uri.charAt(0) !== \"/\"){ return uri }\n    if(uri.charAt(1) === \"/\"){ return `${this.protocol()}:${uri}` }\n\n    return `${this.protocol()}://${location.host}${uri}`\n  }\n\n  /**\n   * Disconnects the socket\n   *\n   * See https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes for valid status codes.\n   *\n   * @param {Function} callback - Optional callback which is called after socket is disconnected.\n   * @param {integer} code - A status code for disconnection (Optional).\n   * @param {string} reason - A textual description of the reason to disconnect. (Optional)\n   */\n  disconnect(callback, code, reason){\n    this.connectClock++\n    this.disconnecting = true\n    this.closeWasClean = true\n    clearTimeout(this.fallbackTimer)\n    this.reconnectTimer.reset()\n    this.teardown(() => {\n      this.disconnecting = false\n      callback && callback()\n    }, code, reason)\n  }\n\n  /**\n   *\n   * @param {Object} params - The params to send when connecting, for example `{user_id: userToken}`\n   *\n   * Passing params to connect is deprecated; pass them in the Socket constructor instead:\n   * `new Socket(\"/socket\", {params: {user_id: userToken}})`.\n   */\n  connect(params){\n    if(params){\n      console && console.log(\"passing params to connect is deprecated. Instead pass :params to the Socket constructor\")\n      this.params = closure(params)\n    }\n    if(this.conn && !this.disconnecting){ return }\n    if(this.longPollFallbackMs && this.transport !== LongPoll){\n      this.connectWithFallback(LongPoll, this.longPollFallbackMs)\n    } else {\n      this.transportConnect()\n    }\n  }\n\n  /**\n   * Logs the message. Override `this.logger` for specialized logging. noops by default\n   * @param {string} kind\n   * @param {string} msg\n   * @param {Object} data\n   */\n  log(kind, msg, data){ this.logger && this.logger(kind, msg, data) }\n\n  /**\n   * Returns true if a logger has been set on this socket.\n   */\n  hasLogger(){ return this.logger !== null }\n\n  /**\n   * Registers callbacks for connection open events\n   *\n   * @example socket.onOpen(function(){ console.info(\"the socket was opened\") })\n   *\n   * @param {Function} callback\n   */\n  onOpen(callback){\n    let ref = this.makeRef()\n    this.stateChangeCallbacks.open.push([ref, callback])\n    return ref\n  }\n\n  /**\n   * Registers callbacks for connection close events\n   * @param {Function} callback\n   */\n  onClose(callback){\n    let ref = this.makeRef()\n    this.stateChangeCallbacks.close.push([ref, callback])\n    return ref\n  }\n\n  /**\n   * Registers callbacks for connection error events\n   *\n   * @example socket.onError(function(error){ alert(\"An error occurred\") })\n   *\n   * @param {Function} callback\n   */\n  onError(callback){\n    let ref = this.makeRef()\n    this.stateChangeCallbacks.error.push([ref, callback])\n    return ref\n  }\n\n  /**\n   * Registers callbacks for connection message events\n   * @param {Function} callback\n   */\n  onMessage(callback){\n    let ref = this.makeRef()\n    this.stateChangeCallbacks.message.push([ref, callback])\n    return ref\n  }\n\n  /**\n   * Pings the server and invokes the callback with the RTT in milliseconds\n   * @param {Function} callback\n   *\n   * Returns true if the ping was pushed or false if unable to be pushed.\n   */\n  ping(callback){\n    if(!this.isConnected()){ return false }\n    let ref = this.makeRef()\n    let startTime = Date.now()\n    this.push({topic: \"phoenix\", event: \"heartbeat\", payload: {}, ref: ref})\n    let onMsgRef = this.onMessage(msg => {\n      if(msg.ref === ref){\n        this.off([onMsgRef])\n        callback(Date.now() - startTime)\n      }\n    })\n    return true\n  }\n\n  /**\n   * @private\n   */\n\n  transportConnect(){\n    this.connectClock++\n    this.closeWasClean = false\n    this.conn = new this.transport(this.endPointURL())\n    this.conn.binaryType = this.binaryType\n    this.conn.timeout = this.longpollerTimeout\n    this.conn.onopen = () => this.onConnOpen()\n    this.conn.onerror = error => this.onConnError(error)\n    this.conn.onmessage = event => this.onConnMessage(event)\n    this.conn.onclose = event => this.onConnClose(event)\n  }\n\n  getSession(key){ return this.sessionStore && this.sessionStore.getItem(key) }\n\n  storeSession(key, val){ this.sessionStore && this.sessionStore.setItem(key, val) }\n\n  connectWithFallback(fallbackTransport, fallbackThreshold = 2500){\n    clearTimeout(this.fallbackTimer)\n    let established = false\n    let primaryTransport = true\n    let openRef, errorRef\n    let fallback = (reason) => {\n      this.log(\"transport\", `falling back to ${fallbackTransport.name}...`, reason)\n      this.off([openRef, errorRef])\n      primaryTransport = false\n      this.replaceTransport(fallbackTransport)\n      this.transportConnect()\n    }\n    if(this.getSession(`phx:fallback:${fallbackTransport.name}`)){ return fallback(\"memorized\") }\n\n    this.fallbackTimer = setTimeout(fallback, fallbackThreshold)\n\n    errorRef = this.onError(reason => {\n      this.log(\"transport\", \"error\", reason)\n      if(primaryTransport && !established){\n        clearTimeout(this.fallbackTimer)\n        fallback(reason)\n      }\n    })\n    this.onOpen(() => {\n      established = true\n      if(!primaryTransport){\n        // only memorize LP if we never connected to primary\n        if(!this.primaryPassedHealthCheck){ this.storeSession(`phx:fallback:${fallbackTransport.name}`, \"true\") }\n        return this.log(\"transport\", `established ${fallbackTransport.name} fallback`)\n      }\n      // if we've established primary, give the fallback a new period to attempt ping\n      clearTimeout(this.fallbackTimer)\n      this.fallbackTimer = setTimeout(fallback, fallbackThreshold)\n      this.ping(rtt => {\n        this.log(\"transport\", \"connected to primary after\", rtt)\n        this.primaryPassedHealthCheck = true\n        clearTimeout(this.fallbackTimer)\n      })\n    })\n    this.transportConnect()\n  }\n\n  clearHeartbeats(){\n    clearTimeout(this.heartbeatTimer)\n    clearTimeout(this.heartbeatTimeoutTimer)\n  }\n\n  onConnOpen(){\n    if(this.hasLogger()) this.log(\"transport\", `${this.transport.name} connected to ${this.endPointURL()}`)\n    this.closeWasClean = false\n    this.disconnecting = false\n    this.establishedConnections++\n    this.flushSendBuffer()\n    this.reconnectTimer.reset()\n    this.resetHeartbeat()\n    this.stateChangeCallbacks.open.forEach(([, callback]) => callback())\n  }\n\n  /**\n   * @private\n   */\n\n  heartbeatTimeout(){\n    if(this.pendingHeartbeatRef){\n      this.pendingHeartbeatRef = null\n      if(this.hasLogger()){ this.log(\"transport\", \"heartbeat timeout. Attempting to re-establish connection\") }\n      this.triggerChanError()\n      this.closeWasClean = false\n      this.teardown(() => this.reconnectTimer.scheduleTimeout(), WS_CLOSE_NORMAL, \"heartbeat timeout\")\n    }\n  }\n\n  resetHeartbeat(){\n    if(this.conn && this.conn.skipHeartbeat){ return }\n    this.pendingHeartbeatRef = null\n    this.clearHeartbeats()\n    this.heartbeatTimer = setTimeout(() => this.sendHeartbeat(), this.heartbeatIntervalMs)\n  }\n\n  teardown(callback, code, reason){\n    if(!this.conn){\n      return callback && callback()\n    }\n    let connectClock = this.connectClock\n\n    this.waitForBufferDone(() => {\n      if(connectClock !== this.connectClock){ return }\n      if(this.conn){\n        if(code){ this.conn.close(code, reason || \"\") } else { this.conn.close() }\n      }\n\n      this.waitForSocketClosed(() => {\n        if(connectClock !== this.connectClock){ return }\n        if(this.conn){\n          this.conn.onopen = function (){ } // noop\n          this.conn.onerror = function (){ } // noop\n          this.conn.onmessage = function (){ } // noop\n          this.conn.onclose = function (){ } // noop\n          this.conn = null\n        }\n\n        callback && callback()\n      })\n    })\n  }\n\n  waitForBufferDone(callback, tries = 1){\n    if(tries === 5 || !this.conn || !this.conn.bufferedAmount){\n      callback()\n      return\n    }\n\n    setTimeout(() => {\n      this.waitForBufferDone(callback, tries + 1)\n    }, 150 * tries)\n  }\n\n  waitForSocketClosed(callback, tries = 1){\n    if(tries === 5 || !this.conn || this.conn.readyState === SOCKET_STATES.closed){\n      callback()\n      return\n    }\n\n    setTimeout(() => {\n      this.waitForSocketClosed(callback, tries + 1)\n    }, 150 * tries)\n  }\n\n  onConnClose(event){\n    let closeCode = event && event.code\n    if(this.hasLogger()) this.log(\"transport\", \"close\", event)\n    this.triggerChanError()\n    this.clearHeartbeats()\n    if(!this.closeWasClean && closeCode !== 1000){\n      this.reconnectTimer.scheduleTimeout()\n    }\n    this.stateChangeCallbacks.close.forEach(([, callback]) => callback(event))\n  }\n\n  /**\n   * @private\n   */\n  onConnError(error){\n    if(this.hasLogger()) this.log(\"transport\", error)\n    let transportBefore = this.transport\n    let establishedBefore = this.establishedConnections\n    this.stateChangeCallbacks.error.forEach(([, callback]) => {\n      callback(error, transportBefore, establishedBefore)\n    })\n    if(transportBefore === this.transport || establishedBefore > 0){\n      this.triggerChanError()\n    }\n  }\n\n  /**\n   * @private\n   */\n  triggerChanError(){\n    this.channels.forEach(channel => {\n      if(!(channel.isErrored() || channel.isLeaving() || channel.isClosed())){\n        channel.trigger(CHANNEL_EVENTS.error)\n      }\n    })\n  }\n\n  /**\n   * @returns {string}\n   */\n  connectionState(){\n    switch(this.conn && this.conn.readyState){\n      case SOCKET_STATES.connecting: return \"connecting\"\n      case SOCKET_STATES.open: return \"open\"\n      case SOCKET_STATES.closing: return \"closing\"\n      default: return \"closed\"\n    }\n  }\n\n  /**\n   * @returns {boolean}\n   */\n  isConnected(){ return this.connectionState() === \"open\" }\n\n  /**\n   * @private\n   *\n   * @param {Channel}\n   */\n  remove(channel){\n    this.off(channel.stateChangeRefs)\n    this.channels = this.channels.filter(c => c !== channel)\n  }\n\n  /**\n   * Removes `onOpen`, `onClose`, `onError,` and `onMessage` registrations.\n   *\n   * @param {refs} - list of refs returned by calls to\n   *                 `onOpen`, `onClose`, `onError,` and `onMessage`\n   */\n  off(refs){\n    for(let key in this.stateChangeCallbacks){\n      this.stateChangeCallbacks[key] = this.stateChangeCallbacks[key].filter(([ref]) => {\n        return refs.indexOf(ref) === -1\n      })\n    }\n  }\n\n  /**\n   * Initiates a new channel for the given topic\n   *\n   * @param {string} topic\n   * @param {Object} chanParams - Parameters for the channel\n   * @returns {Channel}\n   */\n  channel(topic, chanParams = {}){\n    let chan = new Channel(topic, chanParams, this)\n    this.channels.push(chan)\n    return chan\n  }\n\n  /**\n   * @param {Object} data\n   */\n  push(data){\n    if(this.hasLogger()){\n      let {topic, event, payload, ref, join_ref} = data\n      this.log(\"push\", `${topic} ${event} (${join_ref}, ${ref})`, payload)\n    }\n\n    if(this.isConnected()){\n      this.encode(data, result => this.conn.send(result))\n    } else {\n      this.sendBuffer.push(() => this.encode(data, result => this.conn.send(result)))\n    }\n  }\n\n  /**\n   * Return the next message ref, accounting for overflows\n   * @returns {string}\n   */\n  makeRef(){\n    let newRef = this.ref + 1\n    if(newRef === this.ref){ this.ref = 0 } else { this.ref = newRef }\n\n    return this.ref.toString()\n  }\n\n  sendHeartbeat(){\n    if(this.pendingHeartbeatRef && !this.isConnected()){ return }\n    this.pendingHeartbeatRef = this.makeRef()\n    this.push({topic: \"phoenix\", event: \"heartbeat\", payload: {}, ref: this.pendingHeartbeatRef})\n    this.heartbeatTimeoutTimer = setTimeout(() => this.heartbeatTimeout(), this.heartbeatIntervalMs)\n  }\n\n  flushSendBuffer(){\n    if(this.isConnected() && this.sendBuffer.length > 0){\n      this.sendBuffer.forEach(callback => callback())\n      this.sendBuffer = []\n    }\n  }\n\n  onConnMessage(rawMessage){\n    this.decode(rawMessage.data, msg => {\n      let {topic, event, payload, ref, join_ref} = msg\n      if(ref && ref === this.pendingHeartbeatRef){\n        this.clearHeartbeats()\n        this.pendingHeartbeatRef = null\n        this.heartbeatTimer = setTimeout(() => this.sendHeartbeat(), this.heartbeatIntervalMs)\n      }\n\n      if(this.hasLogger()) this.log(\"receive\", `${payload.status || \"\"} ${topic} ${event} ${ref && \"(\" + ref + \")\" || \"\"}`, payload)\n\n      for(let i = 0; i < this.channels.length; i++){\n        const channel = this.channels[i]\n        if(!channel.isMember(topic, event, payload, join_ref)){ continue }\n        channel.trigger(event, payload, ref, join_ref)\n      }\n\n      for(let i = 0; i < this.stateChangeCallbacks.message.length; i++){\n        let [, callback] = this.stateChangeCallbacks.message[i]\n        callback(msg)\n      }\n    })\n  }\n\n  leaveOpenTopic(topic){\n    let dupChannel = this.channels.find(c => c.topic === topic && (c.isJoined() || c.isJoining()))\n    if(dupChannel){\n      if(this.hasLogger()) this.log(\"transport\", `leaving duplicate topic \"${topic}\"`)\n      dupChannel.leave()\n    }\n  }\n}\n", "export const CONSECUTIVE_RELOADS = \"consecutive-reloads\"\nexport const MAX_RELOADS = 10\nexport const RELOAD_JITTER_MIN = 5000\nexport const RELOAD_JITTER_MAX = 10000\nexport const FAILSAFE_JITTER = 30000\nexport const PHX_EVENT_CLASSES = [\n  \"phx-click-loading\", \"phx-change-loading\", \"phx-submit-loading\",\n  \"phx-keydown-loading\", \"phx-keyup-loading\", \"phx-blur-loading\", \"phx-focus-loading\",\n  \"phx-hook-loading\"\n]\nexport const PHX_COMPONENT = \"data-phx-component\"\nexport const PHX_LIVE_LINK = \"data-phx-link\"\nexport const PHX_TRACK_STATIC = \"track-static\"\nexport const PHX_LINK_STATE = \"data-phx-link-state\"\nexport const PHX_REF_LOADING = \"data-phx-ref-loading\"\nexport const PHX_REF_SRC = \"data-phx-ref-src\"\nexport const PHX_REF_LOCK = \"data-phx-ref-lock\"\nexport const PHX_PENDING_REFS = \"phx-pending-refs\"\nexport const PHX_TRACK_UPLOADS = \"track-uploads\"\nexport const PHX_UPLOAD_REF = \"data-phx-upload-ref\"\nexport const PHX_PREFLIGHTED_REFS = \"data-phx-preflighted-refs\"\nexport const PHX_DONE_REFS = \"data-phx-done-refs\"\nexport const PHX_DROP_TARGET = \"drop-target\"\nexport const PHX_ACTIVE_ENTRY_REFS = \"data-phx-active-refs\"\nexport const PHX_LIVE_FILE_UPDATED = \"phx:live-file:updated\"\nexport const PHX_SKIP = \"data-phx-skip\"\nexport const PHX_MAGIC_ID = \"data-phx-id\"\nexport const PHX_PRUNE = \"data-phx-prune\"\nexport const PHX_CONNECTED_CLASS = \"phx-connected\"\nexport const PHX_LOADING_CLASS = \"phx-loading\"\nexport const PHX_ERROR_CLASS = \"phx-error\"\nexport const PHX_CLIENT_ERROR_CLASS = \"phx-client-error\"\nexport const PHX_SERVER_ERROR_CLASS = \"phx-server-error\"\nexport const PHX_PARENT_ID = \"data-phx-parent-id\"\nexport const PHX_MAIN = \"data-phx-main\"\nexport const PHX_ROOT_ID = \"data-phx-root-id\"\nexport const PHX_VIEWPORT_TOP = \"viewport-top\"\nexport const PHX_VIEWPORT_BOTTOM = \"viewport-bottom\"\nexport const PHX_TRIGGER_ACTION = \"trigger-action\"\nexport const PHX_HAS_FOCUSED = \"phx-has-focused\"\nexport const FOCUSABLE_INPUTS = [\"text\", \"textarea\", \"number\", \"email\", \"password\", \"search\", \"tel\", \"url\", \"date\", \"time\", \"datetime-local\", \"color\", \"range\"]\nexport const CHECKABLE_INPUTS = [\"checkbox\", \"radio\"]\nexport const PHX_HAS_SUBMITTED = \"phx-has-submitted\"\nexport const PHX_SESSION = \"data-phx-session\"\nexport const PHX_VIEW_SELECTOR = `[${PHX_SESSION}]`\nexport const PHX_STICKY = \"data-phx-sticky\"\nexport const PHX_STATIC = \"data-phx-static\"\nexport const PHX_READONLY = \"data-phx-readonly\"\nexport const PHX_DISABLED = \"data-phx-disabled\"\nexport const PHX_DISABLE_WITH = \"disable-with\"\nexport const PHX_DISABLE_WITH_RESTORE = \"data-phx-disable-with-restore\"\nexport const PHX_HOOK = \"hook\"\nexport const PHX_DEBOUNCE = \"debounce\"\nexport const PHX_THROTTLE = \"throttle\"\nexport const PHX_UPDATE = \"update\"\nexport const PHX_STREAM = \"stream\"\nexport const PHX_STREAM_REF = \"data-phx-stream\"\nexport const PHX_KEY = \"key\"\nexport const PHX_PRIVATE = \"phxPrivate\"\nexport const PHX_AUTO_RECOVER = \"auto-recover\"\nexport const PHX_LV_DEBUG = \"phx:live-socket:debug\"\nexport const PHX_LV_PROFILE = \"phx:live-socket:profiling\"\nexport const PHX_LV_LATENCY_SIM = \"phx:live-socket:latency-sim\"\nexport const PHX_LV_HISTORY_POSITION = \"phx:nav-history-position\"\nexport const PHX_PROGRESS = \"progress\"\nexport const PHX_MOUNTED = \"mounted\"\nexport const PHX_RELOAD_STATUS = \"__phoenix_reload_status__\"\nexport const LOADER_TIMEOUT = 1\nexport const MAX_CHILD_JOIN_ATTEMPTS = 3\nexport const BEFORE_UNLOAD_LOADER_TIMEOUT = 200\nexport const DISCONNECTED_TIMEOUT = 500\nexport const BINDING_PREFIX = \"phx-\"\nexport const PUSH_TIMEOUT = 30000\nexport const LINK_HEADER = \"x-requested-with\"\nexport const RESPONSE_URL_HEADER = \"x-response-url\"\nexport const DEBOUNCE_TRIGGER = \"debounce-trigger\"\nexport const THROTTLED = \"throttled\"\nexport const DEBOUNCE_PREV_KEY = \"debounce-prev-key\"\nexport const DEFAULTS = {\n  debounce: 300,\n  throttle: 300\n}\nexport const PHX_PENDING_ATTRS = [PHX_REF_LOADING, PHX_REF_SRC, PHX_REF_LOCK]\n// Rendered\nexport const DYNAMICS = \"d\"\nexport const STATIC = \"s\"\nexport const ROOT = \"r\"\nexport const COMPONENTS = \"c\"\nexport const EVENTS = \"e\"\nexport const REPLY = \"r\"\nexport const TITLE = \"t\"\nexport const TEMPLATES = \"p\"\nexport const STREAM = \"stream\"\n", "import {\n  logError\n} from \"./utils\"\n\nexport default class EntryUploader {\n  constructor(entry, config, liveSocket){\n    let {chunk_size, chunk_timeout} = config\n    this.liveSocket = liveSocket\n    this.entry = entry\n    this.offset = 0\n    this.chunkSize = chunk_size\n    this.chunkTimeout = chunk_timeout\n    this.chunkTimer = null\n    this.errored = false\n    this.uploadChannel = liveSocket.channel(`lvu:${entry.ref}`, {token: entry.metadata()})\n  }\n\n  error(reason){\n    if(this.errored){ return }\n    this.uploadChannel.leave()\n    this.errored = true\n    clearTimeout(this.chunkTimer)\n    this.entry.error(reason)\n  }\n\n  upload(){\n    this.uploadChannel.onError(reason => this.error(reason))\n    this.uploadChannel.join()\n      .receive(\"ok\", _data => this.readNextChunk())\n      .receive(\"error\", reason => this.error(reason))\n  }\n\n  isDone(){ return this.offset >= this.entry.file.size }\n\n  readNextChunk(){\n    let reader = new window.FileReader()\n    let blob = this.entry.file.slice(this.offset, this.chunkSize + this.offset)\n    reader.onload = (e) => {\n      if(e.target.error === null){\n        this.offset += e.target.result.byteLength\n        this.pushChunk(e.target.result)\n      } else {\n        return logError(\"Read error: \" + e.target.error)\n      }\n    }\n    reader.readAsArrayBuffer(blob)\n  }\n\n  pushChunk(chunk){\n    if(!this.uploadChannel.isJoined()){ return }\n    this.uploadChannel.push(\"chunk\", chunk, this.chunkTimeout)\n      .receive(\"ok\", () => {\n        this.entry.progress((this.offset / this.entry.file.size) * 100)\n        if(!this.isDone()){\n          this.chunkTimer = setTimeout(() => this.readNextChunk(), this.liveSocket.getLatencySim() || 0)\n        }\n      })\n      .receive(\"error\", ({reason}) => this.error(reason))\n  }\n}\n", "import {\n  PHX_VIEW_SELECTOR\n} from \"./constants\"\n\nimport EntryUploader from \"./entry_uploader\"\n\nexport let logError = (msg, obj) => console.error && console.error(msg, obj)\n\nexport let isCid = (cid) => {\n  let type = typeof(cid)\n  return type === \"number\" || (type === \"string\" && /^(0|[1-9]\\d*)$/.test(cid))\n}\n\nexport function detectDuplicateIds(){\n  let ids = new Set()\n  let elems = document.querySelectorAll(\"*[id]\")\n  for(let i = 0, len = elems.length; i < len; i++){\n    if(ids.has(elems[i].id)){\n      console.error(`Multiple IDs detected: ${elems[i].id}. Ensure unique element ids.`)\n    } else {\n      ids.add(elems[i].id)\n    }\n  }\n}\n\nexport function detectInvalidStreamInserts(inserts){\n  const errors = new Set()\n  Object.keys(inserts).forEach((id) => {\n    const streamEl = document.getElementById(id)\n    if(streamEl && streamEl.parentElement && streamEl.parentElement.getAttribute(\"phx-update\") !== \"stream\"){\n      errors.add(`The stream container with id \"${streamEl.parentElement.id}\" is missing the phx-update=\"stream\" attribute. Ensure it is set for streams to work properly.`)\n    }\n  })\n  errors.forEach(error => console.error(error))\n}\n\nexport let debug = (view, kind, msg, obj) => {\n  if(view.liveSocket.isDebugEnabled()){\n    console.log(`${view.id} ${kind}: ${msg} - `, obj)\n  }\n}\n\n// wraps value in closure or returns closure\nexport let closure = (val) => typeof val === \"function\" ? val : function (){ return val }\n\nexport let clone = (obj) => { return JSON.parse(JSON.stringify(obj)) }\n\nexport let closestPhxBinding = (el, binding, borderEl) => {\n  do {\n    if(el.matches(`[${binding}]`) && !el.disabled){ return el }\n    el = el.parentElement || el.parentNode\n  } while(el !== null && el.nodeType === 1 && !((borderEl && borderEl.isSameNode(el)) || el.matches(PHX_VIEW_SELECTOR)))\n  return null\n}\n\nexport let isObject = (obj) => {\n  return obj !== null && typeof obj === \"object\" && !(obj instanceof Array)\n}\n\nexport let isEqualObj = (obj1, obj2) => JSON.stringify(obj1) === JSON.stringify(obj2)\n\nexport let isEmpty = (obj) => {\n  for(let x in obj){ return false }\n  return true\n}\n\nexport let maybe = (el, callback) => el && callback(el)\n\nexport let channelUploader = function (entries, onError, resp, liveSocket){\n  entries.forEach(entry => {\n    let entryUploader = new EntryUploader(entry, resp.config, liveSocket)\n    entryUploader.upload()\n  })\n}\n", "let Browser = {\n  canPushState(){ return (typeof (history.pushState) !== \"undefined\") },\n\n  dropLocal(localStorage, namespace, subkey){\n    return localStorage.removeItem(this.localKey(namespace, subkey))\n  },\n\n  updateLocal(localStorage, namespace, subkey, initial, func){\n    let current = this.getLocal(localStorage, namespace, subkey)\n    let key = this.localKey(namespace, subkey)\n    let newVal = current === null ? initial : func(current)\n    localStorage.setItem(key, JSON.stringify(newVal))\n    return newVal\n  },\n\n  getLocal(localStorage, namespace, subkey){\n    return JSON.parse(localStorage.getItem(this.localKey(namespace, subkey)))\n  },\n\n  updateCurrentState(callback){\n    if(!this.canPushState()){ return }\n    history.replaceState(callback(history.state || {}), \"\", window.location.href)\n  },\n\n  pushState(kind, meta, to){\n    if(this.canPushState()){\n      if(to !== window.location.href){\n        if(meta.type == \"redirect\" && meta.scroll){\n          // If we're redirecting store the current scrollY for the current history state.\n          let currentState = history.state || {}\n          currentState.scroll = meta.scroll\n          history.replaceState(currentState, \"\", window.location.href)\n        }\n\n        delete meta.scroll // Only store the scroll in the redirect case.\n        history[kind + \"State\"](meta, \"\", to || null) // IE will coerce undefined to string\n\n        // when using navigate, we'd call pushState immediately before patching the DOM,\n        // jumping back to the top of the page, effectively ignoring the scrollIntoView;\n        // therefore we wait for the next frame (after the DOM patch) and only then try\n        // to scroll to the hashEl\n        window.requestAnimationFrame(() => {\n          let hashEl = this.getHashTargetEl(window.location.hash)\n  \n          if(hashEl){\n            hashEl.scrollIntoView()\n          } else if(meta.type === \"redirect\"){\n            window.scroll(0, 0)\n          }\n        })\n      }\n    } else {\n      this.redirect(to)\n    }\n  },\n\n  setCookie(name, value, maxAgeSeconds){\n    let expires = typeof(maxAgeSeconds) === \"number\" ? ` max-age=${maxAgeSeconds};` : \"\"\n    document.cookie = `${name}=${value};${expires} path=/`\n  },\n\n  getCookie(name){\n    return document.cookie.replace(new RegExp(`(?:(?:^|.*;\\s*)${name}\\s*\\=\\s*([^;]*).*$)|^.*$`), \"$1\")\n  },\n\n  deleteCookie(name){\n    document.cookie = `${name}=; max-age=-1; path=/`\n  },\n\n  redirect(toURL, flash){\n    if(flash){ this.setCookie(\"__phoenix_flash__\", flash, 60) }\n    window.location = toURL\n  },\n\n  localKey(namespace, subkey){ return `${namespace}-${subkey}` },\n\n  getHashTargetEl(maybeHash){\n    let hash = maybeHash.toString().substring(1)\n    if(hash === \"\"){ return }\n    return document.getElementById(hash) || document.querySelector(`a[name=\"${hash}\"]`)\n  }\n}\n\nexport default Browser\n", "import {\n  CHECKABLE_INPUTS,\n  DEBOUNCE_PREV_KEY,\n  DEBOUNCE_TRIGGER,\n  FOCUSABLE_INPUTS,\n  PHX_COMPONENT,\n  PHX_HAS_FOCUSED,\n  PHX_HAS_SUBMITTED,\n  PHX_MAIN,\n  PHX_PARENT_ID,\n  PHX_PRIVATE,\n  PHX_REF_SRC,\n  PHX_REF_LOCK,\n  PHX_PENDING_ATTRS,\n  PHX_ROOT_ID,\n  PHX_SESSION,\n  PHX_STATIC,\n  PHX_UPLOAD_REF,\n  PHX_VIEW_SELECTOR,\n  PHX_STICKY,\n  PHX_EVENT_CLASSES,\n  THROTTLED,\n  PHX_STREAM,\n} from \"./constants\"\n\nimport {\n  logError\n} from \"./utils\"\n\nlet DOM = {\n  byId(id){ return document.getElementById(id) || logError(`no id found for ${id}`) },\n\n  removeClass(el, className){\n    el.classList.remove(className)\n    if(el.classList.length === 0){ el.removeAttribute(\"class\") }\n  },\n\n  all(node, query, callback){\n    if(!node){ return [] }\n    let array = Array.from(node.querySelectorAll(query))\n    return callback ? array.forEach(callback) : array\n  },\n\n  childNodeLength(html){\n    let template = document.createElement(\"template\")\n    template.innerHTML = html\n    return template.content.childElementCount\n  },\n\n  isUploadInput(el){ return el.type === \"file\" && el.getAttribute(PHX_UPLOAD_REF) !== null },\n\n  isAutoUpload(inputEl){ return inputEl.hasAttribute(\"data-phx-auto-upload\") },\n\n  findUploadInputs(node){\n    const formId = node.id\n    const inputsOutsideForm = this.all(document, `input[type=\"file\"][${PHX_UPLOAD_REF}][form=\"${formId}\"]`)\n    return this.all(node, `input[type=\"file\"][${PHX_UPLOAD_REF}]`).concat(inputsOutsideForm)\n  },\n\n  findComponentNodeList(node, cid){\n    return this.filterWithinSameLiveView(this.all(node, `[${PHX_COMPONENT}=\"${cid}\"]`), node)\n  },\n\n  isPhxDestroyed(node){\n    return node.id && DOM.private(node, \"destroyed\") ? true : false\n  },\n\n  wantsNewTab(e){\n    let wantsNewTab = e.ctrlKey || e.shiftKey || e.metaKey || (e.button && e.button === 1)\n    let isDownload = (e.target instanceof HTMLAnchorElement && e.target.hasAttribute(\"download\"))\n    let isTargetBlank = e.target.hasAttribute(\"target\") && e.target.getAttribute(\"target\").toLowerCase() === \"_blank\"\n    let isTargetNamedTab = e.target.hasAttribute(\"target\") && !e.target.getAttribute(\"target\").startsWith(\"_\")\n    return wantsNewTab || isTargetBlank || isDownload || isTargetNamedTab\n  },\n\n  isUnloadableFormSubmit(e){\n    // Ignore form submissions intended to close a native <dialog> element\n    // https://developer.mozilla.org/en-US/docs/Web/HTML/Element/dialog#usage_notes\n    let isDialogSubmit = (e.target && e.target.getAttribute(\"method\") === \"dialog\") ||\n      (e.submitter && e.submitter.getAttribute(\"formmethod\") === \"dialog\")\n\n    if(isDialogSubmit){\n      return false\n    } else {\n      return !e.defaultPrevented && !this.wantsNewTab(e)\n    }\n  },\n\n  isNewPageClick(e, currentLocation){\n    let href = e.target instanceof HTMLAnchorElement ? e.target.getAttribute(\"href\") : null\n    let url\n\n    if(e.defaultPrevented || href === null || this.wantsNewTab(e)){ return false }\n    if(href.startsWith(\"mailto:\") || href.startsWith(\"tel:\")){ return false }\n    if(e.target.isContentEditable){ return false }\n\n    try {\n      url = new URL(href)\n    } catch {\n      try {\n        url = new URL(href, currentLocation)\n      } catch {\n        // bad URL, fallback to let browser try it as external\n        return true\n      }\n    }\n\n    if(url.host === currentLocation.host && url.protocol === currentLocation.protocol){\n      if(url.pathname === currentLocation.pathname && url.search === currentLocation.search){\n        return url.hash === \"\" && !url.href.endsWith(\"#\")\n      }\n    }\n    return url.protocol.startsWith(\"http\")\n  },\n\n  markPhxChildDestroyed(el){\n    if(this.isPhxChild(el)){ el.setAttribute(PHX_SESSION, \"\") }\n    this.putPrivate(el, \"destroyed\", true)\n  },\n\n  findPhxChildrenInFragment(html, parentId){\n    let template = document.createElement(\"template\")\n    template.innerHTML = html\n    return this.findPhxChildren(template.content, parentId)\n  },\n\n  isIgnored(el, phxUpdate){\n    return (el.getAttribute(phxUpdate) || el.getAttribute(\"data-phx-update\")) === \"ignore\"\n  },\n\n  isPhxUpdate(el, phxUpdate, updateTypes){\n    return el.getAttribute && updateTypes.indexOf(el.getAttribute(phxUpdate)) >= 0\n  },\n\n  findPhxSticky(el){ return this.all(el, `[${PHX_STICKY}]`) },\n\n  findPhxChildren(el, parentId){\n    return this.all(el, `${PHX_VIEW_SELECTOR}[${PHX_PARENT_ID}=\"${parentId}\"]`)\n  },\n\n  findExistingParentCIDs(node, cids){\n    // we only want to find parents that exist on the page\n    // if a cid is not on the page, the only way it can be added back to the page\n    // is if a parent adds it back, therefore if a cid does not exist on the page,\n    // we should not try to render it by itself (because it would be rendered twice,\n    // one by the parent, and a second time by itself)\n    let parentCids = new Set()\n    let childrenCids = new Set()\n\n    cids.forEach(cid => {\n      this.filterWithinSameLiveView(this.all(node, `[${PHX_COMPONENT}=\"${cid}\"]`), node).forEach(parent => {\n        parentCids.add(cid)\n        this.filterWithinSameLiveView(this.all(parent, `[${PHX_COMPONENT}]`), parent)\n          .map(el => parseInt(el.getAttribute(PHX_COMPONENT)))\n          .forEach(childCID => childrenCids.add(childCID))\n      })\n    })\n\n    childrenCids.forEach(childCid => parentCids.delete(childCid))\n\n    return parentCids\n  },\n\n  filterWithinSameLiveView(nodes, parent){\n    if(parent.querySelector(PHX_VIEW_SELECTOR)){\n      return nodes.filter(el => this.withinSameLiveView(el, parent))\n    } else {\n      return nodes\n    }\n  },\n\n  withinSameLiveView(node, parent){\n    while(node = node.parentNode){\n      if(node.isSameNode(parent)){ return true }\n      if(node.getAttribute(PHX_SESSION) !== null){ return false }\n    }\n  },\n\n  private(el, key){ return el[PHX_PRIVATE] && el[PHX_PRIVATE][key] },\n\n  deletePrivate(el, key){ el[PHX_PRIVATE] && delete (el[PHX_PRIVATE][key]) },\n\n  putPrivate(el, key, value){\n    if(!el[PHX_PRIVATE]){ el[PHX_PRIVATE] = {} }\n    el[PHX_PRIVATE][key] = value\n  },\n\n  updatePrivate(el, key, defaultVal, updateFunc){\n    let existing = this.private(el, key)\n    if(existing === undefined){\n      this.putPrivate(el, key, updateFunc(defaultVal))\n    } else {\n      this.putPrivate(el, key, updateFunc(existing))\n    }\n  },\n\n  syncPendingAttrs(fromEl, toEl){\n    if(!fromEl.hasAttribute(PHX_REF_SRC)){ return }\n    PHX_EVENT_CLASSES.forEach(className => {\n      fromEl.classList.contains(className) && toEl.classList.add(className)\n    })\n    PHX_PENDING_ATTRS.filter(attr => fromEl.hasAttribute(attr)).forEach(attr => {\n      toEl.setAttribute(attr, fromEl.getAttribute(attr))\n    })\n  },\n\n  copyPrivates(target, source){\n    if(source[PHX_PRIVATE]){\n      target[PHX_PRIVATE] = source[PHX_PRIVATE]\n    }\n  },\n\n  putTitle(str){\n    let titleEl = document.querySelector(\"title\")\n    if(titleEl){\n      let {prefix, suffix, default: defaultTitle} = titleEl.dataset\n      let isEmpty = typeof(str) !== \"string\" || str.trim() === \"\"\n      if(isEmpty && typeof(defaultTitle) !== \"string\"){ return }\n\n      let inner = isEmpty ? defaultTitle : str\n      document.title = `${prefix || \"\"}${inner || \"\"}${suffix || \"\"}`\n    } else {\n      document.title = str\n    }\n  },\n\n  debounce(el, event, phxDebounce, defaultDebounce, phxThrottle, defaultThrottle, asyncFilter, callback){\n    let debounce = el.getAttribute(phxDebounce)\n    let throttle = el.getAttribute(phxThrottle)\n\n    if(debounce === \"\"){ debounce = defaultDebounce }\n    if(throttle === \"\"){ throttle = defaultThrottle }\n    let value = debounce || throttle\n    switch(value){\n      case null: return callback()\n\n      case \"blur\":\n        this.incCycle(el, \"debounce-blur-cycle\", () => {\n          if(asyncFilter()){ callback() }\n        })\n        if(this.once(el, \"debounce-blur\")){\n          el.addEventListener(\"blur\", () => this.triggerCycle(el, \"debounce-blur-cycle\"))\n        }\n        return\n\n      default:\n        let timeout = parseInt(value)\n        let trigger = () => throttle ? this.deletePrivate(el, THROTTLED) : callback()\n        let currentCycle = this.incCycle(el, DEBOUNCE_TRIGGER, trigger)\n        if(isNaN(timeout)){ return logError(`invalid throttle/debounce value: ${value}`) }\n        if(throttle){\n          let newKeyDown = false\n          if(event.type === \"keydown\"){\n            let prevKey = this.private(el, DEBOUNCE_PREV_KEY)\n            this.putPrivate(el, DEBOUNCE_PREV_KEY, event.key)\n            newKeyDown = prevKey !== event.key\n          }\n\n          if(!newKeyDown && this.private(el, THROTTLED)){\n            return false\n          } else {\n            callback()\n            const t = setTimeout(() => {\n              if(asyncFilter()){ this.triggerCycle(el, DEBOUNCE_TRIGGER) }\n            }, timeout)\n            this.putPrivate(el, THROTTLED, t)\n          }\n        } else {\n          setTimeout(() => {\n            if(asyncFilter()){ this.triggerCycle(el, DEBOUNCE_TRIGGER, currentCycle) }\n          }, timeout)\n        }\n\n        let form = el.form\n        if(form && this.once(form, \"bind-debounce\")){\n          form.addEventListener(\"submit\", () => {\n            Array.from((new FormData(form)).entries(), ([name]) => {\n              let input = form.querySelector(`[name=\"${name}\"]`)\n              this.incCycle(input, DEBOUNCE_TRIGGER)\n              this.deletePrivate(input, THROTTLED)\n            })\n          })\n        }\n        if(this.once(el, \"bind-debounce\")){\n          el.addEventListener(\"blur\", () => {\n            // because we trigger the callback here,\n            // we also clear the throttle timeout to prevent the callback\n            // from being called again after the timeout fires\n            clearTimeout(this.private(el, THROTTLED))\n            this.triggerCycle(el, DEBOUNCE_TRIGGER)\n          })\n        }\n    }\n  },\n\n  triggerCycle(el, key, currentCycle){\n    let [cycle, trigger] = this.private(el, key)\n    if(!currentCycle){ currentCycle = cycle }\n    if(currentCycle === cycle){\n      this.incCycle(el, key)\n      trigger()\n    }\n  },\n\n  once(el, key){\n    if(this.private(el, key) === true){ return false }\n    this.putPrivate(el, key, true)\n    return true\n  },\n\n  incCycle(el, key, trigger = function (){ }){\n    let [currentCycle] = this.private(el, key) || [0, trigger]\n    currentCycle++\n    this.putPrivate(el, key, [currentCycle, trigger])\n    return currentCycle\n  },\n\n  // maintains or adds privately used hook information\n  // fromEl and toEl can be the same element in the case of a newly added node\n  // fromEl and toEl can be any HTML node type, so we need to check if it's an element node\n  maintainPrivateHooks(fromEl, toEl, phxViewportTop, phxViewportBottom){\n    // maintain the hooks created with createHook\n    if(fromEl.hasAttribute && fromEl.hasAttribute(\"data-phx-hook\") && !toEl.hasAttribute(\"data-phx-hook\")){\n      toEl.setAttribute(\"data-phx-hook\", fromEl.getAttribute(\"data-phx-hook\"))\n    }\n    // add hooks to elements with viewport attributes\n    if(toEl.hasAttribute && (toEl.hasAttribute(phxViewportTop) || toEl.hasAttribute(phxViewportBottom))){\n      toEl.setAttribute(\"data-phx-hook\", \"Phoenix.InfiniteScroll\")\n    }\n  },\n\n  putCustomElHook(el, hook){\n    if(el.isConnected){\n      el.setAttribute(\"data-phx-hook\", \"\")\n    } else {\n      console.error(`\n        hook attached to non-connected DOM element\n        ensure you are calling createHook within your connectedCallback. ${el.outerHTML}\n      `)\n    }\n    this.putPrivate(el, \"custom-el-hook\", hook)\n  },\n\n  getCustomElHook(el){ return this.private(el, \"custom-el-hook\") },\n\n  isUsedInput(el){\n    return (el.nodeType === Node.ELEMENT_NODE &&\n      (this.private(el, PHX_HAS_FOCUSED) || this.private(el, PHX_HAS_SUBMITTED)))\n  },\n\n  resetForm(form){\n    Array.from(form.elements).forEach(input => {\n      this.deletePrivate(input, PHX_HAS_FOCUSED)\n      this.deletePrivate(input, PHX_HAS_SUBMITTED)\n    })\n  },\n\n  isPhxChild(node){\n    return node.getAttribute && node.getAttribute(PHX_PARENT_ID)\n  },\n\n  isPhxSticky(node){\n    return node.getAttribute && node.getAttribute(PHX_STICKY) !== null\n  },\n\n  isChildOfAny(el, parents){\n    return !!parents.find(parent => parent.contains(el))\n  },\n\n  firstPhxChild(el){\n    return this.isPhxChild(el) ? el : this.all(el, `[${PHX_PARENT_ID}]`)[0]\n  },\n\n  dispatchEvent(target, name, opts = {}){\n    let defaultBubble = true\n    let isUploadTarget = target.nodeName === \"INPUT\" && target.type === \"file\"\n    if(isUploadTarget && name === \"click\"){\n      defaultBubble = false\n    }\n    let bubbles = opts.bubbles === undefined ? defaultBubble : !!opts.bubbles\n    let eventOpts = {bubbles: bubbles, cancelable: true, detail: opts.detail || {}}\n    let event = name === \"click\" ? new MouseEvent(\"click\", eventOpts) : new CustomEvent(name, eventOpts)\n    target.dispatchEvent(event)\n  },\n\n  cloneNode(node, html){\n    if(typeof (html) === \"undefined\"){\n      return node.cloneNode(true)\n    } else {\n      let cloned = node.cloneNode(false)\n      cloned.innerHTML = html\n      return cloned\n    }\n  },\n\n  // merge attributes from source to target\n  // if an element is ignored, we only merge data attributes\n  // including removing data attributes that are no longer in the source\n  mergeAttrs(target, source, opts = {}){\n    let exclude = new Set(opts.exclude || [])\n    let isIgnored = opts.isIgnored\n    let sourceAttrs = source.attributes\n    for(let i = sourceAttrs.length - 1; i >= 0; i--){\n      let name = sourceAttrs[i].name\n      if(!exclude.has(name)){\n        const sourceValue = source.getAttribute(name)\n        if(target.getAttribute(name) !== sourceValue && (!isIgnored || (isIgnored && name.startsWith(\"data-\")))){\n          target.setAttribute(name, sourceValue)\n        }\n      } else {\n        // We exclude the value from being merged on focused inputs, because the\n        // user's input should always win.\n        // We can still assign it as long as the value property is the same, though.\n        // This prevents a situation where the updated hook is not being triggered\n        // when an input is back in its \"original state\", because the attribute\n        // was never changed, see:\n        // https://github.com/phoenixframework/phoenix_live_view/issues/2163\n        if(name === \"value\" && target.value === source.value){\n          // actually set the value attribute to sync it with the value property\n          target.setAttribute(\"value\", source.getAttribute(name))\n        }\n      }\n    }\n\n    let targetAttrs = target.attributes\n    for(let i = targetAttrs.length - 1; i >= 0; i--){\n      let name = targetAttrs[i].name\n      if(isIgnored){\n        if(name.startsWith(\"data-\") && !source.hasAttribute(name) && !PHX_PENDING_ATTRS.includes(name)){ target.removeAttribute(name) }\n      } else {\n        if(!source.hasAttribute(name)){ target.removeAttribute(name) }\n      }\n    }\n  },\n\n  mergeFocusedInput(target, source){\n    // skip selects because FF will reset highlighted index for any setAttribute\n    if(!(target instanceof HTMLSelectElement)){ DOM.mergeAttrs(target, source, {exclude: [\"value\"]}) }\n\n    if(source.readOnly){\n      target.setAttribute(\"readonly\", true)\n    } else {\n      target.removeAttribute(\"readonly\")\n    }\n  },\n\n  hasSelectionRange(el){\n    return el.setSelectionRange && (el.type === \"text\" || el.type === \"textarea\")\n  },\n\n  restoreFocus(focused, selectionStart, selectionEnd){\n    if(focused instanceof HTMLSelectElement){ focused.focus() }\n    if(!DOM.isTextualInput(focused)){ return }\n\n    let wasFocused = focused.matches(\":focus\")\n    if(!wasFocused){ focused.focus() }\n    if(this.hasSelectionRange(focused)){\n      focused.setSelectionRange(selectionStart, selectionEnd)\n    }\n  },\n\n  isFormInput(el){ return /^(?:input|select|textarea)$/i.test(el.tagName) && el.type !== \"button\" },\n\n  syncAttrsToProps(el){\n    if(el instanceof HTMLInputElement && CHECKABLE_INPUTS.indexOf(el.type.toLocaleLowerCase()) >= 0){\n      el.checked = el.getAttribute(\"checked\") !== null\n    }\n  },\n\n  isTextualInput(el){ return FOCUSABLE_INPUTS.indexOf(el.type) >= 0 },\n\n  isNowTriggerFormExternal(el, phxTriggerExternal){\n    return el.getAttribute && el.getAttribute(phxTriggerExternal) !== null && document.body.contains(el)\n  },\n\n  cleanChildNodes(container, phxUpdate){\n    if(DOM.isPhxUpdate(container, phxUpdate, [\"append\", \"prepend\", PHX_STREAM])){\n      let toRemove = []\n      container.childNodes.forEach(childNode => {\n        if(!childNode.id){\n          // Skip warning if it's an empty text node (e.g. a new-line)\n          let isEmptyTextNode = childNode.nodeType === Node.TEXT_NODE && childNode.nodeValue.trim() === \"\"\n          if(!isEmptyTextNode && childNode.nodeType !== Node.COMMENT_NODE){\n            logError(\"only HTML element tags with an id are allowed inside containers with phx-update.\\n\\n\" +\n              `removing illegal node: \"${(childNode.outerHTML || childNode.nodeValue).trim()}\"\\n\\n`)\n          }\n          toRemove.push(childNode)\n        }\n      })\n      toRemove.forEach(childNode => childNode.remove())\n    }\n  },\n\n  replaceRootContainer(container, tagName, attrs){\n    let retainedAttrs = new Set([\"id\", PHX_SESSION, PHX_STATIC, PHX_MAIN, PHX_ROOT_ID])\n    if(container.tagName.toLowerCase() === tagName.toLowerCase()){\n      Array.from(container.attributes)\n        .filter(attr => !retainedAttrs.has(attr.name.toLowerCase()))\n        .forEach(attr => container.removeAttribute(attr.name))\n\n      Object.keys(attrs)\n        .filter(name => !retainedAttrs.has(name.toLowerCase()))\n        .forEach(attr => container.setAttribute(attr, attrs[attr]))\n\n      return container\n\n    } else {\n      let newContainer = document.createElement(tagName)\n      Object.keys(attrs).forEach(attr => newContainer.setAttribute(attr, attrs[attr]))\n      retainedAttrs.forEach(attr => newContainer.setAttribute(attr, container.getAttribute(attr)))\n      newContainer.innerHTML = container.innerHTML\n      container.replaceWith(newContainer)\n      return newContainer\n    }\n  },\n\n  getSticky(el, name, defaultVal){\n    let op = (DOM.private(el, \"sticky\") || []).find(([existingName,]) => name === existingName)\n    if(op){\n      let [_name, _op, stashedResult] = op\n      return stashedResult\n    } else {\n      return typeof(defaultVal) === \"function\" ? defaultVal() : defaultVal\n    }\n  },\n\n  deleteSticky(el, name){\n    this.updatePrivate(el, \"sticky\", [], ops => {\n      return ops.filter(([existingName, _]) => existingName !== name)\n    })\n  },\n\n  putSticky(el, name, op){\n    let stashedResult = op(el)\n    this.updatePrivate(el, \"sticky\", [], ops => {\n      let existingIndex = ops.findIndex(([existingName,]) => name === existingName)\n      if(existingIndex >= 0){\n        ops[existingIndex] = [name, op, stashedResult]\n      } else {\n        ops.push([name, op, stashedResult])\n      }\n      return ops\n    })\n  },\n\n  applyStickyOperations(el){\n    let ops = DOM.private(el, \"sticky\")\n    if(!ops){ return }\n\n    ops.forEach(([name, op, _stashed]) => this.putSticky(el, name, op))\n  },\n\n  isLocked(el){\n    return el.hasAttribute && el.hasAttribute(PHX_REF_LOCK)\n  }\n}\n\nexport default DOM\n", "import {\n  PHX_ACTIVE_ENTRY_REFS,\n  PHX_LIVE_FILE_UPDATED,\n  PHX_PREFLIGHTED_REFS\n} from \"./constants\"\n\nimport {\n  channelUploader,\n  logError\n} from \"./utils\"\n\nimport LiveUploader from \"./live_uploader\"\n\nexport default class UploadEntry {\n  static isActive(fileEl, file){\n    let isNew = file._phxRef === undefined\n    let activeRefs = fileEl.getAttribute(PHX_ACTIVE_ENTRY_REFS).split(\",\")\n    let isActive = activeRefs.indexOf(LiveUploader.genFileRef(file)) >= 0\n    return file.size > 0 && (isNew || isActive)\n  }\n\n  static isPreflighted(fileEl, file){\n    let preflightedRefs = fileEl.getAttribute(PHX_PREFLIGHTED_REFS).split(\",\")\n    let isPreflighted = preflightedRefs.indexOf(LiveUploader.genFileRef(file)) >= 0\n    return isPreflighted && this.isActive(fileEl, file)\n  }\n\n  static isPreflightInProgress(file){\n    return file._preflightInProgress === true\n  }\n\n  static markPreflightInProgress(file){\n    file._preflightInProgress = true\n  }\n\n  constructor(fileEl, file, view, autoUpload){\n    this.ref = LiveUploader.genFileRef(file)\n    this.fileEl = fileEl\n    this.file = file\n    this.view = view\n    this.meta = null\n    this._isCancelled = false\n    this._isDone = false\n    this._progress = 0\n    this._lastProgressSent = -1\n    this._onDone = function(){ }\n    this._onElUpdated = this.onElUpdated.bind(this)\n    this.fileEl.addEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated)\n    this.autoUpload = autoUpload\n  }\n\n  metadata(){ return this.meta }\n\n  progress(progress){\n    this._progress = Math.floor(progress)\n    if(this._progress > this._lastProgressSent){\n      if(this._progress >= 100){\n        this._progress = 100\n        this._lastProgressSent = 100\n        this._isDone = true\n        this.view.pushFileProgress(this.fileEl, this.ref, 100, () => {\n          LiveUploader.untrackFile(this.fileEl, this.file)\n          this._onDone()\n        })\n      } else {\n        this._lastProgressSent = this._progress\n        this.view.pushFileProgress(this.fileEl, this.ref, this._progress)\n      }\n    }\n  }\n\n  isCancelled(){ return this._isCancelled }\n\n  cancel(){\n    this.file._preflightInProgress = false\n    this._isCancelled = true\n    this._isDone = true\n    this._onDone()\n  }\n\n  isDone(){ return this._isDone }\n\n  error(reason = \"failed\"){\n    this.fileEl.removeEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated)\n    this.view.pushFileProgress(this.fileEl, this.ref, {error: reason})\n    if(!this.isAutoUpload()){ LiveUploader.clearFiles(this.fileEl) }\n  }\n\n  isAutoUpload(){ return this.autoUpload }\n\n  //private\n\n  onDone(callback){\n    this._onDone = () => {\n      this.fileEl.removeEventListener(PHX_LIVE_FILE_UPDATED, this._onElUpdated)\n      callback()\n    }\n  }\n\n  onElUpdated(){\n    let activeRefs = this.fileEl.getAttribute(PHX_ACTIVE_ENTRY_REFS).split(\",\")\n    if(activeRefs.indexOf(this.ref) === -1){\n      LiveUploader.untrackFile(this.fileEl, this.file)\n      this.cancel()\n    }\n  }\n\n  toPreflightPayload(){\n    return {\n      last_modified: this.file.lastModified,\n      name: this.file.name,\n      relative_path: this.file.webkitRelativePath,\n      size: this.file.size,\n      type: this.file.type,\n      ref: this.ref,\n      meta: typeof(this.file.meta) === \"function\" ? this.file.meta() : undefined\n    }\n  }\n\n  uploader(uploaders){\n    if(this.meta.uploader){\n      let callback = uploaders[this.meta.uploader] || logError(`no uploader configured for ${this.meta.uploader}`)\n      return {name: this.meta.uploader, callback: callback}\n    } else {\n      return {name: \"channel\", callback: channelUploader}\n    }\n  }\n\n  zipPostFlight(resp){\n    this.meta = resp.entries[this.ref]\n    if(!this.meta){ logError(`no preflight upload response returned with ref ${this.ref}`, {input: this.fileEl, response: resp}) }\n  }\n}\n", "import {\n  PHX_DONE_REFS,\n  PHX_PREFLIGHTED_REFS,\n  PHX_UPLOAD_REF\n} from \"./constants\"\n\nimport {\n} from \"./utils\"\n\nimport DOM from \"./dom\"\nimport UploadEntry from \"./upload_entry\"\n\nlet liveUploaderFileRef = 0\n\nexport default class LiveUploader {\n  static genFileRef(file){\n    let ref = file._phxRef\n    if(ref !== undefined){\n      return ref\n    } else {\n      file._phxRef = (liveUploaderFileRef++).toString()\n      return file._phxRef\n    }\n  }\n\n  static getEntryDataURL(inputEl, ref, callback){\n    let file = this.activeFiles(inputEl).find(file => this.genFileRef(file) === ref)\n    callback(URL.createObjectURL(file))\n  }\n\n  static hasUploadsInProgress(formEl){\n    let active = 0\n    DOM.findUploadInputs(formEl).forEach(input => {\n      if(input.getAttribute(PHX_PREFLIGHTED_REFS) !== input.getAttribute(PHX_DONE_REFS)){\n        active++\n      }\n    })\n    return active > 0\n  }\n\n  static serializeUploads(inputEl){\n    let files = this.activeFiles(inputEl)\n    let fileData = {}\n    files.forEach(file => {\n      let entry = {path: inputEl.name}\n      let uploadRef = inputEl.getAttribute(PHX_UPLOAD_REF)\n      fileData[uploadRef] = fileData[uploadRef] || []\n      entry.ref = this.genFileRef(file)\n      entry.last_modified = file.lastModified\n      entry.name = file.name || entry.ref\n      entry.relative_path = file.webkitRelativePath\n      entry.type = file.type\n      entry.size = file.size\n      if(typeof(file.meta) === \"function\"){ entry.meta = file.meta() }\n      fileData[uploadRef].push(entry)\n    })\n    return fileData\n  }\n\n  static clearFiles(inputEl){\n    inputEl.value = null\n    inputEl.removeAttribute(PHX_UPLOAD_REF)\n    DOM.putPrivate(inputEl, \"files\", [])\n  }\n\n  static untrackFile(inputEl, file){\n    DOM.putPrivate(inputEl, \"files\", DOM.private(inputEl, \"files\").filter(f => !Object.is(f, file)))\n  }\n\n  static trackFiles(inputEl, files, dataTransfer){\n    if(inputEl.getAttribute(\"multiple\") !== null){\n      let newFiles = files.filter(file => !this.activeFiles(inputEl).find(f => Object.is(f, file)))\n      DOM.updatePrivate(inputEl, \"files\", [], (existing) => existing.concat(newFiles))\n      inputEl.value = null\n    } else {\n      // Reset inputEl files to align output with programmatic changes (i.e. drag and drop)\n      if(dataTransfer && dataTransfer.files.length > 0){ inputEl.files = dataTransfer.files }\n      DOM.putPrivate(inputEl, \"files\", files)\n    }\n  }\n\n  static activeFileInputs(formEl){\n    let fileInputs = DOM.findUploadInputs(formEl)\n    return Array.from(fileInputs).filter(el => el.files && this.activeFiles(el).length > 0)\n  }\n\n  static activeFiles(input){\n    return (DOM.private(input, \"files\") || []).filter(f => UploadEntry.isActive(input, f))\n  }\n\n  static inputsAwaitingPreflight(formEl){\n    let fileInputs = DOM.findUploadInputs(formEl)\n    return Array.from(fileInputs).filter(input => this.filesAwaitingPreflight(input).length > 0)\n  }\n\n  static filesAwaitingPreflight(input){\n    return this.activeFiles(input).filter(f => !UploadEntry.isPreflighted(input, f) && !UploadEntry.isPreflightInProgress(f))\n  }\n\n  static markPreflightInProgress(entries){\n    entries.forEach(entry => UploadEntry.markPreflightInProgress(entry.file))\n  }\n\n  constructor(inputEl, view, onComplete){\n    this.autoUpload = DOM.isAutoUpload(inputEl)\n    this.view = view\n    this.onComplete = onComplete\n    this._entries =\n      Array.from(LiveUploader.filesAwaitingPreflight(inputEl) || [])\n        .map(file => new UploadEntry(inputEl, file, view, this.autoUpload))\n\n    // prevent sending duplicate preflight requests\n    LiveUploader.markPreflightInProgress(this._entries)\n\n    this.numEntriesInProgress = this._entries.length\n  }\n\n  isAutoUpload(){ return this.autoUpload }\n\n  entries(){ return this._entries }\n\n  initAdapterUpload(resp, onError, liveSocket){\n    this._entries =\n      this._entries.map(entry => {\n        if(entry.isCancelled()){\n          this.numEntriesInProgress--\n          if(this.numEntriesInProgress === 0){ this.onComplete() }\n        } else {\n          entry.zipPostFlight(resp)\n          entry.onDone(() => {\n            this.numEntriesInProgress--\n            if(this.numEntriesInProgress === 0){ this.onComplete() }\n          })\n        }\n        return entry\n      })\n\n    let groupedEntries = this._entries.reduce((acc, entry) => {\n      if(!entry.meta){ return acc }\n      let {name, callback} = entry.uploader(liveSocket.uploaders)\n      acc[name] = acc[name] || {callback: callback, entries: []}\n      acc[name].entries.push(entry)\n      return acc\n    }, {})\n\n    for(let name in groupedEntries){\n      let {callback, entries} = groupedEntries[name]\n      callback(entries, onError, resp, liveSocket)\n    }\n  }\n}\n", "let ARIA = {\n  anyOf(instance, classes){ return classes.find(name => instance instanceof name) },\n\n  isFocusable(el, interactiveOnly){\n    return (\n      (el instanceof HTMLAnchorElement && el.rel !== \"ignore\") ||\n      (el instanceof HTMLAreaElement && el.href !== undefined) ||\n      (!el.disabled && (this.anyOf(el, [HTMLInputElement, HTMLSelectElement, HTMLTextAreaElement, HTMLButtonElement]))) ||\n      (el instanceof HTMLIFrameElement) ||\n      ((el.tabIndex >= 0 && el.getAttribute(\"aria-hidden\") !== \"true\") || (!interactiveOnly && el.getAttribute(\"tabindex\") !== null && el.getAttribute(\"aria-hidden\") !== \"true\"))\n    )\n  },\n\n  attemptFocus(el, interactiveOnly){\n    if(this.isFocusable(el, interactiveOnly)){ try { el.focus() } catch {} }\n    return !!document.activeElement && document.activeElement.isSameNode(el)\n  },\n\n  focusFirstInteractive(el){\n    let child = el.firstElementChild\n    while(child){\n      if(this.attemptFocus(child, true) || this.focusFirstInteractive(child, true)){\n        return true\n      }\n      child = child.nextElementSibling\n    }\n  },\n\n  focusFirst(el){\n    let child = el.firstElementChild\n    while(child){\n      if(this.attemptFocus(child) || this.focusFirst(child)){\n        return true\n      }\n      child = child.nextElementSibling\n    }\n  },\n\n  focusLast(el){\n    let child = el.lastElementChild\n    while(child){\n      if(this.attemptFocus(child) || this.focusLast(child)){\n        return true\n      }\n      child = child.previousElementSibling\n    }\n  }\n}\nexport default ARIA\n", "import {\n  PHX_ACTIVE_ENTRY_REFS,\n  PHX_LIVE_FILE_UPDATED,\n  PHX_PREFLIGHTED_REFS,\n  PHX_UPLOAD_REF\n} from \"./constants\"\n\nimport LiveUploader from \"./live_uploader\"\nimport ARIA from \"./aria\"\n\nlet Hooks = {\n  LiveFileUpload: {\n    activeRefs(){ return this.el.getAttribute(PHX_ACTIVE_ENTRY_REFS) },\n\n    preflightedRefs(){ return this.el.getAttribute(PHX_PREFLIGHTED_REFS) },\n\n    mounted(){ this.preflightedWas = this.preflightedRefs() },\n\n    updated(){\n      let newPreflights = this.preflightedRefs()\n      if(this.preflightedWas !== newPreflights){\n        this.preflightedWas = newPreflights\n        if(newPreflights === \"\"){\n          this.__view().cancelSubmit(this.el.form)\n        }\n      }\n\n      if(this.activeRefs() === \"\"){ this.el.value = null }\n      this.el.dispatchEvent(new CustomEvent(PHX_LIVE_FILE_UPDATED))\n    }\n  },\n\n  LiveImgPreview: {\n    mounted(){\n      this.ref = this.el.getAttribute(\"data-phx-entry-ref\")\n      this.inputEl = document.getElementById(this.el.getAttribute(PHX_UPLOAD_REF))\n      LiveUploader.getEntryDataURL(this.inputEl, this.ref, url => {\n        this.url = url\n        this.el.src = url\n      })\n    },\n    destroyed(){\n      URL.revokeObjectURL(this.url)\n    }\n  },\n  FocusWrap: {\n    mounted(){\n      this.focusStart = this.el.firstElementChild\n      this.focusEnd = this.el.lastElementChild\n      this.focusStart.addEventListener(\"focus\", (e) => {\n        if(!e.relatedTarget || !this.el.contains(e.relatedTarget)){ \n          // Handle focus entering from outside (e.g. Tab when body is focused)\n          // https://github.com/phoenixframework/phoenix_live_view/issues/3636\n          const nextFocus = e.target.nextElementSibling\n          ARIA.attemptFocus(nextFocus) || ARIA.focusFirst(nextFocus)\n        } else {\n          ARIA.focusLast(this.el)\n        }\n      })\n      this.focusEnd.addEventListener(\"focus\", (e) => {\n        if(!e.relatedTarget || !this.el.contains(e.relatedTarget)){ \n          // Handle focus entering from outside (e.g. Shift+Tab when body is focused)\n          // https://github.com/phoenixframework/phoenix_live_view/issues/3636\n          const nextFocus = e.target.previousElementSibling\n          ARIA.attemptFocus(nextFocus) || ARIA.focusLast(nextFocus)\n        } else {\n          ARIA.focusFirst(this.el)\n        }\n      })\n      this.el.addEventListener(\"phx:show-end\", () => this.el.focus())\n      if(window.getComputedStyle(this.el).display !== \"none\"){\n        ARIA.focusFirst(this.el)\n      }\n    }\n  }\n}\n\nlet findScrollContainer = (el) => {\n  // the scroll event won't be fired on the html/body element even if overflow is set\n  // therefore we return null to instead listen for scroll events on document\n  if([\"HTML\", \"BODY\"].indexOf(el.nodeName.toUpperCase()) >= 0) return null\n  if([\"scroll\", \"auto\"].indexOf(getComputedStyle(el).overflowY) >= 0) return el\n  return findScrollContainer(el.parentElement)\n}\n\nlet scrollTop = (scrollContainer) => {\n  if(scrollContainer){\n    return scrollContainer.scrollTop\n  } else {\n    return document.documentElement.scrollTop || document.body.scrollTop\n  }\n}\n\nlet bottom = (scrollContainer) => {\n  if(scrollContainer){\n    return scrollContainer.getBoundingClientRect().bottom\n  } else {\n    // when we have no container, the whole page scrolls,\n    // therefore the bottom coordinate is the viewport height\n    return window.innerHeight || document.documentElement.clientHeight\n  }\n}\n\nlet top = (scrollContainer) => {\n  if(scrollContainer){\n    return scrollContainer.getBoundingClientRect().top\n  } else {\n    // when we have no container the whole page scrolls,\n    // therefore the top coordinate is 0\n    return 0\n  }\n}\n\nlet isAtViewportTop = (el, scrollContainer) => {\n  let rect = el.getBoundingClientRect()\n  return Math.ceil(rect.top) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.top) <= bottom(scrollContainer)\n}\n\nlet isAtViewportBottom = (el, scrollContainer) => {\n  let rect = el.getBoundingClientRect()\n  return Math.ceil(rect.bottom) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.bottom) <= bottom(scrollContainer)\n}\n\nlet isWithinViewport = (el, scrollContainer) => {\n  let rect = el.getBoundingClientRect()\n  return Math.ceil(rect.top) >= top(scrollContainer) && Math.ceil(rect.left) >= 0 && Math.floor(rect.top) <= bottom(scrollContainer)\n}\n\nHooks.InfiniteScroll = {\n  mounted(){\n    this.scrollContainer = findScrollContainer(this.el)\n    let scrollBefore = scrollTop(this.scrollContainer)\n    let topOverran = false\n    let throttleInterval = 500\n    let pendingOp = null\n\n    let onTopOverrun = this.throttle(throttleInterval, (topEvent, firstChild) => {\n      pendingOp = () => true\n      this.liveSocket.execJSHookPush(this.el, topEvent, {id: firstChild.id, _overran: true}, () => {\n        pendingOp = null\n      })\n    })\n\n    let onFirstChildAtTop = this.throttle(throttleInterval, (topEvent, firstChild) => {\n      pendingOp = () => firstChild.scrollIntoView({block: \"start\"})\n      this.liveSocket.execJSHookPush(this.el, topEvent, {id: firstChild.id}, () => {\n        pendingOp = null\n        // make sure that the DOM is patched by waiting for the next tick\n        window.requestAnimationFrame(() => {\n          if(!isWithinViewport(firstChild, this.scrollContainer)){\n            firstChild.scrollIntoView({block: \"start\"})\n          }\n        })\n      })\n    })\n\n    let onLastChildAtBottom = this.throttle(throttleInterval, (bottomEvent, lastChild) => {\n      pendingOp = () => lastChild.scrollIntoView({block: \"end\"})\n      this.liveSocket.execJSHookPush(this.el, bottomEvent, {id: lastChild.id}, () => {\n        pendingOp = null\n        // make sure that the DOM is patched by waiting for the next tick\n        window.requestAnimationFrame(() => {\n          if(!isWithinViewport(lastChild, this.scrollContainer)){\n            lastChild.scrollIntoView({block: \"end\"})\n          }\n        })\n      })\n    })\n\n    this.onScroll = (_e) => {\n      let scrollNow = scrollTop(this.scrollContainer)\n\n      if(pendingOp){\n        scrollBefore = scrollNow\n        return pendingOp()\n      }\n      let rect = this.el.getBoundingClientRect()\n      let topEvent = this.el.getAttribute(this.liveSocket.binding(\"viewport-top\"))\n      let bottomEvent = this.el.getAttribute(this.liveSocket.binding(\"viewport-bottom\"))\n      let lastChild = this.el.lastElementChild\n      let firstChild = this.el.firstElementChild\n      let isScrollingUp = scrollNow < scrollBefore\n      let isScrollingDown = scrollNow > scrollBefore\n\n      // el overran while scrolling up\n      if(isScrollingUp && topEvent && !topOverran && rect.top >= 0){\n        topOverran = true\n        onTopOverrun(topEvent, firstChild)\n      } else if(isScrollingDown && topOverran && rect.top <= 0){\n        topOverran = false\n      }\n\n      if(topEvent && isScrollingUp && isAtViewportTop(firstChild, this.scrollContainer)){\n        onFirstChildAtTop(topEvent, firstChild)\n      } else if(bottomEvent && isScrollingDown && isAtViewportBottom(lastChild, this.scrollContainer)){\n        onLastChildAtBottom(bottomEvent, lastChild)\n      }\n      scrollBefore = scrollNow\n    }\n\n    if(this.scrollContainer){\n      this.scrollContainer.addEventListener(\"scroll\", this.onScroll)\n    } else {\n      window.addEventListener(\"scroll\", this.onScroll)\n    }\n  },\n  \n  destroyed(){\n    if(this.scrollContainer){\n      this.scrollContainer.removeEventListener(\"scroll\", this.onScroll)\n    } else {\n      window.removeEventListener(\"scroll\", this.onScroll)\n    }\n  },\n\n  throttle(interval, callback){\n    let lastCallAt = 0\n    let timer\n\n    return (...args) => {\n      let now = Date.now()\n      let remainingTime = interval - (now - lastCallAt)\n\n      if(remainingTime <= 0 || remainingTime > interval){\n        if(timer){\n          clearTimeout(timer)\n          timer = null\n        }\n        lastCallAt = now\n        callback(...args)\n      } else if(!timer){\n        timer = setTimeout(() => {\n          lastCallAt = Date.now()\n          timer = null\n          callback(...args)\n        }, remainingTime)\n      }\n    }\n  }\n}\nexport default Hooks\n", "import {\n  PHX_REF_LOADING,\n  PHX_REF_LOCK,\n  PHX_REF_SRC,\n  PHX_PENDING_REFS,\n  PHX_EVENT_CLASSES,\n  PHX_DISABLED,\n  PHX_READONLY,\n  PHX_DISABLE_WITH_RESTORE\n} from \"./constants\"\n\nimport DOM from \"./dom\"\n\nexport default class ElementRef {\n  static onUnlock(el, callback){\n    if(!DOM.isLocked(el) && !el.closest(`[${PHX_REF_LOCK}]`)){ return callback() }\n    const closestLock = el.closest(`[${PHX_REF_LOCK}]`)\n    const ref = closestLock.closest(`[${PHX_REF_LOCK}]`).getAttribute(PHX_REF_LOCK)\n    closestLock.addEventListener(`phx:undo-lock:${ref}`, () => {\n      callback()\n    }, {once: true})\n  }\n\n  constructor(el){\n    this.el = el\n    this.loadingRef = el.hasAttribute(PHX_REF_LOADING) ? parseInt(el.getAttribute(PHX_REF_LOADING), 10) : null\n    this.lockRef = el.hasAttribute(PHX_REF_LOCK) ? parseInt(el.getAttribute(PHX_REF_LOCK), 10) : null\n  }\n\n  // public\n\n  maybeUndo(ref, phxEvent, eachCloneCallback){\n    if(!this.isWithin(ref)){\n      // we cannot undo the lock / loading now, as there is a newer one already set;\n      // we need to store the original ref we tried to send the undo event later\n      DOM.updatePrivate(this.el, PHX_PENDING_REFS, [], (pendingRefs) => {\n        pendingRefs.push(ref)\n        return pendingRefs\n      })\n      return\n    }\n\n    // undo locks and apply clones\n    this.undoLocks(ref, phxEvent, eachCloneCallback)\n\n    // undo loading states\n    this.undoLoading(ref, phxEvent)\n\n    // ensure undo events are fired for pending refs that\n    // are resolved by the current ref, otherwise we'd leak event listeners\n    DOM.updatePrivate(this.el, PHX_PENDING_REFS, [], (pendingRefs) => {\n      return pendingRefs.filter((pendingRef) => {\n        let opts = {\n          detail: {ref: pendingRef, event: phxEvent},\n          bubbles: true,\n          cancelable: false,\n        }\n        if(this.loadingRef && this.loadingRef > pendingRef){\n          this.el.dispatchEvent(\n            new CustomEvent(`phx:undo-loading:${pendingRef}`, opts),\n          )\n        }\n        if(this.lockRef && this.lockRef > pendingRef){\n          this.el.dispatchEvent(\n            new CustomEvent(`phx:undo-lock:${pendingRef}`, opts),\n          )\n        }\n        return pendingRef > ref\n      })\n    })\n\n    // clean up if fully resolved\n    if(this.isFullyResolvedBy(ref)){ this.el.removeAttribute(PHX_REF_SRC) }\n  }\n\n  // private\n\n  isWithin(ref){\n    return !((this.loadingRef !== null && this.loadingRef > ref) && (this.lockRef !== null && this.lockRef > ref))\n  }\n\n  // Check for cloned PHX_REF_LOCK element that has been morphed behind\n  // the scenes while this element was locked in the DOM.\n  // When we apply the cloned tree to the active DOM element, we must\n  //\n  //   1. execute pending mounted hooks for nodes now in the DOM\n  //   2. undo any ref inside the cloned tree that has since been ack'd\n  undoLocks(ref, phxEvent, eachCloneCallback){\n    if(!this.isLockUndoneBy(ref)){ return }\n\n    let clonedTree = DOM.private(this.el, PHX_REF_LOCK)\n    if(clonedTree){\n      eachCloneCallback(clonedTree)\n      DOM.deletePrivate(this.el, PHX_REF_LOCK)\n    }\n    this.el.removeAttribute(PHX_REF_LOCK)\n\n    let opts = {detail: {ref: ref, event: phxEvent}, bubbles: true, cancelable: false}\n    this.el.dispatchEvent(new CustomEvent(`phx:undo-lock:${this.lockRef}`, opts))\n  }\n\n  undoLoading(ref, phxEvent){\n    if(!this.isLoadingUndoneBy(ref)){\n      if(this.canUndoLoading(ref) && this.el.classList.contains(\"phx-submit-loading\")){\n        this.el.classList.remove(\"phx-change-loading\")\n      }\n      return\n    }\n\n    if(this.canUndoLoading(ref)){\n      this.el.removeAttribute(PHX_REF_LOADING)\n      let disabledVal = this.el.getAttribute(PHX_DISABLED)\n      let readOnlyVal = this.el.getAttribute(PHX_READONLY)\n      // restore inputs\n      if(readOnlyVal !== null){\n        this.el.readOnly = readOnlyVal === \"true\" ? true : false\n        this.el.removeAttribute(PHX_READONLY)\n      }\n      if(disabledVal !== null){\n        this.el.disabled = disabledVal === \"true\" ? true : false\n        this.el.removeAttribute(PHX_DISABLED)\n      }\n      // restore disables\n      let disableRestore = this.el.getAttribute(PHX_DISABLE_WITH_RESTORE)\n      if(disableRestore !== null){\n        this.el.innerText = disableRestore\n        this.el.removeAttribute(PHX_DISABLE_WITH_RESTORE)\n      }\n\n      let opts = {detail: {ref: ref, event: phxEvent}, bubbles: true, cancelable: false}\n      this.el.dispatchEvent(new CustomEvent(`phx:undo-loading:${this.loadingRef}`, opts))\n    }\n\n    // remove classes\n    PHX_EVENT_CLASSES.forEach(name => {\n      if(name !== \"phx-submit-loading\" || this.canUndoLoading(ref)){\n        DOM.removeClass(this.el, name)\n      }\n    })\n  }\n\n  isLoadingUndoneBy(ref){ return this.loadingRef === null ? false : this.loadingRef <= ref }\n  isLockUndoneBy(ref){ return this.lockRef === null ? false : this.lockRef <= ref }\n\n  isFullyResolvedBy(ref){\n    return (this.loadingRef === null || this.loadingRef <= ref) && (this.lockRef === null || this.lockRef <= ref)\n  }\n\n  // only remove the phx-submit-loading class if we are not locked\n  canUndoLoading(ref){ return this.lockRef === null || this.lockRef <= ref }\n}\n", "import {\n  maybe\n} from \"./utils\"\n\nimport DOM from \"./dom\"\n\nexport default class DOMPostMorphRestorer {\n  constructor(containerBefore, containerAfter, updateType){\n    let idsBefore = new Set()\n    let idsAfter = new Set([...containerAfter.children].map(child => child.id))\n\n    let elementsToModify = []\n\n    Array.from(containerBefore.children).forEach(child => {\n      if(child.id){ // all of our children should be elements with ids\n        idsBefore.add(child.id)\n        if(idsAfter.has(child.id)){\n          let previousElementId = child.previousElementSibling && child.previousElementSibling.id\n          elementsToModify.push({elementId: child.id, previousElementId: previousElementId})\n        }\n      }\n    })\n\n    this.containerId = containerAfter.id\n    this.updateType = updateType\n    this.elementsToModify = elementsToModify\n    this.elementIdsToAdd = [...idsAfter].filter(id => !idsBefore.has(id))\n  }\n\n  // We do the following to optimize append/prepend operations:\n  //   1) Track ids of modified elements & of new elements\n  //   2) All the modified elements are put back in the correct position in the DOM tree\n  //      by storing the id of their previous sibling\n  //   3) New elements are going to be put in the right place by morphdom during append.\n  //      For prepend, we move them to the first position in the container\n  perform(){\n    let container = DOM.byId(this.containerId)\n    this.elementsToModify.forEach(elementToModify => {\n      if(elementToModify.previousElementId){\n        maybe(document.getElementById(elementToModify.previousElementId), previousElem => {\n          maybe(document.getElementById(elementToModify.elementId), elem => {\n            let isInRightPlace = elem.previousElementSibling && elem.previousElementSibling.id == previousElem.id\n            if(!isInRightPlace){\n              previousElem.insertAdjacentElement(\"afterend\", elem)\n            }\n          })\n        })\n      } else {\n        // This is the first element in the container\n        maybe(document.getElementById(elementToModify.elementId), elem => {\n          let isInRightPlace = elem.previousElementSibling == null\n          if(!isInRightPlace){\n            container.insertAdjacentElement(\"afterbegin\", elem)\n          }\n        })\n      }\n    })\n\n    if(this.updateType == \"prepend\"){\n      this.elementIdsToAdd.reverse().forEach(elemId => {\n        maybe(document.getElementById(elemId), elem => container.insertAdjacentElement(\"afterbegin\", elem))\n      })\n    }\n  }\n}\n", "var DOCUMENT_FRAGMENT_NODE = 11;\n\nfunction morphAttrs(fromNode, toNode) {\n    var toNodeAttrs = toNode.attributes;\n    var attr;\n    var attrName;\n    var attrNamespaceURI;\n    var attrValue;\n    var fromValue;\n\n    // document-fragments dont have attributes so lets not do anything\n    if (toNode.nodeType === DOCUMENT_FRAGMENT_NODE || fromNode.nodeType === DOCUMENT_FRAGMENT_NODE) {\n      return;\n    }\n\n    // update attributes on original DOM element\n    for (var i = toNodeAttrs.length - 1; i >= 0; i--) {\n        attr = toNodeAttrs[i];\n        attrName = attr.name;\n        attrNamespaceURI = attr.namespaceURI;\n        attrValue = attr.value;\n\n        if (attrNamespaceURI) {\n            attrName = attr.localName || attrName;\n            fromValue = fromNode.getAttributeNS(attrNamespaceURI, attrName);\n\n            if (fromValue !== attrValue) {\n                if (attr.prefix === 'xmlns'){\n                    attrName = attr.name; // It's not allowed to set an attribute with the XMLNS namespace without specifying the `xmlns` prefix\n                }\n                fromNode.setAttributeNS(attrNamespaceURI, attrName, attrValue);\n            }\n        } else {\n            fromValue = fromNode.getAttribute(attrName);\n\n            if (fromValue !== attrValue) {\n                fromNode.setAttribute(attrName, attrValue);\n            }\n        }\n    }\n\n    // Remove any extra attributes found on the original DOM element that\n    // weren't found on the target element.\n    var fromNodeAttrs = fromNode.attributes;\n\n    for (var d = fromNodeAttrs.length - 1; d >= 0; d--) {\n        attr = fromNodeAttrs[d];\n        attrName = attr.name;\n        attrNamespaceURI = attr.namespaceURI;\n\n        if (attrNamespaceURI) {\n            attrName = attr.localName || attrName;\n\n            if (!toNode.hasAttributeNS(attrNamespaceURI, attrName)) {\n                fromNode.removeAttributeNS(attrNamespaceURI, attrName);\n            }\n        } else {\n            if (!toNode.hasAttribute(attrName)) {\n                fromNode.removeAttribute(attrName);\n            }\n        }\n    }\n}\n\nvar range; // Create a range object for efficently rendering strings to elements.\nvar NS_XHTML = 'http://www.w3.org/1999/xhtml';\n\nvar doc = typeof document === 'undefined' ? undefined : document;\nvar HAS_TEMPLATE_SUPPORT = !!doc && 'content' in doc.createElement('template');\nvar HAS_RANGE_SUPPORT = !!doc && doc.createRange && 'createContextualFragment' in doc.createRange();\n\nfunction createFragmentFromTemplate(str) {\n    var template = doc.createElement('template');\n    template.innerHTML = str;\n    return template.content.childNodes[0];\n}\n\nfunction createFragmentFromRange(str) {\n    if (!range) {\n        range = doc.createRange();\n        range.selectNode(doc.body);\n    }\n\n    var fragment = range.createContextualFragment(str);\n    return fragment.childNodes[0];\n}\n\nfunction createFragmentFromWrap(str) {\n    var fragment = doc.createElement('body');\n    fragment.innerHTML = str;\n    return fragment.childNodes[0];\n}\n\n/**\n * This is about the same\n * var html = new DOMParser().parseFromString(str, 'text/html');\n * return html.body.firstChild;\n *\n * @method toElement\n * @param {String} str\n */\nfunction toElement(str) {\n    str = str.trim();\n    if (HAS_TEMPLATE_SUPPORT) {\n      // avoid restrictions on content for things like `<tr><th>Hi</th></tr>` which\n      // createContextualFragment doesn't support\n      // <template> support not available in IE\n      return createFragmentFromTemplate(str);\n    } else if (HAS_RANGE_SUPPORT) {\n      return createFragmentFromRange(str);\n    }\n\n    return createFragmentFromWrap(str);\n}\n\n/**\n * Returns true if two node's names are the same.\n *\n * NOTE: We don't bother checking `namespaceURI` because you will never find two HTML elements with the same\n *       nodeName and different namespace URIs.\n *\n * @param {Element} a\n * @param {Element} b The target element\n * @return {boolean}\n */\nfunction compareNodeNames(fromEl, toEl) {\n    var fromNodeName = fromEl.nodeName;\n    var toNodeName = toEl.nodeName;\n    var fromCodeStart, toCodeStart;\n\n    if (fromNodeName === toNodeName) {\n        return true;\n    }\n\n    fromCodeStart = fromNodeName.charCodeAt(0);\n    toCodeStart = toNodeName.charCodeAt(0);\n\n    // If the target element is a virtual DOM node or SVG node then we may\n    // need to normalize the tag name before comparing. Normal HTML elements that are\n    // in the \"http://www.w3.org/1999/xhtml\"\n    // are converted to upper case\n    if (fromCodeStart <= 90 && toCodeStart >= 97) { // from is upper and to is lower\n        return fromNodeName === toNodeName.toUpperCase();\n    } else if (toCodeStart <= 90 && fromCodeStart >= 97) { // to is upper and from is lower\n        return toNodeName === fromNodeName.toUpperCase();\n    } else {\n        return false;\n    }\n}\n\n/**\n * Create an element, optionally with a known namespace URI.\n *\n * @param {string} name the element name, e.g. 'div' or 'svg'\n * @param {string} [namespaceURI] the element's namespace URI, i.e. the value of\n * its `xmlns` attribute or its inferred namespace.\n *\n * @return {Element}\n */\nfunction createElementNS(name, namespaceURI) {\n    return !namespaceURI || namespaceURI === NS_XHTML ?\n        doc.createElement(name) :\n        doc.createElementNS(namespaceURI, name);\n}\n\n/**\n * Copies the children of one DOM element to another DOM element\n */\nfunction moveChildren(fromEl, toEl) {\n    var curChild = fromEl.firstChild;\n    while (curChild) {\n        var nextChild = curChild.nextSibling;\n        toEl.appendChild(curChild);\n        curChild = nextChild;\n    }\n    return toEl;\n}\n\nfunction syncBooleanAttrProp(fromEl, toEl, name) {\n    if (fromEl[name] !== toEl[name]) {\n        fromEl[name] = toEl[name];\n        if (fromEl[name]) {\n            fromEl.setAttribute(name, '');\n        } else {\n            fromEl.removeAttribute(name);\n        }\n    }\n}\n\nvar specialElHandlers = {\n    OPTION: function(fromEl, toEl) {\n        var parentNode = fromEl.parentNode;\n        if (parentNode) {\n            var parentName = parentNode.nodeName.toUpperCase();\n            if (parentName === 'OPTGROUP') {\n                parentNode = parentNode.parentNode;\n                parentName = parentNode && parentNode.nodeName.toUpperCase();\n            }\n            if (parentName === 'SELECT' && !parentNode.hasAttribute('multiple')) {\n                if (fromEl.hasAttribute('selected') && !toEl.selected) {\n                    // Workaround for MS Edge bug where the 'selected' attribute can only be\n                    // removed if set to a non-empty value:\n                    // https://developer.microsoft.com/en-us/microsoft-edge/platform/issues/12087679/\n                    fromEl.setAttribute('selected', 'selected');\n                    fromEl.removeAttribute('selected');\n                }\n                // We have to reset select element's selectedIndex to -1, otherwise setting\n                // fromEl.selected using the syncBooleanAttrProp below has no effect.\n                // The correct selectedIndex will be set in the SELECT special handler below.\n                parentNode.selectedIndex = -1;\n            }\n        }\n        syncBooleanAttrProp(fromEl, toEl, 'selected');\n    },\n    /**\n     * The \"value\" attribute is special for the <input> element since it sets\n     * the initial value. Changing the \"value\" attribute without changing the\n     * \"value\" property will have no effect since it is only used to the set the\n     * initial value.  Similar for the \"checked\" attribute, and \"disabled\".\n     */\n    INPUT: function(fromEl, toEl) {\n        syncBooleanAttrProp(fromEl, toEl, 'checked');\n        syncBooleanAttrProp(fromEl, toEl, 'disabled');\n\n        if (fromEl.value !== toEl.value) {\n            fromEl.value = toEl.value;\n        }\n\n        if (!toEl.hasAttribute('value')) {\n            fromEl.removeAttribute('value');\n        }\n    },\n\n    TEXTAREA: function(fromEl, toEl) {\n        var newValue = toEl.value;\n        if (fromEl.value !== newValue) {\n            fromEl.value = newValue;\n        }\n\n        var firstChild = fromEl.firstChild;\n        if (firstChild) {\n            // Needed for IE. Apparently IE sets the placeholder as the\n            // node value and vise versa. This ignores an empty update.\n            var oldValue = firstChild.nodeValue;\n\n            if (oldValue == newValue || (!newValue && oldValue == fromEl.placeholder)) {\n                return;\n            }\n\n            firstChild.nodeValue = newValue;\n        }\n    },\n    SELECT: function(fromEl, toEl) {\n        if (!toEl.hasAttribute('multiple')) {\n            var selectedIndex = -1;\n            var i = 0;\n            // We have to loop through children of fromEl, not toEl since nodes can be moved\n            // from toEl to fromEl directly when morphing.\n            // At the time this special handler is invoked, all children have already been morphed\n            // and appended to / removed from fromEl, so using fromEl here is safe and correct.\n            var curChild = fromEl.firstChild;\n            var optgroup;\n            var nodeName;\n            while(curChild) {\n                nodeName = curChild.nodeName && curChild.nodeName.toUpperCase();\n                if (nodeName === 'OPTGROUP') {\n                    optgroup = curChild;\n                    curChild = optgroup.firstChild;\n                } else {\n                    if (nodeName === 'OPTION') {\n                        if (curChild.hasAttribute('selected')) {\n                            selectedIndex = i;\n                            break;\n                        }\n                        i++;\n                    }\n                    curChild = curChild.nextSibling;\n                    if (!curChild && optgroup) {\n                        curChild = optgroup.nextSibling;\n                        optgroup = null;\n                    }\n                }\n            }\n\n            fromEl.selectedIndex = selectedIndex;\n        }\n    }\n};\n\nvar ELEMENT_NODE = 1;\nvar DOCUMENT_FRAGMENT_NODE$1 = 11;\nvar TEXT_NODE = 3;\nvar COMMENT_NODE = 8;\n\nfunction noop() {}\n\nfunction defaultGetNodeKey(node) {\n  if (node) {\n    return (node.getAttribute && node.getAttribute('id')) || node.id;\n  }\n}\n\nfunction morphdomFactory(morphAttrs) {\n\n  return function morphdom(fromNode, toNode, options) {\n    if (!options) {\n      options = {};\n    }\n\n    if (typeof toNode === 'string') {\n      if (fromNode.nodeName === '#document' || fromNode.nodeName === 'HTML' || fromNode.nodeName === 'BODY') {\n        var toNodeHtml = toNode;\n        toNode = doc.createElement('html');\n        toNode.innerHTML = toNodeHtml;\n      } else {\n        toNode = toElement(toNode);\n      }\n    } else if (toNode.nodeType === DOCUMENT_FRAGMENT_NODE$1) {\n      toNode = toNode.firstElementChild;\n    }\n\n    var getNodeKey = options.getNodeKey || defaultGetNodeKey;\n    var onBeforeNodeAdded = options.onBeforeNodeAdded || noop;\n    var onNodeAdded = options.onNodeAdded || noop;\n    var onBeforeElUpdated = options.onBeforeElUpdated || noop;\n    var onElUpdated = options.onElUpdated || noop;\n    var onBeforeNodeDiscarded = options.onBeforeNodeDiscarded || noop;\n    var onNodeDiscarded = options.onNodeDiscarded || noop;\n    var onBeforeElChildrenUpdated = options.onBeforeElChildrenUpdated || noop;\n    var skipFromChildren = options.skipFromChildren || noop;\n    var addChild = options.addChild || function(parent, child){ return parent.appendChild(child); };\n    var childrenOnly = options.childrenOnly === true;\n\n    // This object is used as a lookup to quickly find all keyed elements in the original DOM tree.\n    var fromNodesLookup = Object.create(null);\n    var keyedRemovalList = [];\n\n    function addKeyedRemoval(key) {\n      keyedRemovalList.push(key);\n    }\n\n    function walkDiscardedChildNodes(node, skipKeyedNodes) {\n      if (node.nodeType === ELEMENT_NODE) {\n        var curChild = node.firstChild;\n        while (curChild) {\n\n          var key = undefined;\n\n          if (skipKeyedNodes && (key = getNodeKey(curChild))) {\n            // If we are skipping keyed nodes then we add the key\n            // to a list so that it can be handled at the very end.\n            addKeyedRemoval(key);\n          } else {\n            // Only report the node as discarded if it is not keyed. We do this because\n            // at the end we loop through all keyed elements that were unmatched\n            // and then discard them in one final pass.\n            onNodeDiscarded(curChild);\n            if (curChild.firstChild) {\n              walkDiscardedChildNodes(curChild, skipKeyedNodes);\n            }\n          }\n\n          curChild = curChild.nextSibling;\n        }\n      }\n    }\n\n    /**\n    * Removes a DOM node out of the original DOM\n    *\n    * @param  {Node} node The node to remove\n    * @param  {Node} parentNode The nodes parent\n    * @param  {Boolean} skipKeyedNodes If true then elements with keys will be skipped and not discarded.\n    * @return {undefined}\n    */\n    function removeNode(node, parentNode, skipKeyedNodes) {\n      if (onBeforeNodeDiscarded(node) === false) {\n        return;\n      }\n\n      if (parentNode) {\n        parentNode.removeChild(node);\n      }\n\n      onNodeDiscarded(node);\n      walkDiscardedChildNodes(node, skipKeyedNodes);\n    }\n\n    // // TreeWalker implementation is no faster, but keeping this around in case this changes in the future\n    // function indexTree(root) {\n    //     var treeWalker = document.createTreeWalker(\n    //         root,\n    //         NodeFilter.SHOW_ELEMENT);\n    //\n    //     var el;\n    //     while((el = treeWalker.nextNode())) {\n    //         var key = getNodeKey(el);\n    //         if (key) {\n    //             fromNodesLookup[key] = el;\n    //         }\n    //     }\n    // }\n\n    // // NodeIterator implementation is no faster, but keeping this around in case this changes in the future\n    //\n    // function indexTree(node) {\n    //     var nodeIterator = document.createNodeIterator(node, NodeFilter.SHOW_ELEMENT);\n    //     var el;\n    //     while((el = nodeIterator.nextNode())) {\n    //         var key = getNodeKey(el);\n    //         if (key) {\n    //             fromNodesLookup[key] = el;\n    //         }\n    //     }\n    // }\n\n    function indexTree(node) {\n      if (node.nodeType === ELEMENT_NODE || node.nodeType === DOCUMENT_FRAGMENT_NODE$1) {\n        var curChild = node.firstChild;\n        while (curChild) {\n          var key = getNodeKey(curChild);\n          if (key) {\n            fromNodesLookup[key] = curChild;\n          }\n\n          // Walk recursively\n          indexTree(curChild);\n\n          curChild = curChild.nextSibling;\n        }\n      }\n    }\n\n    indexTree(fromNode);\n\n    function handleNodeAdded(el) {\n      onNodeAdded(el);\n\n      var curChild = el.firstChild;\n      while (curChild) {\n        var nextSibling = curChild.nextSibling;\n\n        var key = getNodeKey(curChild);\n        if (key) {\n          var unmatchedFromEl = fromNodesLookup[key];\n          // if we find a duplicate #id node in cache, replace `el` with cache value\n          // and morph it to the child node.\n          if (unmatchedFromEl && compareNodeNames(curChild, unmatchedFromEl)) {\n            curChild.parentNode.replaceChild(unmatchedFromEl, curChild);\n            morphEl(unmatchedFromEl, curChild);\n          } else {\n            handleNodeAdded(curChild);\n          }\n        } else {\n          // recursively call for curChild and it's children to see if we find something in\n          // fromNodesLookup\n          handleNodeAdded(curChild);\n        }\n\n        curChild = nextSibling;\n      }\n    }\n\n    function cleanupFromEl(fromEl, curFromNodeChild, curFromNodeKey) {\n      // We have processed all of the \"to nodes\". If curFromNodeChild is\n      // non-null then we still have some from nodes left over that need\n      // to be removed\n      while (curFromNodeChild) {\n        var fromNextSibling = curFromNodeChild.nextSibling;\n        if ((curFromNodeKey = getNodeKey(curFromNodeChild))) {\n          // Since the node is keyed it might be matched up later so we defer\n          // the actual removal to later\n          addKeyedRemoval(curFromNodeKey);\n        } else {\n          // NOTE: we skip nested keyed nodes from being removed since there is\n          //       still a chance they will be matched up later\n          removeNode(curFromNodeChild, fromEl, true /* skip keyed nodes */);\n        }\n        curFromNodeChild = fromNextSibling;\n      }\n    }\n\n    function morphEl(fromEl, toEl, childrenOnly) {\n      var toElKey = getNodeKey(toEl);\n\n      if (toElKey) {\n        // If an element with an ID is being morphed then it will be in the final\n        // DOM so clear it out of the saved elements collection\n        delete fromNodesLookup[toElKey];\n      }\n\n      if (!childrenOnly) {\n        // optional\n        var beforeUpdateResult = onBeforeElUpdated(fromEl, toEl);\n        if (beforeUpdateResult === false) {\n          return;\n        } else if (beforeUpdateResult instanceof HTMLElement) {\n          fromEl = beforeUpdateResult;\n          // reindex the new fromEl in case it's not in the same\n          // tree as the original fromEl\n          // (Phoenix LiveView sometimes returns a cloned tree,\n          //  but keyed lookups would still point to the original tree)\n          indexTree(fromEl);\n        }\n\n        // update attributes on original DOM element first\n        morphAttrs(fromEl, toEl);\n        // optional\n        onElUpdated(fromEl);\n\n        if (onBeforeElChildrenUpdated(fromEl, toEl) === false) {\n          return;\n        }\n      }\n\n      if (fromEl.nodeName !== 'TEXTAREA') {\n        morphChildren(fromEl, toEl);\n      } else {\n        specialElHandlers.TEXTAREA(fromEl, toEl);\n      }\n    }\n\n    function morphChildren(fromEl, toEl) {\n      var skipFrom = skipFromChildren(fromEl, toEl);\n      var curToNodeChild = toEl.firstChild;\n      var curFromNodeChild = fromEl.firstChild;\n      var curToNodeKey;\n      var curFromNodeKey;\n\n      var fromNextSibling;\n      var toNextSibling;\n      var matchingFromEl;\n\n      // walk the children\n      outer: while (curToNodeChild) {\n        toNextSibling = curToNodeChild.nextSibling;\n        curToNodeKey = getNodeKey(curToNodeChild);\n\n        // walk the fromNode children all the way through\n        while (!skipFrom && curFromNodeChild) {\n          fromNextSibling = curFromNodeChild.nextSibling;\n\n          if (curToNodeChild.isSameNode && curToNodeChild.isSameNode(curFromNodeChild)) {\n            curToNodeChild = toNextSibling;\n            curFromNodeChild = fromNextSibling;\n            continue outer;\n          }\n\n          curFromNodeKey = getNodeKey(curFromNodeChild);\n\n          var curFromNodeType = curFromNodeChild.nodeType;\n\n          // this means if the curFromNodeChild doesnt have a match with the curToNodeChild\n          var isCompatible = undefined;\n\n          if (curFromNodeType === curToNodeChild.nodeType) {\n            if (curFromNodeType === ELEMENT_NODE) {\n              // Both nodes being compared are Element nodes\n\n              if (curToNodeKey) {\n                // The target node has a key so we want to match it up with the correct element\n                // in the original DOM tree\n                if (curToNodeKey !== curFromNodeKey) {\n                  // The current element in the original DOM tree does not have a matching key so\n                  // let's check our lookup to see if there is a matching element in the original\n                  // DOM tree\n                  if ((matchingFromEl = fromNodesLookup[curToNodeKey])) {\n                    if (fromNextSibling === matchingFromEl) {\n                      // Special case for single element removals. To avoid removing the original\n                      // DOM node out of the tree (since that can break CSS transitions, etc.),\n                      // we will instead discard the current node and wait until the next\n                      // iteration to properly match up the keyed target element with its matching\n                      // element in the original tree\n                      isCompatible = false;\n                    } else {\n                      // We found a matching keyed element somewhere in the original DOM tree.\n                      // Let's move the original DOM node into the current position and morph\n                      // it.\n\n                      // NOTE: We use insertBefore instead of replaceChild because we want to go through\n                      // the `removeNode()` function for the node that is being discarded so that\n                      // all lifecycle hooks are correctly invoked\n                      fromEl.insertBefore(matchingFromEl, curFromNodeChild);\n\n                      // fromNextSibling = curFromNodeChild.nextSibling;\n\n                      if (curFromNodeKey) {\n                        // Since the node is keyed it might be matched up later so we defer\n                        // the actual removal to later\n                        addKeyedRemoval(curFromNodeKey);\n                      } else {\n                        // NOTE: we skip nested keyed nodes from being removed since there is\n                        //       still a chance they will be matched up later\n                        removeNode(curFromNodeChild, fromEl, true /* skip keyed nodes */);\n                      }\n\n                      curFromNodeChild = matchingFromEl;\n                      curFromNodeKey = getNodeKey(curFromNodeChild);\n                    }\n                  } else {\n                    // The nodes are not compatible since the \"to\" node has a key and there\n                    // is no matching keyed node in the source tree\n                    isCompatible = false;\n                  }\n                }\n              } else if (curFromNodeKey) {\n                // The original has a key\n                isCompatible = false;\n              }\n\n              isCompatible = isCompatible !== false && compareNodeNames(curFromNodeChild, curToNodeChild);\n              if (isCompatible) {\n                // We found compatible DOM elements so transform\n                // the current \"from\" node to match the current\n                // target DOM node.\n                // MORPH\n                morphEl(curFromNodeChild, curToNodeChild);\n              }\n\n            } else if (curFromNodeType === TEXT_NODE || curFromNodeType == COMMENT_NODE) {\n              // Both nodes being compared are Text or Comment nodes\n              isCompatible = true;\n              // Simply update nodeValue on the original node to\n              // change the text value\n              if (curFromNodeChild.nodeValue !== curToNodeChild.nodeValue) {\n                curFromNodeChild.nodeValue = curToNodeChild.nodeValue;\n              }\n\n            }\n          }\n\n          if (isCompatible) {\n            // Advance both the \"to\" child and the \"from\" child since we found a match\n            // Nothing else to do as we already recursively called morphChildren above\n            curToNodeChild = toNextSibling;\n            curFromNodeChild = fromNextSibling;\n            continue outer;\n          }\n\n          // No compatible match so remove the old node from the DOM and continue trying to find a\n          // match in the original DOM. However, we only do this if the from node is not keyed\n          // since it is possible that a keyed node might match up with a node somewhere else in the\n          // target tree and we don't want to discard it just yet since it still might find a\n          // home in the final DOM tree. After everything is done we will remove any keyed nodes\n          // that didn't find a home\n          if (curFromNodeKey) {\n            // Since the node is keyed it might be matched up later so we defer\n            // the actual removal to later\n            addKeyedRemoval(curFromNodeKey);\n          } else {\n            // NOTE: we skip nested keyed nodes from being removed since there is\n            //       still a chance they will be matched up later\n            removeNode(curFromNodeChild, fromEl, true /* skip keyed nodes */);\n          }\n\n          curFromNodeChild = fromNextSibling;\n        } // END: while(curFromNodeChild) {}\n\n        // If we got this far then we did not find a candidate match for\n        // our \"to node\" and we exhausted all of the children \"from\"\n        // nodes. Therefore, we will just append the current \"to\" node\n        // to the end\n        if (curToNodeKey && (matchingFromEl = fromNodesLookup[curToNodeKey]) && compareNodeNames(matchingFromEl, curToNodeChild)) {\n          // MORPH\n          if(!skipFrom){ addChild(fromEl, matchingFromEl); }\n          morphEl(matchingFromEl, curToNodeChild);\n        } else {\n          var onBeforeNodeAddedResult = onBeforeNodeAdded(curToNodeChild);\n          if (onBeforeNodeAddedResult !== false) {\n            if (onBeforeNodeAddedResult) {\n              curToNodeChild = onBeforeNodeAddedResult;\n            }\n\n            if (curToNodeChild.actualize) {\n              curToNodeChild = curToNodeChild.actualize(fromEl.ownerDocument || doc);\n            }\n            addChild(fromEl, curToNodeChild);\n            handleNodeAdded(curToNodeChild);\n          }\n        }\n\n        curToNodeChild = toNextSibling;\n        curFromNodeChild = fromNextSibling;\n      }\n\n      cleanupFromEl(fromEl, curFromNodeChild, curFromNodeKey);\n\n      var specialElHandler = specialElHandlers[fromEl.nodeName];\n      if (specialElHandler) {\n        specialElHandler(fromEl, toEl);\n      }\n    } // END: morphChildren(...)\n\n    var morphedNode = fromNode;\n    var morphedNodeType = morphedNode.nodeType;\n    var toNodeType = toNode.nodeType;\n\n    if (!childrenOnly) {\n      // Handle the case where we are given two DOM nodes that are not\n      // compatible (e.g. <div> --> <span> or <div> --> TEXT)\n      if (morphedNodeType === ELEMENT_NODE) {\n        if (toNodeType === ELEMENT_NODE) {\n          if (!compareNodeNames(fromNode, toNode)) {\n            onNodeDiscarded(fromNode);\n            morphedNode = moveChildren(fromNode, createElementNS(toNode.nodeName, toNode.namespaceURI));\n          }\n        } else {\n          // Going from an element node to a text node\n          morphedNode = toNode;\n        }\n      } else if (morphedNodeType === TEXT_NODE || morphedNodeType === COMMENT_NODE) { // Text or comment node\n        if (toNodeType === morphedNodeType) {\n          if (morphedNode.nodeValue !== toNode.nodeValue) {\n            morphedNode.nodeValue = toNode.nodeValue;\n          }\n\n          return morphedNode;\n        } else {\n          // Text node to something else\n          morphedNode = toNode;\n        }\n      }\n    }\n\n    if (morphedNode === toNode) {\n      // The \"to node\" was not compatible with the \"from node\" so we had to\n      // toss out the \"from node\" and use the \"to node\"\n      onNodeDiscarded(fromNode);\n    } else {\n      if (toNode.isSameNode && toNode.isSameNode(morphedNode)) {\n        return;\n      }\n\n      morphEl(morphedNode, toNode, childrenOnly);\n\n      // We now need to loop over any keyed nodes that might need to be\n      // removed. We only do the removal if we know that the keyed node\n      // never found a match. When a keyed node is matched up we remove\n      // it out of fromNodesLookup and we use fromNodesLookup to determine\n      // if a keyed node has been matched up or not\n      if (keyedRemovalList) {\n        for (var i=0, len=keyedRemovalList.length; i<len; i++) {\n          var elToRemove = fromNodesLookup[keyedRemovalList[i]];\n          if (elToRemove) {\n            removeNode(elToRemove, elToRemove.parentNode, false);\n          }\n        }\n      }\n    }\n\n    if (!childrenOnly && morphedNode !== fromNode && fromNode.parentNode) {\n      if (morphedNode.actualize) {\n        morphedNode = morphedNode.actualize(fromNode.ownerDocument || doc);\n      }\n      // If we had to swap out the from node with a new node because the old\n      // node was not compatible with the target node then we need to\n      // replace the old DOM node in the original DOM tree. This is only\n      // possible if the original DOM node was part of a DOM tree which\n      // we know is the case if it has a parent node.\n      fromNode.parentNode.replaceChild(morphedNode, fromNode);\n    }\n\n    return morphedNode;\n  };\n}\n\nvar morphdom = morphdomFactory(morphAttrs);\n\nexport default morphdom;\n", "import {\n  PHX_COMPONENT,\n  PHX_PRUNE,\n  PHX_ROOT_ID,\n  PHX_SESSION,\n  PHX_SKIP,\n  PHX_MAGIC_ID,\n  PHX_STATIC,\n  PHX_TRIGGER_ACTION,\n  PHX_UPDATE,\n  PHX_REF_SRC,\n  PHX_REF_LOCK,\n  PHX_STREAM,\n  PHX_STREAM_REF,\n  PHX_VIEWPORT_TOP,\n  PHX_VIEWPORT_BOTTOM,\n} from \"./constants\"\n\nimport {\n  detectDuplicateIds,\n  detectInvalidStreamInserts,\n  isCid\n} from \"./utils\"\nimport ElementRef from \"./element_ref\"\nimport DOM from \"./dom\"\nimport DOMPostMorphRestorer from \"./dom_post_morph_restorer\"\nimport morphdom from \"morphdom\"\n\nexport default class DOMPatch {\n  constructor(view, container, id, html, streams, targetCID, opts={}){\n    this.view = view\n    this.liveSocket = view.liveSocket\n    this.container = container\n    this.id = id\n    this.rootID = view.root.id\n    this.html = html\n    this.streams = streams\n    this.streamInserts = {}\n    this.streamComponentRestore = {}\n    this.targetCID = targetCID\n    this.cidPatch = isCid(this.targetCID)\n    this.pendingRemoves = []\n    this.phxRemove = this.liveSocket.binding(\"remove\")\n    this.targetContainer = this.isCIDPatch() ? this.targetCIDContainer(html) : container\n    this.callbacks = {\n      beforeadded: [], beforeupdated: [], beforephxChildAdded: [],\n      afteradded: [], afterupdated: [], afterdiscarded: [], afterphxChildAdded: [],\n      aftertransitionsDiscarded: []\n    }\n    this.withChildren = opts.withChildren || opts.undoRef || false\n    this.undoRef = opts.undoRef\n  }\n\n  before(kind, callback){ this.callbacks[`before${kind}`].push(callback) }\n  after(kind, callback){ this.callbacks[`after${kind}`].push(callback) }\n\n  trackBefore(kind, ...args){\n    this.callbacks[`before${kind}`].forEach(callback => callback(...args))\n  }\n\n  trackAfter(kind, ...args){\n    this.callbacks[`after${kind}`].forEach(callback => callback(...args))\n  }\n\n  markPrunableContentForRemoval(){\n    let phxUpdate = this.liveSocket.binding(PHX_UPDATE)\n    DOM.all(this.container, `[${phxUpdate}=append] > *, [${phxUpdate}=prepend] > *`, el => {\n      el.setAttribute(PHX_PRUNE, \"\")\n    })\n  }\n\n  perform(isJoinPatch){\n    let {view, liveSocket, html, container, targetContainer} = this\n    if(this.isCIDPatch() && !targetContainer){ return }\n\n    let focused = liveSocket.getActiveElement()\n    let {selectionStart, selectionEnd} = focused && DOM.hasSelectionRange(focused) ? focused : {}\n    let phxUpdate = liveSocket.binding(PHX_UPDATE)\n    let phxViewportTop = liveSocket.binding(PHX_VIEWPORT_TOP)\n    let phxViewportBottom = liveSocket.binding(PHX_VIEWPORT_BOTTOM)\n    let phxTriggerExternal = liveSocket.binding(PHX_TRIGGER_ACTION)\n    let added = []\n    let updates = []\n    let appendPrependUpdates = []\n\n    let externalFormTriggered = null\n\n    function morph(targetContainer, source, withChildren=this.withChildren){\n      let morphCallbacks = {\n        // normally, we are running with childrenOnly, as the patch HTML for a LV\n        // does not include the LV attrs (data-phx-session, etc.)\n        // when we are patching a live component, we do want to patch the root element as well;\n        // another case is the recursive patch of a stream item that was kept on reset (-> onBeforeNodeAdded)\n        childrenOnly: targetContainer.getAttribute(PHX_COMPONENT) === null && !withChildren,\n        getNodeKey: (node) => {\n          if(DOM.isPhxDestroyed(node)){ return null }\n          // If we have a join patch, then by definition there was no PHX_MAGIC_ID.\n          // This is important to reduce the amount of elements morphdom discards.\n          if(isJoinPatch){ return node.id }\n          return node.id || (node.getAttribute && node.getAttribute(PHX_MAGIC_ID))\n        },\n        // skip indexing from children when container is stream\n        skipFromChildren: (from) => { return from.getAttribute(phxUpdate) === PHX_STREAM },\n        // tell morphdom how to add a child\n        addChild: (parent, child) => {\n          let {ref, streamAt} = this.getStreamInsert(child)\n          if(ref === undefined){ return parent.appendChild(child) }\n\n          this.setStreamRef(child, ref)\n\n          // streaming\n          if(streamAt === 0){\n            parent.insertAdjacentElement(\"afterbegin\", child)\n          } else if(streamAt === -1){\n            let lastChild = parent.lastElementChild\n            if(lastChild && !lastChild.hasAttribute(PHX_STREAM_REF)){\n              let nonStreamChild = Array.from(parent.children).find(c => !c.hasAttribute(PHX_STREAM_REF))\n              parent.insertBefore(child, nonStreamChild)\n            } else {\n              parent.appendChild(child)\n            }\n          } else if(streamAt > 0){\n            let sibling = Array.from(parent.children)[streamAt]\n            parent.insertBefore(child, sibling)\n          }\n        },\n        onBeforeNodeAdded: (el) => {\n          DOM.maintainPrivateHooks(el, el, phxViewportTop, phxViewportBottom)\n          this.trackBefore(\"added\", el)\n\n          let morphedEl = el\n          // this is a stream item that was kept on reset, recursively morph it\n          if(this.streamComponentRestore[el.id]){\n            morphedEl = this.streamComponentRestore[el.id]\n            delete this.streamComponentRestore[el.id]\n            morph.call(this, morphedEl, el, true)\n          }\n\n          return morphedEl\n        },\n        onNodeAdded: (el) => {\n          if(el.getAttribute){ this.maybeReOrderStream(el, true) }\n\n          // hack to fix Safari handling of img srcset and video tags\n          if(el instanceof HTMLImageElement && el.srcset){\n            el.srcset = el.srcset\n          } else if(el instanceof HTMLVideoElement && el.autoplay){\n            el.play()\n          }\n          if(DOM.isNowTriggerFormExternal(el, phxTriggerExternal)){\n            externalFormTriggered = el\n          }\n\n          // nested view handling\n          if((DOM.isPhxChild(el) && view.ownsElement(el)) || DOM.isPhxSticky(el) && view.ownsElement(el.parentNode)){\n            this.trackAfter(\"phxChildAdded\", el)\n          }\n          added.push(el)\n        },\n        onNodeDiscarded: (el) => this.onNodeDiscarded(el),\n        onBeforeNodeDiscarded: (el) => {\n          if(el.getAttribute && el.getAttribute(PHX_PRUNE) !== null){ return true }\n          if(el.parentElement !== null && el.id &&\n            DOM.isPhxUpdate(el.parentElement, phxUpdate, [PHX_STREAM, \"append\", \"prepend\"])){\n            return false\n          }\n          if(this.maybePendingRemove(el)){ return false }\n          if(this.skipCIDSibling(el)){ return false }\n\n          return true\n        },\n        onElUpdated: (el) => {\n          if(DOM.isNowTriggerFormExternal(el, phxTriggerExternal)){\n            externalFormTriggered = el\n          }\n          updates.push(el)\n          this.maybeReOrderStream(el, false)\n        },\n        onBeforeElUpdated: (fromEl, toEl) => {\n          // if we are patching the root target container and the id has changed, treat it as a new node\n          // by replacing the fromEl with the toEl, which ensures hooks are torn down and re-created\n          if(fromEl.id && fromEl.isSameNode(targetContainer) && fromEl.id !== toEl.id){\n            morphCallbacks.onNodeDiscarded(fromEl)\n            fromEl.replaceWith(toEl)\n            return morphCallbacks.onNodeAdded(toEl)\n          }\n          DOM.syncPendingAttrs(fromEl, toEl)\n          DOM.maintainPrivateHooks(fromEl, toEl, phxViewportTop, phxViewportBottom)\n          DOM.cleanChildNodes(toEl, phxUpdate)\n          if(this.skipCIDSibling(toEl)){\n            // if this is a live component used in a stream, we may need to reorder it\n            this.maybeReOrderStream(fromEl)\n            return false\n          }\n          if(DOM.isPhxSticky(fromEl)){\n            [PHX_SESSION, PHX_STATIC, PHX_ROOT_ID]\n              .map(attr => [attr, fromEl.getAttribute(attr), toEl.getAttribute(attr)])\n              .forEach(([attr, fromVal, toVal]) => {\n                if(toVal && fromVal !== toVal){ fromEl.setAttribute(attr, toVal) }\n              })\n\n            return false\n          }\n          if(DOM.isIgnored(fromEl, phxUpdate) || (fromEl.form && fromEl.form.isSameNode(externalFormTriggered))){\n            this.trackBefore(\"updated\", fromEl, toEl)\n            DOM.mergeAttrs(fromEl, toEl, {isIgnored: DOM.isIgnored(fromEl, phxUpdate)})\n            updates.push(fromEl)\n            DOM.applyStickyOperations(fromEl)\n            return false\n          }\n          if(fromEl.type === \"number\" && (fromEl.validity && fromEl.validity.badInput)){ return false }\n          // If the element has PHX_REF_SRC, it is loading or locked and awaiting an ack.\n          // If it's locked, we clone the fromEl tree and instruct morphdom to use\n          // the cloned tree as the source of the morph for this branch from here on out.\n          // We keep a reference to the cloned tree in the element's private data, and\n          // on ack (view.undoRefs), we morph the cloned tree with the true fromEl in the DOM to\n          // apply any changes that happened while the element was locked.\n          let isFocusedFormEl = focused && fromEl.isSameNode(focused) && DOM.isFormInput(fromEl)\n          let focusedSelectChanged = isFocusedFormEl && this.isChangedSelect(fromEl, toEl)\n          if(fromEl.hasAttribute(PHX_REF_SRC)){\n            const ref = new ElementRef(fromEl)\n            // only perform the clone step if this is not a patch that unlocks\n            if(ref.lockRef && (!this.undoRef || !ref.isLockUndoneBy(this.undoRef))){\n              if(DOM.isUploadInput(fromEl)){\n                DOM.mergeAttrs(fromEl, toEl, {isIgnored: true})\n                this.trackBefore(\"updated\", fromEl, toEl)\n                updates.push(fromEl)\n              }\n              DOM.applyStickyOperations(fromEl)\n              let isLocked = fromEl.hasAttribute(PHX_REF_LOCK)\n              let clone = isLocked ? DOM.private(fromEl, PHX_REF_LOCK) || fromEl.cloneNode(true) : null\n              if(clone){\n                DOM.putPrivate(fromEl, PHX_REF_LOCK, clone)\n                if(!isFocusedFormEl){\n                  fromEl = clone\n                }\n              }\n            }\n          }\n\n          // nested view handling\n          if(DOM.isPhxChild(toEl)){\n            let prevSession = fromEl.getAttribute(PHX_SESSION)\n            DOM.mergeAttrs(fromEl, toEl, {exclude: [PHX_STATIC]})\n            if(prevSession !== \"\"){ fromEl.setAttribute(PHX_SESSION, prevSession) }\n            fromEl.setAttribute(PHX_ROOT_ID, this.rootID)\n            DOM.applyStickyOperations(fromEl)\n            return false\n          }\n\n          // if we are undoing a lock, copy potentially nested clones over\n          if(this.undoRef && DOM.private(toEl, PHX_REF_LOCK)){\n            DOM.putPrivate(fromEl, PHX_REF_LOCK, DOM.private(toEl, PHX_REF_LOCK))\n          }\n          // now copy regular DOM.private data\n          DOM.copyPrivates(toEl, fromEl)\n\n          // skip patching focused inputs unless focus is a select that has changed options\n          if(isFocusedFormEl && fromEl.type !== \"hidden\" && !focusedSelectChanged){\n            this.trackBefore(\"updated\", fromEl, toEl)\n            DOM.mergeFocusedInput(fromEl, toEl)\n            DOM.syncAttrsToProps(fromEl)\n            updates.push(fromEl)\n            DOM.applyStickyOperations(fromEl)\n            return false\n          } else {\n            // blur focused select if it changed so native UI is updated (ie safari won't update visible options)\n            if(focusedSelectChanged){ fromEl.blur() }\n            if(DOM.isPhxUpdate(toEl, phxUpdate, [\"append\", \"prepend\"])){\n              appendPrependUpdates.push(new DOMPostMorphRestorer(fromEl, toEl, toEl.getAttribute(phxUpdate)))\n            }\n\n            DOM.syncAttrsToProps(toEl)\n            DOM.applyStickyOperations(toEl)\n            this.trackBefore(\"updated\", fromEl, toEl)\n            return fromEl\n          }\n        }\n      }\n      morphdom(targetContainer, source, morphCallbacks)\n    }\n\n    this.trackBefore(\"added\", container)\n    this.trackBefore(\"updated\", container, container)\n\n    liveSocket.time(\"morphdom\", () => {\n      this.streams.forEach(([ref, inserts, deleteIds, reset]) => {\n        inserts.forEach(([key, streamAt, limit]) => {\n          this.streamInserts[key] = {ref, streamAt, limit, reset}\n        })\n        if(reset !== undefined){\n          DOM.all(container, `[${PHX_STREAM_REF}=\"${ref}\"]`, child => {\n            this.removeStreamChildElement(child)\n          })\n        }\n        deleteIds.forEach(id => {\n          let child = container.querySelector(`[id=\"${id}\"]`)\n          if(child){ this.removeStreamChildElement(child) }\n        })\n      })\n\n      // clear stream items from the dead render if they are not inserted again\n      if(isJoinPatch){\n        DOM.all(this.container, `[${phxUpdate}=${PHX_STREAM}]`)\n          // it is important to filter the element before removing them, as\n          // it may happen that streams are nested and the owner check fails if\n          // a parent is removed before a child\n          .filter(el => this.view.ownsElement(el))\n          .forEach(el => {\n            Array.from(el.children).forEach(child => {\n              // we already performed the owner check, each child is guaranteed to be owned\n              // by the view. To prevent the nested owner check from failing in case of nested\n              // streams where the parent is removed before the child, we force the removal\n              this.removeStreamChildElement(child, true)\n            })\n          })\n      }\n\n      morph.call(this, targetContainer, html)\n    })\n\n    if(liveSocket.isDebugEnabled()){\n      detectDuplicateIds()\n      detectInvalidStreamInserts(this.streamInserts)\n      // warn if there are any inputs named \"id\"\n      Array.from(document.querySelectorAll(\"input[name=id]\")).forEach(node => {\n        if(node.form){\n          console.error(\"Detected an input with name=\\\"id\\\" inside a form! This will cause problems when patching the DOM.\\n\", node)\n        }\n      })\n    }\n\n    if(appendPrependUpdates.length > 0){\n      liveSocket.time(\"post-morph append/prepend restoration\", () => {\n        appendPrependUpdates.forEach(update => update.perform())\n      })\n    }\n\n    liveSocket.silenceEvents(() => DOM.restoreFocus(focused, selectionStart, selectionEnd))\n    DOM.dispatchEvent(document, \"phx:update\")\n    added.forEach(el => this.trackAfter(\"added\", el))\n    updates.forEach(el => this.trackAfter(\"updated\", el))\n\n    this.transitionPendingRemoves()\n\n    if(externalFormTriggered){\n      liveSocket.unload()\n      // check for submitter and inject it as hidden input for external submit;\n      // In theory, it could happen that the stored submitter is outdated and doesn't\n      // exist in the DOM any more, but this is unlikely, so we just accept it for now.\n      const submitter = DOM.private(externalFormTriggered, \"submitter\")\n      if(submitter && submitter.name && targetContainer.contains(submitter)){\n        const input = document.createElement(\"input\")\n        input.type = \"hidden\"\n        const formId = submitter.getAttribute(\"form\")\n        if(formId){\n          input.setAttribute(\"form\", formId)\n        }\n        input.name = submitter.name\n        input.value = submitter.value\n        submitter.parentElement.insertBefore(input, submitter)\n      }\n      // use prototype's submit in case there's a form control with name or id of \"submit\"\n      // https://developer.mozilla.org/en-US/docs/Web/API/HTMLFormElement/submit\n      Object.getPrototypeOf(externalFormTriggered).submit.call(externalFormTriggered)\n    }\n    return true\n  }\n\n  onNodeDiscarded(el){\n    // nested view handling\n    if(DOM.isPhxChild(el) || DOM.isPhxSticky(el)){ this.liveSocket.destroyViewByEl(el) }\n    this.trackAfter(\"discarded\", el)\n  }\n\n  maybePendingRemove(node){\n    if(node.getAttribute && node.getAttribute(this.phxRemove) !== null){\n      this.pendingRemoves.push(node)\n      return true\n    } else {\n      return false\n    }\n  }\n\n  removeStreamChildElement(child, force=false){\n    // make sure to only remove elements owned by the current view\n    // see https://github.com/phoenixframework/phoenix_live_view/issues/3047\n    // and https://github.com/phoenixframework/phoenix_live_view/issues/3681\n    if(!force && !this.view.ownsElement(child)){ return }\n\n    // we need to store the node if it is actually re-added in the same patch\n    // we do NOT want to execute phx-remove, we do NOT want to call onNodeDiscarded\n    if(this.streamInserts[child.id]){\n      this.streamComponentRestore[child.id] = child\n      child.remove()\n    } else {\n      // only remove the element now if it has no phx-remove binding\n      if(!this.maybePendingRemove(child)){\n        child.remove()\n        this.onNodeDiscarded(child)\n      }\n    }\n  }\n\n  getStreamInsert(el){\n    let insert = el.id ? this.streamInserts[el.id] : {}\n    return insert || {}\n  }\n\n  setStreamRef(el, ref){\n    DOM.putSticky(el, PHX_STREAM_REF, el => el.setAttribute(PHX_STREAM_REF, ref))\n  }\n\n  maybeReOrderStream(el, isNew){\n    let {ref, streamAt, reset} = this.getStreamInsert(el)\n    if(streamAt === undefined){ return }\n\n    // we need to set the PHX_STREAM_REF here as well as addChild is invoked only for parents\n    this.setStreamRef(el, ref)\n\n    if(!reset && !isNew){\n      // we only reorder if the element is new or it's a stream reset\n      return\n    }\n\n    // check if the element has a parent element;\n    // it doesn't if we are currently recursively morphing (restoring a saved stream child)\n    // because the element is not yet added to the real dom;\n    // reordering does not make sense in that case anyway\n    if(!el.parentElement){ return }\n\n    if(streamAt === 0){\n      el.parentElement.insertBefore(el, el.parentElement.firstElementChild)\n    } else if(streamAt > 0){\n      let children = Array.from(el.parentElement.children)\n      let oldIndex = children.indexOf(el)\n      if(streamAt >= children.length - 1){\n        el.parentElement.appendChild(el)\n      } else {\n        let sibling = children[streamAt]\n        if(oldIndex > streamAt){\n          el.parentElement.insertBefore(el, sibling)\n        } else {\n          el.parentElement.insertBefore(el, sibling.nextElementSibling)\n        }\n      }\n    }\n\n    this.maybeLimitStream(el)\n  }\n\n  maybeLimitStream(el){\n    let {limit} = this.getStreamInsert(el)\n    let children = limit !== null && Array.from(el.parentElement.children)\n    if(limit && limit < 0 && children.length > limit * -1){\n      children.slice(0, children.length + limit).forEach(child => this.removeStreamChildElement(child))\n    } else if(limit && limit >= 0 && children.length > limit){\n      children.slice(limit).forEach(child => this.removeStreamChildElement(child))\n    }\n  }\n\n  transitionPendingRemoves(){\n    let {pendingRemoves, liveSocket} = this\n    if(pendingRemoves.length > 0){\n      liveSocket.transitionRemoves(pendingRemoves, () => {\n        pendingRemoves.forEach(el => {\n          let child = DOM.firstPhxChild(el)\n          if(child){ liveSocket.destroyViewByEl(child) }\n          el.remove()\n        })\n        this.trackAfter(\"transitionsDiscarded\", pendingRemoves)\n      })\n    }\n  }\n\n  isChangedSelect(fromEl, toEl){\n    if(!(fromEl instanceof HTMLSelectElement) || fromEl.multiple){ return false }\n    if(fromEl.options.length !== toEl.options.length){ return true }\n\n    // keep the current value\n    toEl.value = fromEl.value\n\n    // in general we have to be very careful with using isEqualNode as it does not a reliable\n    // DOM tree equality check, but for selection attributes and options it works fine\n    return !fromEl.isEqualNode(toEl)\n  }\n\n  isCIDPatch(){ return this.cidPatch }\n\n  skipCIDSibling(el){\n    return el.nodeType === Node.ELEMENT_NODE && el.hasAttribute(PHX_SKIP)\n  }\n\n  targetCIDContainer(html){\n    if(!this.isCIDPatch()){ return }\n    let [first, ...rest] = DOM.findComponentNodeList(this.container, this.targetCID)\n    if(rest.length === 0 && DOM.childNodeLength(html) === 1){\n      return first\n    } else {\n      return first && first.parentNode\n    }\n  }\n\n  indexOf(parent, child){ return Array.from(parent.children).indexOf(child) }\n}\n", "import {\n  COMPONENTS,\n  DYNAMICS,\n  TEMPLATES,\n  EVENTS,\n  PHX_COMPONENT,\n  PHX_SKIP,\n  PHX_MAGIC_ID,\n  REPLY,\n  STATIC,\n  TITLE,\n  STREAM,\n  ROOT,\n} from \"./constants\"\n\nimport {\n  isObject,\n  logError,\n  isCid,\n} from \"./utils\"\n\nconst VOID_TAGS = new Set([\n  \"area\",\n  \"base\",\n  \"br\",\n  \"col\",\n  \"command\",\n  \"embed\",\n  \"hr\",\n  \"img\",\n  \"input\",\n  \"keygen\",\n  \"link\",\n  \"meta\",\n  \"param\",\n  \"source\",\n  \"track\",\n  \"wbr\"\n])\nconst quoteChars = new Set([\"'\", \"\\\"\"])\n\nexport let modifyRoot = (html, attrs, clearInnerHTML) => {\n  let i = 0\n  let insideComment = false\n  let beforeTag, afterTag, tag, tagNameEndsAt, id, newHTML\n\n  let lookahead = html.match(/^(\\s*(?:<!--.*?-->\\s*)*)<([^\\s\\/>]+)/)\n  if(lookahead === null){ throw new Error(`malformed html ${html}`) }\n\n  i = lookahead[0].length\n  beforeTag = lookahead[1]\n  tag = lookahead[2]\n  tagNameEndsAt = i\n\n  // Scan the opening tag for id, if there is any\n  for(i; i < html.length; i++){\n    if(html.charAt(i) === \">\" ){ break }\n    if(html.charAt(i) === \"=\"){\n      let isId = html.slice(i - 3, i) === \" id\"\n      i++\n      let char = html.charAt(i)\n      if(quoteChars.has(char)){\n        let attrStartsAt = i\n        i++\n        for(i; i < html.length; i++){\n          if(html.charAt(i) === char){ break }\n        }\n        if(isId){\n          id = html.slice(attrStartsAt + 1, i)\n          break\n        }\n      }\n    }\n  }\n\n  let closeAt = html.length - 1\n  insideComment = false\n  while(closeAt >= beforeTag.length + tag.length){\n    let char = html.charAt(closeAt)\n    if(insideComment){\n      if(char === \"-\" && html.slice(closeAt - 3, closeAt) === \"<!-\"){\n        insideComment = false\n        closeAt -= 4\n      } else {\n        closeAt -= 1\n      }\n    } else if(char === \">\" && html.slice(closeAt - 2, closeAt) === \"--\"){\n      insideComment = true\n      closeAt -= 3\n    } else if(char === \">\"){\n      break\n    } else {\n      closeAt -= 1\n    }\n  }\n  afterTag = html.slice(closeAt + 1, html.length)\n\n  let attrsStr =\n    Object.keys(attrs)\n      .map(attr => attrs[attr] === true ? attr : `${attr}=\"${attrs[attr]}\"`)\n      .join(\" \")\n\n  if(clearInnerHTML){\n    // Keep the id if any\n    let idAttrStr = id ? ` id=\"${id}\"` : \"\"\n    if(VOID_TAGS.has(tag)){\n      newHTML = `<${tag}${idAttrStr}${attrsStr === \"\" ? \"\" : \" \"}${attrsStr}/>`\n    } else {\n      newHTML = `<${tag}${idAttrStr}${attrsStr === \"\" ? \"\" : \" \"}${attrsStr}></${tag}>`\n    }\n  } else {\n    let rest = html.slice(tagNameEndsAt, closeAt + 1)\n    newHTML = `<${tag}${attrsStr === \"\" ? \"\" : \" \"}${attrsStr}${rest}`\n  }\n\n  return [newHTML, beforeTag, afterTag]\n}\n\nexport default class Rendered {\n  static extract(diff){\n    let {[REPLY]: reply, [EVENTS]: events, [TITLE]: title} = diff\n    delete diff[REPLY]\n    delete diff[EVENTS]\n    delete diff[TITLE]\n    return {diff, title, reply: reply || null, events: events || []}\n  }\n\n  constructor(viewId, rendered){\n    this.viewId = viewId\n    this.rendered = {}\n    this.magicId = 0\n    this.mergeDiff(rendered)\n  }\n\n  parentViewId(){ return this.viewId }\n\n  toString(onlyCids){\n    let [str, streams] = this.recursiveToString(this.rendered, this.rendered[COMPONENTS], onlyCids, true, {})\n    return [str, streams]\n  }\n\n  recursiveToString(rendered, components = rendered[COMPONENTS], onlyCids, changeTracking, rootAttrs){\n    onlyCids = onlyCids ? new Set(onlyCids) : null\n    let output = {buffer: \"\", components: components, onlyCids: onlyCids, streams: new Set()}\n    this.toOutputBuffer(rendered, null, output, changeTracking, rootAttrs)\n    return [output.buffer, output.streams]\n  }\n\n  componentCIDs(diff){ return Object.keys(diff[COMPONENTS] || {}).map(i => parseInt(i)) }\n\n  isComponentOnlyDiff(diff){\n    if(!diff[COMPONENTS]){ return false }\n    return Object.keys(diff).length === 1\n  }\n\n  getComponent(diff, cid){ return diff[COMPONENTS][cid] }\n\n  resetRender(cid){\n    // we are racing a component destroy, it could not exist, so\n    // make sure that we don't try to set reset on undefined\n    if(this.rendered[COMPONENTS][cid]){\n      this.rendered[COMPONENTS][cid].reset = true\n    }\n  }\n\n  mergeDiff(diff){\n    let newc = diff[COMPONENTS]\n    let cache = {}\n    delete diff[COMPONENTS]\n    this.rendered = this.mutableMerge(this.rendered, diff)\n    this.rendered[COMPONENTS] = this.rendered[COMPONENTS] || {}\n\n    if(newc){\n      let oldc = this.rendered[COMPONENTS]\n\n      for(let cid in newc){\n        newc[cid] = this.cachedFindComponent(cid, newc[cid], oldc, newc, cache)\n      }\n\n      for(let cid in newc){ oldc[cid] = newc[cid] }\n      diff[COMPONENTS] = newc\n    }\n  }\n\n  cachedFindComponent(cid, cdiff, oldc, newc, cache){\n    if(cache[cid]){\n      return cache[cid]\n    } else {\n      let ndiff, stat, scid = cdiff[STATIC]\n\n      if(isCid(scid)){\n        let tdiff\n\n        if(scid > 0){\n          tdiff = this.cachedFindComponent(scid, newc[scid], oldc, newc, cache)\n        } else {\n          tdiff = oldc[-scid]\n        }\n\n        stat = tdiff[STATIC]\n        ndiff = this.cloneMerge(tdiff, cdiff, true)\n        ndiff[STATIC] = stat\n      } else {\n        ndiff = cdiff[STATIC] !== undefined || oldc[cid] === undefined ?\n          cdiff : this.cloneMerge(oldc[cid], cdiff, false)\n      }\n\n      cache[cid] = ndiff\n      return ndiff\n    }\n  }\n\n  mutableMerge(target, source){\n    if(source[STATIC] !== undefined){\n      return source\n    } else {\n      this.doMutableMerge(target, source)\n      return target\n    }\n  }\n\n  doMutableMerge(target, source){\n    for(let key in source){\n      let val = source[key]\n      let targetVal = target[key]\n      let isObjVal = isObject(val)\n      if(isObjVal && val[STATIC] === undefined && isObject(targetVal)){\n        this.doMutableMerge(targetVal, val)\n      } else {\n        target[key] = val\n      }\n    }\n    if(target[ROOT]){\n      target.newRender = true\n    }\n  }\n\n  // Merges cid trees together, copying statics from source tree.\n  //\n  // The `pruneMagicId` is passed to control pruning the magicId of the\n  // target. We must always prune the magicId when we are sharing statics\n  // from another component. If not pruning, we replicate the logic from\n  // mutableMerge, where we set newRender to true if there is a root\n  // (effectively forcing the new version to be rendered instead of skipped)\n  //\n  cloneMerge(target, source, pruneMagicId){\n    let merged = {...target, ...source}\n    for(let key in merged){\n      let val = source[key]\n      let targetVal = target[key]\n      if(isObject(val) && val[STATIC] === undefined && isObject(targetVal)){\n        merged[key] = this.cloneMerge(targetVal, val, pruneMagicId)\n      } else if(val === undefined && isObject(targetVal)){\n        merged[key] = this.cloneMerge(targetVal, {}, pruneMagicId)\n      }\n    }\n    if(pruneMagicId){\n      delete merged.magicId\n      delete merged.newRender\n    } else if(target[ROOT]){\n      merged.newRender = true\n    }\n    return merged\n  }\n\n  componentToString(cid){\n    let [str, streams] = this.recursiveCIDToString(this.rendered[COMPONENTS], cid, null)\n    let [strippedHTML, _before, _after] = modifyRoot(str, {})\n    return [strippedHTML, streams]\n  }\n\n  pruneCIDs(cids){\n    cids.forEach(cid => delete this.rendered[COMPONENTS][cid])\n  }\n\n  // private\n\n  get(){ return this.rendered }\n\n  isNewFingerprint(diff = {}){ return !!diff[STATIC] }\n\n  templateStatic(part, templates){\n    if(typeof (part) === \"number\"){\n      return templates[part]\n    } else {\n      return part\n    }\n  }\n\n  nextMagicID(){\n    this.magicId++\n    return `m${this.magicId}-${this.parentViewId()}`\n  }\n\n  // Converts rendered tree to output buffer.\n  //\n  // changeTracking controls if we can apply the PHX_SKIP optimization.\n  // It is disabled for comprehensions since we must re-render the entire collection\n  // and no individual element is tracked inside the comprehension.\n  toOutputBuffer(rendered, templates, output, changeTracking, rootAttrs = {}){\n    if(rendered[DYNAMICS]){ return this.comprehensionToBuffer(rendered, templates, output) }\n    let {[STATIC]: statics} = rendered\n    statics = this.templateStatic(statics, templates)\n    let isRoot = rendered[ROOT]\n    let prevBuffer = output.buffer\n    if(isRoot){ output.buffer = \"\" }\n\n    // this condition is called when first rendering an optimizable function component.\n    // LC have their magicId previously set\n    if(changeTracking && isRoot && !rendered.magicId){\n      rendered.newRender = true\n      rendered.magicId = this.nextMagicID()\n    }\n\n    output.buffer += statics[0]\n    for(let i = 1; i < statics.length; i++){\n      this.dynamicToBuffer(rendered[i - 1], templates, output, changeTracking)\n      output.buffer += statics[i]\n    }\n\n    // Applies the root tag \"skip\" optimization if supported, which clears\n    // the root tag attributes and innerHTML, and only maintains the magicId.\n    // We can only skip when changeTracking is supported (outside of a comprehension),\n    // and when the root element hasn't experienced an unrendered merge (newRender true).\n    if(isRoot){\n      let skip = false\n      let attrs\n      // When a LC is re-added to the page, we need to re-render the entire LC tree,\n      // therefore changeTracking is false; however, we need to keep all the magicIds\n      // from any function component so the next time the LC is updated, we can apply\n      // the skip optimization\n      if(changeTracking || rendered.magicId){\n        skip = changeTracking && !rendered.newRender\n        attrs = {[PHX_MAGIC_ID]: rendered.magicId, ...rootAttrs}\n      } else {\n        attrs = rootAttrs\n      }\n      if(skip){ attrs[PHX_SKIP] = true }\n      let [newRoot, commentBefore, commentAfter] = modifyRoot(output.buffer, attrs, skip)\n      rendered.newRender = false\n      output.buffer = prevBuffer + commentBefore + newRoot + commentAfter\n    }\n  }\n\n  comprehensionToBuffer(rendered, templates, output){\n    let {[DYNAMICS]: dynamics, [STATIC]: statics, [STREAM]: stream} = rendered\n    let [_ref, _inserts, deleteIds, reset] = stream || [null, {}, [], null]\n    statics = this.templateStatic(statics, templates)\n    let compTemplates = templates || rendered[TEMPLATES]\n    for(let d = 0; d < dynamics.length; d++){\n      let dynamic = dynamics[d]\n      output.buffer += statics[0]\n      for(let i = 1; i < statics.length; i++){\n        // Inside a comprehension, we don't track how dynamics change\n        // over time (and features like streams would make that impossible\n        // unless we move the stream diffing away from morphdom),\n        // so we can't perform root change tracking.\n        let changeTracking = false\n        this.dynamicToBuffer(dynamic[i - 1], compTemplates, output, changeTracking)\n        output.buffer += statics[i]\n      }\n    }\n\n    if(stream !== undefined && (rendered[DYNAMICS].length > 0 || deleteIds.length > 0 || reset)){\n      delete rendered[STREAM]\n      rendered[DYNAMICS] = []\n      output.streams.add(stream)\n    }\n  }\n\n  dynamicToBuffer(rendered, templates, output, changeTracking){\n    if(typeof (rendered) === \"number\"){\n      let [str, streams] = this.recursiveCIDToString(output.components, rendered, output.onlyCids)\n      output.buffer += str\n      output.streams = new Set([...output.streams, ...streams])\n    } else if(isObject(rendered)){\n      this.toOutputBuffer(rendered, templates, output, changeTracking, {})\n    } else {\n      output.buffer += rendered\n    }\n  }\n\n  recursiveCIDToString(components, cid, onlyCids){\n    let component = components[cid] || logError(`no component for CID ${cid}`, components)\n    let attrs = {[PHX_COMPONENT]: cid}\n    let skip = onlyCids && !onlyCids.has(cid)\n    // Two optimization paths apply here:\n    //\n    //   1. The onlyCids optimization works by the server diff telling us only specific\n    //     cid's have changed. This allows us to skip rendering any component that hasn't changed,\n    //     which ultimately sets PHX_SKIP root attribute and avoids rendering the innerHTML.\n    //\n    //   2. The root PHX_SKIP optimization generalizes to all HEEx function components, and\n    //     works in the same PHX_SKIP attribute fashion as 1, but the newRender tracking is done\n    //     at the general diff merge level. If we merge a diff with new dynamics, we necessarily have\n    //     experienced a change which must be a newRender, and thus we can't skip the render.\n    //\n    // Both optimization flows apply here. newRender is set based on the onlyCids optimization, and\n    // we track a deterministic magicId based on the cid.\n    //\n    // changeTracking is about the entire tree\n    // newRender is about the current root in the tree\n    //\n    // By default changeTracking is enabled, but we special case the flow where the client is pruning\n    // cids and the server adds the component back. In such cases, we explicitly disable changeTracking\n    // with resetRender for this cid, then re-enable it after the recursive call to skip the optimization\n    // for the entire component tree.\n    component.newRender = !skip\n    component.magicId = `c${cid}-${this.parentViewId()}`\n    // enable change tracking as long as the component hasn't been reset\n    let changeTracking = !component.reset\n    let [html, streams] = this.recursiveToString(component, components, onlyCids, changeTracking, attrs)\n    // disable reset after we've rendered\n    delete component.reset\n\n    return [html, streams]\n  }\n}\n", "import DOM from \"./dom\"\nimport ARIA from \"./aria\"\n\nlet focusStack = []\nlet default_transition_time = 200\n\nlet JS = {\n  // private\n  exec(e, eventType, phxEvent, view, sourceEl, defaults){\n    let [defaultKind, defaultArgs] = defaults || [null, {callback: defaults && defaults.callback}]\n    let commands = phxEvent.charAt(0) === \"[\" ?\n      JSON.parse(phxEvent) : [[defaultKind, defaultArgs]]\n\n    commands.forEach(([kind, args]) => {\n      if(kind === defaultKind){\n        // always prefer the args, but keep existing keys from the defaultArgs\n        args = {...defaultArgs, ...args}\n        args.callback = args.callback || defaultArgs.callback\n      }\n      this.filterToEls(view.liveSocket, sourceEl, args).forEach(el => {\n        this[`exec_${kind}`](e, eventType, phxEvent, view, sourceEl, el, args)\n      })\n    })\n  },\n\n  isVisible(el){\n    return !!(el.offsetWidth || el.offsetHeight || el.getClientRects().length > 0)\n  },\n\n  // returns true if any part of the element is inside the viewport\n  isInViewport(el){\n    const rect = el.getBoundingClientRect()\n    const windowHeight = window.innerHeight || document.documentElement.clientHeight\n    const windowWidth = window.innerWidth || document.documentElement.clientWidth\n\n    return (\n      rect.right > 0 &&\n      rect.bottom > 0 &&\n      rect.left < windowWidth &&\n      rect.top < windowHeight\n    )\n  },\n\n  // private\n\n  // commands\n\n  exec_exec(e, eventType, phxEvent, view, sourceEl, el, {attr, to}){\n    let encodedJS = el.getAttribute(attr)\n    if(!encodedJS){ throw new Error(`expected ${attr} to contain JS command on \"${to}\"`) }\n    view.liveSocket.execJS(el, encodedJS, eventType)\n  },\n\n  exec_dispatch(e, eventType, phxEvent, view, sourceEl, el, {event, detail, bubbles}){\n    detail = detail || {}\n    detail.dispatcher = sourceEl\n    DOM.dispatchEvent(el, event, {detail, bubbles})\n  },\n\n  exec_push(e, eventType, phxEvent, view, sourceEl, el, args){\n    let {event, data, target, page_loading, loading, value, dispatcher, callback} = args\n    let pushOpts = {loading, value, target, page_loading: !!page_loading}\n    let targetSrc = eventType === \"change\" && dispatcher ? dispatcher : sourceEl\n    let phxTarget = target || targetSrc.getAttribute(view.binding(\"target\")) || targetSrc\n    const handler = (targetView, targetCtx) => {\n      if(!targetView.isConnected()){ return }\n      if(eventType === \"change\"){\n        let {newCid, _target} = args\n        _target = _target || (DOM.isFormInput(sourceEl) ? sourceEl.name : undefined)\n        if(_target){ pushOpts._target = _target }\n        targetView.pushInput(sourceEl, targetCtx, newCid, event || phxEvent, pushOpts, callback)\n      } else if(eventType === \"submit\"){\n        let {submitter} = args\n        targetView.submitForm(sourceEl, targetCtx, event || phxEvent, submitter, pushOpts, callback)\n      } else {\n        targetView.pushEvent(eventType, sourceEl, targetCtx, event || phxEvent, data, pushOpts, callback)\n      }\n    }\n    // in case of formRecovery, targetView and targetCtx are passed as argument\n    // as they are looked up in a template element, not the real DOM\n    if(args.targetView && args.targetCtx){\n      handler(args.targetView, args.targetCtx)\n    } else {\n      view.withinTargets(phxTarget, handler)\n    }\n  },\n\n  exec_navigate(e, eventType, phxEvent, view, sourceEl, el, {href, replace}){\n    view.liveSocket.historyRedirect(e, href, replace ? \"replace\" : \"push\", null, sourceEl)\n  },\n\n  exec_patch(e, eventType, phxEvent, view, sourceEl, el, {href, replace}){\n    view.liveSocket.pushHistoryPatch(e, href, replace ? \"replace\" : \"push\", sourceEl)\n  },\n\n  exec_focus(e, eventType, phxEvent, view, sourceEl, el){\n    ARIA.attemptFocus(el)\n    // in case the JS.focus command is in a JS.show/hide/toggle chain, for show we need\n    // to wait for JS.show to have updated the element's display property (see exec_toggle)\n    // but that run in nested animation frames, therefore we need to use them here as well\n    window.requestAnimationFrame(() => {\n      window.requestAnimationFrame(() => ARIA.attemptFocus(el))\n    })\n  },\n\n  exec_focus_first(e, eventType, phxEvent, view, sourceEl, el){\n    ARIA.focusFirstInteractive(el) || ARIA.focusFirst(el)\n    // if you wonder about the nested animation frames, see exec_focus\n    window.requestAnimationFrame(() => {\n      window.requestAnimationFrame(() => ARIA.focusFirstInteractive(el) || ARIA.focusFirst(el))\n    })\n  },\n\n  exec_push_focus(e, eventType, phxEvent, view, sourceEl, el){\n    focusStack.push(el || sourceEl)\n  },\n\n  exec_pop_focus(_e, _eventType, _phxEvent, _view, _sourceEl, _el){\n    const el = focusStack.pop()\n    if(el){\n      el.focus()\n      // if you wonder about the nested animation frames, see exec_focus\n      window.requestAnimationFrame(() => {\n        window.requestAnimationFrame(() => el.focus())\n      })\n    }\n  },\n\n  exec_add_class(e, eventType, phxEvent, view, sourceEl, el, {names, transition, time, blocking}){\n    this.addOrRemoveClasses(el, names, [], transition, time, view, blocking)\n  },\n\n  exec_remove_class(e, eventType, phxEvent, view, sourceEl, el, {names, transition, time, blocking}){\n    this.addOrRemoveClasses(el, [], names, transition, time, view, blocking)\n  },\n\n  exec_toggle_class(e, eventType, phxEvent, view, sourceEl, el, {names, transition, time, blocking}){\n    this.toggleClasses(el, names, transition, time, view, blocking)\n  },\n\n  exec_toggle_attr(e, eventType, phxEvent, view, sourceEl, el, {attr: [attr, val1, val2]}){\n    this.toggleAttr(el, attr, val1, val2)\n  },\n\n  exec_transition(e, eventType, phxEvent, view, sourceEl, el, {time, transition, blocking}){\n    this.addOrRemoveClasses(el, [], [], transition, time, view, blocking)\n  },\n\n  exec_toggle(e, eventType, phxEvent, view, sourceEl, el, {display, ins, outs, time, blocking}){\n    this.toggle(eventType, view, el, display, ins, outs, time, blocking)\n  },\n\n  exec_show(e, eventType, phxEvent, view, sourceEl, el, {display, transition, time, blocking}){\n    this.show(eventType, view, el, display, transition, time, blocking)\n  },\n\n  exec_hide(e, eventType, phxEvent, view, sourceEl, el, {display, transition, time, blocking}){\n    this.hide(eventType, view, el, display, transition, time, blocking)\n  },\n\n  exec_set_attr(e, eventType, phxEvent, view, sourceEl, el, {attr: [attr, val]}){\n    this.setOrRemoveAttrs(el, [[attr, val]], [])\n  },\n\n  exec_remove_attr(e, eventType, phxEvent, view, sourceEl, el, {attr}){\n    this.setOrRemoveAttrs(el, [], [attr])\n  },\n\n  // utils for commands\n\n  show(eventType, view, el, display, transition, time, blocking){\n    if(!this.isVisible(el)){\n      this.toggle(eventType, view, el, display, transition, null, time, blocking)\n    }\n  },\n\n  hide(eventType, view, el, display, transition, time, blocking){\n    if(this.isVisible(el)){\n      this.toggle(eventType, view, el, display, null, transition, time, blocking)\n    }\n  },\n\n  toggle(eventType, view, el, display, ins, outs, time, blocking){\n    time = time || default_transition_time\n    let [inClasses, inStartClasses, inEndClasses] = ins || [[], [], []]\n    let [outClasses, outStartClasses, outEndClasses] = outs || [[], [], []]\n    if(inClasses.length > 0 || outClasses.length > 0){\n      if(this.isVisible(el)){\n        let onStart = () => {\n          this.addOrRemoveClasses(el, outStartClasses, inClasses.concat(inStartClasses).concat(inEndClasses))\n          window.requestAnimationFrame(() => {\n            this.addOrRemoveClasses(el, outClasses, [])\n            window.requestAnimationFrame(() => this.addOrRemoveClasses(el, outEndClasses, outStartClasses))\n          })\n        }\n        let onEnd = () => {\n          this.addOrRemoveClasses(el, [], outClasses.concat(outEndClasses))\n          DOM.putSticky(el, \"toggle\", currentEl => currentEl.style.display = \"none\")\n          el.dispatchEvent(new Event(\"phx:hide-end\"))\n        }\n        el.dispatchEvent(new Event(\"phx:hide-start\"))\n        if(blocking === false){\n          onStart()\n          setTimeout(onEnd, time)\n        } else {\n          view.transition(time, onStart, onEnd)\n        }\n      } else {\n        if(eventType === \"remove\"){ return }\n        let onStart = () => {\n          this.addOrRemoveClasses(el, inStartClasses, outClasses.concat(outStartClasses).concat(outEndClasses))\n          const stickyDisplay = display || this.defaultDisplay(el)\n          window.requestAnimationFrame(() => {\n            // first add the starting + active class, THEN make the element visible\n            // otherwise if we toggled the visibility earlier css animations\n            // would flicker, as the element becomes visible before the active animation\n            // class is set (see https://github.com/phoenixframework/phoenix_live_view/issues/3456)\n            this.addOrRemoveClasses(el, inClasses, [])\n            // addOrRemoveClasses uses a requestAnimationFrame itself, therefore we need to move the putSticky\n            // into the next requestAnimationFrame...\n            window.requestAnimationFrame(() => {\n              DOM.putSticky(el, \"toggle\", currentEl => currentEl.style.display = stickyDisplay)\n              this.addOrRemoveClasses(el, inEndClasses, inStartClasses)\n            })\n          })\n        }\n        let onEnd = () => {\n          this.addOrRemoveClasses(el, [], inClasses.concat(inEndClasses))\n          el.dispatchEvent(new Event(\"phx:show-end\"))\n        }\n        el.dispatchEvent(new Event(\"phx:show-start\"))\n        if(blocking === false){\n          onStart()\n          setTimeout(onEnd, time)\n        } else {\n          view.transition(time, onStart, onEnd)\n        }\n      }\n    } else {\n      if(this.isVisible(el)){\n        window.requestAnimationFrame(() => {\n          el.dispatchEvent(new Event(\"phx:hide-start\"))\n          DOM.putSticky(el, \"toggle\", currentEl => currentEl.style.display = \"none\")\n          el.dispatchEvent(new Event(\"phx:hide-end\"))\n        })\n      } else {\n        window.requestAnimationFrame(() => {\n          el.dispatchEvent(new Event(\"phx:show-start\"))\n          let stickyDisplay = display || this.defaultDisplay(el)\n          DOM.putSticky(el, \"toggle\", currentEl => currentEl.style.display = stickyDisplay)\n          el.dispatchEvent(new Event(\"phx:show-end\"))\n        })\n      }\n    }\n  },\n\n  toggleClasses(el, classes, transition, time, view, blocking){\n    window.requestAnimationFrame(() => {\n      let [prevAdds, prevRemoves] = DOM.getSticky(el, \"classes\", [[], []])\n      let newAdds = classes.filter(name => prevAdds.indexOf(name) < 0 && !el.classList.contains(name))\n      let newRemoves = classes.filter(name => prevRemoves.indexOf(name) < 0 && el.classList.contains(name))\n      this.addOrRemoveClasses(el, newAdds, newRemoves, transition, time, view, blocking)\n    })\n  },\n\n  toggleAttr(el, attr, val1, val2){\n    if(el.hasAttribute(attr)){\n      if(val2 !== undefined){\n        // toggle between val1 and val2\n        if(el.getAttribute(attr) === val1){\n          this.setOrRemoveAttrs(el, [[attr, val2]], [])\n        } else {\n          this.setOrRemoveAttrs(el, [[attr, val1]], [])\n        }\n      } else {\n        // remove attr\n        this.setOrRemoveAttrs(el, [], [attr])\n      }\n    } else {\n      this.setOrRemoveAttrs(el, [[attr, val1]], [])\n    }\n  },\n\n  addOrRemoveClasses(el, adds, removes, transition, time, view, blocking){\n    time = time || default_transition_time\n    let [transitionRun, transitionStart, transitionEnd] = transition || [[], [], []]\n    if(transitionRun.length > 0){\n      let onStart = () => {\n        this.addOrRemoveClasses(el, transitionStart, [].concat(transitionRun).concat(transitionEnd))\n        window.requestAnimationFrame(() => {\n          this.addOrRemoveClasses(el, transitionRun, [])\n          window.requestAnimationFrame(() => this.addOrRemoveClasses(el, transitionEnd, transitionStart))\n        })\n      }\n      let onDone = () => this.addOrRemoveClasses(el, adds.concat(transitionEnd), removes.concat(transitionRun).concat(transitionStart))\n      if(blocking === false){\n        onStart()\n        setTimeout(onDone, time)\n      } else {\n        view.transition(time, onStart, onDone)\n      }\n      return\n    }\n\n    window.requestAnimationFrame(() => {\n      let [prevAdds, prevRemoves] = DOM.getSticky(el, \"classes\", [[], []])\n      let keepAdds = adds.filter(name => prevAdds.indexOf(name) < 0 && !el.classList.contains(name))\n      let keepRemoves = removes.filter(name => prevRemoves.indexOf(name) < 0 && el.classList.contains(name))\n      let newAdds = prevAdds.filter(name => removes.indexOf(name) < 0).concat(keepAdds)\n      let newRemoves = prevRemoves.filter(name => adds.indexOf(name) < 0).concat(keepRemoves)\n\n      DOM.putSticky(el, \"classes\", currentEl => {\n        currentEl.classList.remove(...newRemoves)\n        currentEl.classList.add(...newAdds)\n        return [newAdds, newRemoves]\n      })\n    })\n  },\n\n  setOrRemoveAttrs(el, sets, removes){\n    let [prevSets, prevRemoves] = DOM.getSticky(el, \"attrs\", [[], []])\n\n    let alteredAttrs = sets.map(([attr, _val]) => attr).concat(removes)\n    let newSets = prevSets.filter(([attr, _val]) => !alteredAttrs.includes(attr)).concat(sets)\n    let newRemoves = prevRemoves.filter((attr) => !alteredAttrs.includes(attr)).concat(removes)\n\n    DOM.putSticky(el, \"attrs\", currentEl => {\n      newRemoves.forEach(attr => currentEl.removeAttribute(attr))\n      newSets.forEach(([attr, val]) => currentEl.setAttribute(attr, val))\n      return [newSets, newRemoves]\n    })\n  },\n\n  hasAllClasses(el, classes){ return classes.every(name => el.classList.contains(name)) },\n\n  isToggledOut(el, outClasses){\n    return !this.isVisible(el) || this.hasAllClasses(el, outClasses)\n  },\n\n  filterToEls(liveSocket, sourceEl, {to}){\n    let defaultQuery = () => {\n      if(typeof(to) === \"string\"){\n        return document.querySelectorAll(to)\n      } else if(to.closest){\n        let toEl = sourceEl.closest(to.closest)\n        return toEl ? [toEl] : []\n      } else if(to.inner){\n        return sourceEl.querySelectorAll(to.inner)\n      }\n    }\n    return to ? liveSocket.jsQuerySelectorAll(sourceEl, to, defaultQuery) : [sourceEl]\n  },\n\n  defaultDisplay(el){\n    return {tr: \"table-row\", td: \"table-cell\"}[el.tagName.toLowerCase()] || \"block\"\n  },\n\n  transitionClasses(val){\n    if(!val){ return null }\n\n    let [trans, tStart, tEnd] = Array.isArray(val) ? val : [val.split(\" \"), [], []]\n    trans = Array.isArray(trans) ? trans : trans.split(\" \")\n    tStart = Array.isArray(tStart) ? tStart : tStart.split(\" \")\n    tEnd = Array.isArray(tEnd) ? tEnd : tEnd.split(\" \")\n    return [trans, tStart, tEnd]\n  }\n}\n\nexport default JS\n", "import JS from \"./js\"\nimport DOM from \"./dom\"\n\nconst HOOK_ID = \"hookId\"\n\nlet viewHookID = 1\nexport default class ViewHook {\n  static makeID(){ return viewHookID++ }\n  static elementID(el){ return DOM.private(el, HOOK_ID) }\n\n  constructor(view, el, callbacks){\n    this.el = el\n    this.__attachView(view)\n    this.__callbacks = callbacks\n    this.__listeners = new Set()\n    this.__isDisconnected = false\n    DOM.putPrivate(this.el, HOOK_ID, this.constructor.makeID())\n    for(let key in this.__callbacks){ this[key] = this.__callbacks[key] }\n  }\n\n  __attachView(view){\n    if(view){\n      this.__view = () => view\n      this.liveSocket = view.liveSocket\n    } else {\n      this.__view = () => {\n        throw new Error(`hook not yet attached to a live view: ${this.el.outerHTML}`)\n      }\n      this.liveSocket = null\n    }\n  }\n\n  __mounted(){ this.mounted && this.mounted() }\n  __updated(){ this.updated && this.updated() }\n  __beforeUpdate(){ this.beforeUpdate && this.beforeUpdate() }\n  __destroyed(){\n    this.destroyed && this.destroyed()\n    DOM.deletePrivate(this.el, HOOK_ID) // https://github.com/phoenixframework/phoenix_live_view/issues/3496\n  }\n  __reconnected(){\n    if(this.__isDisconnected){\n      this.__isDisconnected = false\n      this.reconnected && this.reconnected()\n    }\n  }\n  __disconnected(){\n    this.__isDisconnected = true\n    this.disconnected && this.disconnected()\n  }\n\n  /**\n   * Binds the hook to JS commands.\n   *\n   * @param {ViewHook} hook - The ViewHook instance to bind.\n   *\n   * @returns {Object} An object with methods to manipulate the DOM and execute JavaScript.\n   */\n  js(){\n    let hook = this\n\n    return {\n      /**\n       * Executes encoded JavaScript in the context of the hook element.\n       *\n       * @param {string} encodedJS - The encoded JavaScript string to execute.\n       */\n      exec(encodedJS){\n        hook.__view().liveSocket.execJS(hook.el, encodedJS, \"hook\")\n      },\n\n      /**\n       * Shows an element.\n       *\n       * @param {HTMLElement} el - The element to show.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.display] - The CSS display value to set. Defaults \"block\".\n       * @param {string} [opts.transition] - The CSS transition classes to set when showing.\n       * @param {number} [opts.time] - The transition duration in milliseconds. Defaults 200.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *  Defaults `true`.\n       */\n      show(el, opts = {}){\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.show(\"hook\", owner, el, opts.display, opts.transition, opts.time, opts.blocking)\n      },\n\n      /**\n       * Hides an element.\n       *\n       * @param {HTMLElement} el - The element to hide.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.transition] - The CSS transition classes to set when hiding.\n       * @param {number} [opts.time] - The transition duration in milliseconds. Defaults 200.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      hide(el, opts = {}){\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.hide(\"hook\", owner, el, null, opts.transition, opts.time, opts.blocking)\n      },\n\n      /**\n       * Toggles the visibility of an element.\n       *\n       * @param {HTMLElement} el - The element to toggle.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.display] - The CSS display value to set. Defaults \"block\".\n       * @param {string} [opts.in] - The CSS transition classes for showing.\n       *   Accepts either the string of classes to apply when toggling in, or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-0\", \"opacity-100\"]\n       *\n       * @param {string} [opts.out] - The CSS transition classes for hiding.\n       *   Accepts either string of classes to apply when toggling out, or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-100\", \"opacity-0\"]\n       *\n       * @param {number} [opts.time] - The transition duration in milliseconds.\n       *\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      toggle(el, opts = {}){\n        let owner = hook.__view().liveSocket.owner(el)\n        opts.in = JS.transitionClasses(opts.in)\n        opts.out = JS.transitionClasses(opts.out)\n        JS.toggle(\"hook\", owner, el, opts.display, opts.in, opts.out, opts.time, opts.blocking)\n      },\n\n      /**\n       * Adds CSS classes to an element.\n       *\n       * @param {HTMLElement} el - The element to add classes to.\n       * @param {string|string[]} names - The class name(s) to add.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.transition] - The CSS transition property to set.\n       *   Accepts a string of classes to apply when adding classes or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-0\", \"opacity-100\"]\n       *\n       * @param {number} [opts.time] - The transition duration in milliseconds.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      addClass(el, names, opts = {}){\n        names = Array.isArray(names) ? names : names.split(\" \")\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.addOrRemoveClasses(el, names, [], opts.transition, opts.time, owner, opts.blocking)\n      },\n\n      /**\n       * Removes CSS classes from an element.\n       *\n       * @param {HTMLElement} el - The element to remove classes from.\n       * @param {string|string[]} names - The class name(s) to remove.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.transition] - The CSS transition classes to set.\n       *   Accepts a string of classes to apply when removing classes or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-100\", \"opacity-0\"]\n       *\n       * @param {number} [opts.time] - The transition duration in milliseconds.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      removeClass(el, names, opts = {}){\n        opts.transition = JS.transitionClasses(opts.transition)\n        names = Array.isArray(names) ? names : names.split(\" \")\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.addOrRemoveClasses(el, [], names, opts.transition, opts.time, owner, opts.blocking)\n      },\n\n      /**\n       * Toggles CSS classes on an element.\n       *\n       * @param {HTMLElement} el - The element to toggle classes on.\n       * @param {string|string[]} names - The class name(s) to toggle.\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {string} [opts.transition] - The CSS transition classes to set.\n       *   Accepts a string of classes to apply when toggling classes or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-100\", \"opacity-0\"]\n       *\n       * @param {number} [opts.time] - The transition duration in milliseconds.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      toggleClass(el, names, opts = {}){\n        opts.transition = JS.transitionClasses(opts.transition)\n        names = Array.isArray(names) ? names : names.split(\" \")\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.toggleClasses(el, names, opts.transition, opts.time, owner, opts.blocking)\n      },\n\n      /**\n       * Applies a CSS transition to an element.\n       *\n       * @param {HTMLElement} el - The element to apply the transition to.\n       * @param {string|string[]} transition - The transition class(es) to apply.\n       *   Accepts a string of classes to apply when transitioning or\n       *   a 3-tuple containing the transition class, the class to apply\n       *   to start the transition, and the ending transition class, such as:\n       *\n       *       [\"ease-out duration-300\", \"opacity-100\", \"opacity-0\"]\n       *\n       * @param {Object} [opts={}] - Optional settings.\n       * @param {number} [opts.time] - The transition duration in milliseconds.\n       * @param {boolean} [opts.blocking] - The boolean flag to block the UI during the transition.\n       *   Defaults `true`.\n       */\n      transition(el, transition, opts = {}){\n        let owner = hook.__view().liveSocket.owner(el)\n        JS.addOrRemoveClasses(el, [], [], JS.transitionClasses(transition), opts.time, owner, opts.blocking)\n      },\n\n      /**\n       * Sets an attribute on an element.\n       *\n       * @param {HTMLElement} el - The element to set the attribute on.\n       * @param {string} attr - The attribute name to set.\n       * @param {string} val - The value to set for the attribute.\n       */\n      setAttribute(el, attr, val){ JS.setOrRemoveAttrs(el, [[attr, val]], []) },\n\n      /**\n       * Removes an attribute from an element.\n       *\n       * @param {HTMLElement} el - The element to remove the attribute from.\n       * @param {string} attr - The attribute name to remove.\n       */\n      removeAttribute(el, attr){ JS.setOrRemoveAttrs(el, [], [attr]) },\n\n      /**\n       * Toggles an attribute on an element between two values.\n       *\n       * @param {HTMLElement} el - The element to toggle the attribute on.\n       * @param {string} attr - The attribute name to toggle.\n       * @param {string} val1 - The first value to toggle between.\n       * @param {string} val2 - The second value to toggle between.\n       */\n      toggleAttribute(el, attr, val1, val2){ JS.toggleAttr(el, attr, val1, val2) },\n    }\n  }\n\n  pushEvent(event, payload = {}, onReply){\n    if(onReply === undefined){\n      return new Promise((resolve, reject) => {\n        try {\n          const ref = this.__view().pushHookEvent(this.el, null, event, payload, (reply, _ref) => resolve(reply))\n          if(ref === false){\n            reject(new Error(\"unable to push hook event. LiveView not connected\"))\n          }\n        } catch (error){\n          reject(error)\n        }\n      })\n    }\n    return this.__view().pushHookEvent(this.el, null, event, payload, onReply)\n  }\n\n  pushEventTo(phxTarget, event, payload = {}, onReply){\n    if(onReply === undefined){\n      return new Promise((resolve, reject) => {\n        try {\n          this.__view().withinTargets(phxTarget, (view, targetCtx) => {\n            const ref = view.pushHookEvent(this.el, targetCtx, event, payload, (reply, _ref) => resolve(reply))\n            if(ref === false){\n              reject(new Error(\"unable to push hook event. LiveView not connected\"))\n            }\n          })\n        } catch (error){\n          reject(error)\n        }\n      })\n    }\n    return this.__view().withinTargets(phxTarget, (view, targetCtx) => {\n      return view.pushHookEvent(this.el, targetCtx, event, payload, onReply)\n    })\n  }\n\n  handleEvent(event, callback){\n    let callbackRef = (customEvent, bypass) => bypass ? event : callback(customEvent.detail)\n    window.addEventListener(`phx:${event}`, callbackRef)\n    this.__listeners.add(callbackRef)\n    return callbackRef\n  }\n\n  removeHandleEvent(callbackRef){\n    let event = callbackRef(null, true)\n    window.removeEventListener(`phx:${event}`, callbackRef)\n    this.__listeners.delete(callbackRef)\n  }\n\n  upload(name, files){\n    return this.__view().dispatchUploads(null, name, files)\n  }\n\n  uploadTo(phxTarget, name, files){\n    return this.__view().withinTargets(phxTarget, (view, targetCtx) => {\n      view.dispatchUploads(targetCtx, name, files)\n    })\n  }\n\n  __cleanup__(){\n    this.__listeners.forEach(callbackRef => this.removeHandleEvent(callbackRef))\n  }\n}\n", "import {\n  BEFORE_UNLOAD_LOADER_TIMEOUT,\n  CHECKABLE_INPUTS,\n  CONSECUTIVE_RELOADS,\n  PHX_AUTO_RECOVER,\n  PHX_COMPONENT,\n  PHX_CONNECTED_CLASS,\n  PHX_DISABLE_WITH,\n  PHX_DISABLE_WITH_RESTORE,\n  PHX_DISABLED,\n  PHX_LOADING_CLASS,\n  PHX_ERROR_CLASS,\n  PHX_CLIENT_ERROR_CLASS,\n  PHX_SERVER_ERROR_CLASS,\n  PHX_HAS_FOCUSED,\n  PHX_HAS_SUBMITTED,\n  PHX_HOOK,\n  PHX_PARENT_ID,\n  PHX_PROGRESS,\n  PHX_READONLY,\n  PHX_REF_LOADING,\n  PHX_REF_SRC,\n  PHX_REF_LOCK,\n  PHX_ROOT_ID,\n  PHX_SESSION,\n  PHX_STATIC,\n  PHX_STICKY,\n  PHX_TRACK_STATIC,\n  PHX_TRACK_UPLOADS,\n  PHX_UPDATE,\n  PHX_UPLOAD_REF,\n  PHX_VIEW_SELECTOR,\n  PHX_MAIN,\n  PHX_MOUNTED,\n  PUSH_TIMEOUT,\n  PHX_VIEWPORT_TOP,\n  PHX_VIEWPORT_BOTTOM,\n  MAX_CHILD_JOIN_ATTEMPTS\n} from \"./constants\"\n\nimport {\n  clone,\n  closestPhxBinding,\n  isEmpty,\n  isEqualObj,\n  logError,\n  maybe,\n  isCid,\n} from \"./utils\"\n\nimport Browser from \"./browser\"\nimport DOM from \"./dom\"\nimport ElementRef from \"./element_ref\"\nimport DOMPatch from \"./dom_patch\"\nimport LiveUploader from \"./live_uploader\"\nimport Rendered from \"./rendered\"\nimport ViewHook from \"./view_hook\"\nimport JS from \"./js\"\n\nimport morphdom from \"morphdom\"\n\nexport let prependFormDataKey = (key, prefix) => {\n  let isArray = key.endsWith(\"[]\")\n  // Remove the \"[]\" if it's an array\n  let baseKey = isArray ? key.slice(0, -2) : key\n  // Replace last occurrence of key before a closing bracket or the end with key plus suffix\n  baseKey = baseKey.replace(/([^\\[\\]]+)(\\]?$)/, `${prefix}$1$2`)\n  // Add back the \"[]\" if it was an array\n  if(isArray){ baseKey += \"[]\" }\n  return baseKey\n}\n\nlet serializeForm = (form, opts, onlyNames = []) => {\n  const {submitter} = opts\n\n  // We must inject the submitter in the order that it exists in the DOM\n  // relative to other inputs. For example, for checkbox groups, the order must be maintained.\n  let injectedElement\n  if(submitter && submitter.name){\n    const input = document.createElement(\"input\")\n    input.type = \"hidden\"\n    // set the form attribute if the submitter has one;\n    // this can happen if the element is outside the actual form element\n    const formId = submitter.getAttribute(\"form\")\n    if(formId){\n      input.setAttribute(\"form\", formId)\n    }\n    input.name = submitter.name\n    input.value = submitter.value\n    submitter.parentElement.insertBefore(input, submitter)\n    injectedElement = input\n  }\n\n  const formData = new FormData(form)\n  const toRemove = []\n\n  formData.forEach((val, key, _index) => {\n    if(val instanceof File){ toRemove.push(key) }\n  })\n\n  // Cleanup after building fileData\n  toRemove.forEach(key => formData.delete(key))\n\n  const params = new URLSearchParams()\n\n  const {inputsUnused, onlyHiddenInputs} = Array.from(form.elements).reduce((acc, input) => {\n    const {inputsUnused, onlyHiddenInputs} = acc\n    const key = input.name\n    if(!key){ return acc }\n\n    if(inputsUnused[key] === undefined){ inputsUnused[key] = true }\n    if(onlyHiddenInputs[key] === undefined){ onlyHiddenInputs[key] = true }\n\n    const isUsed = DOM.private(input, PHX_HAS_FOCUSED) || DOM.private(input, PHX_HAS_SUBMITTED)\n    const isHidden = input.type === \"hidden\"\n    inputsUnused[key] = inputsUnused[key] && !isUsed\n    onlyHiddenInputs[key] = onlyHiddenInputs[key] && isHidden\n\n    return acc\n  }, {inputsUnused: {}, onlyHiddenInputs: {}})\n\n  for(let [key, val] of formData.entries()){\n    if(onlyNames.length === 0 || onlyNames.indexOf(key) >= 0){\n      let isUnused = inputsUnused[key]\n      let hidden = onlyHiddenInputs[key]\n      if(isUnused && !(submitter && submitter.name == key) && !hidden){\n        params.append(prependFormDataKey(key, \"_unused_\"), \"\")\n      }\n      params.append(key, val)\n    }\n  }\n\n  // remove the injected element again\n  // (it would be removed by the next dom patch anyway, but this is cleaner)\n  if(submitter && injectedElement){\n    submitter.parentElement.removeChild(injectedElement)\n  }\n\n  return params.toString()\n}\n\nexport default class View {\n  static closestView(el){\n    let liveViewEl = el.closest(PHX_VIEW_SELECTOR)\n    return liveViewEl ? DOM.private(liveViewEl, \"view\") : null\n  }\n\n  constructor(el, liveSocket, parentView, flash, liveReferer){\n    this.isDead = false\n    this.liveSocket = liveSocket\n    this.flash = flash\n    this.parent = parentView\n    this.root = parentView ? parentView.root : this\n    this.el = el\n    DOM.putPrivate(this.el, \"view\", this)\n    this.id = this.el.id\n    this.ref = 0\n    this.lastAckRef = null\n    this.childJoins = 0\n    this.loaderTimer = null\n    this.disconnectedTimer = null\n    this.pendingDiffs = []\n    this.pendingForms = new Set()\n    this.redirect = false\n    this.href = null\n    this.joinCount = this.parent ? this.parent.joinCount - 1 : 0\n    this.joinAttempts = 0\n    this.joinPending = true\n    this.destroyed = false\n    this.joinCallback = function(onDone){ onDone && onDone() }\n    this.stopCallback = function(){ }\n    this.pendingJoinOps = this.parent ? null : []\n    this.viewHooks = {}\n    this.formSubmits = []\n    this.children = this.parent ? null : {}\n    this.root.children[this.id] = {}\n    this.formsForRecovery = {}\n    this.channel = this.liveSocket.channel(`lv:${this.id}`, () => {\n      let url = this.href && this.expandURL(this.href)\n      return {\n        redirect: this.redirect ? url : undefined,\n        url: this.redirect ? undefined : url || undefined,\n        params: this.connectParams(liveReferer),\n        session: this.getSession(),\n        static: this.getStatic(),\n        flash: this.flash,\n        sticky: this.el.hasAttribute(PHX_STICKY)\n      }\n    })\n  }\n\n  setHref(href){ this.href = href }\n\n  setRedirect(href){\n    this.redirect = true\n    this.href = href\n  }\n\n  isMain(){ return this.el.hasAttribute(PHX_MAIN) }\n\n  connectParams(liveReferer){\n    let params = this.liveSocket.params(this.el)\n    let manifest =\n      DOM.all(document, `[${this.binding(PHX_TRACK_STATIC)}]`)\n        .map(node => node.src || node.href).filter(url => typeof (url) === \"string\")\n\n    if(manifest.length > 0){ params[\"_track_static\"] = manifest }\n    params[\"_mounts\"] = this.joinCount\n    params[\"_mount_attempts\"] = this.joinAttempts\n    params[\"_live_referer\"] = liveReferer\n    this.joinAttempts++\n\n    return params\n  }\n\n  isConnected(){ return this.channel.canPush() }\n\n  getSession(){ return this.el.getAttribute(PHX_SESSION) }\n\n  getStatic(){\n    let val = this.el.getAttribute(PHX_STATIC)\n    return val === \"\" ? null : val\n  }\n\n  destroy(callback = function (){ }){\n    this.destroyAllChildren()\n    this.destroyed = true\n    delete this.root.children[this.id]\n    if(this.parent){ delete this.root.children[this.parent.id][this.id] }\n    clearTimeout(this.loaderTimer)\n    let onFinished = () => {\n      callback()\n      for(let id in this.viewHooks){\n        this.destroyHook(this.viewHooks[id])\n      }\n    }\n\n    DOM.markPhxChildDestroyed(this.el)\n\n    this.log(\"destroyed\", () => [\"the child has been removed from the parent\"])\n    this.channel.leave()\n      .receive(\"ok\", onFinished)\n      .receive(\"error\", onFinished)\n      .receive(\"timeout\", onFinished)\n  }\n\n  setContainerClasses(...classes){\n    this.el.classList.remove(\n      PHX_CONNECTED_CLASS,\n      PHX_LOADING_CLASS,\n      PHX_ERROR_CLASS,\n      PHX_CLIENT_ERROR_CLASS,\n      PHX_SERVER_ERROR_CLASS\n    )\n    this.el.classList.add(...classes)\n  }\n\n  showLoader(timeout){\n    clearTimeout(this.loaderTimer)\n    if(timeout){\n      this.loaderTimer = setTimeout(() => this.showLoader(), timeout)\n    } else {\n      for(let id in this.viewHooks){ this.viewHooks[id].__disconnected() }\n      this.setContainerClasses(PHX_LOADING_CLASS)\n    }\n  }\n\n  execAll(binding){\n    DOM.all(this.el, `[${binding}]`, el => this.liveSocket.execJS(el, el.getAttribute(binding)))\n  }\n\n  hideLoader(){\n    clearTimeout(this.loaderTimer)\n    clearTimeout(this.disconnectedTimer)\n    this.setContainerClasses(PHX_CONNECTED_CLASS)\n    this.execAll(this.binding(\"connected\"))\n  }\n\n  triggerReconnected(){\n    for(let id in this.viewHooks){ this.viewHooks[id].__reconnected() }\n  }\n\n  log(kind, msgCallback){\n    this.liveSocket.log(this, kind, msgCallback)\n  }\n\n  transition(time, onStart, onDone = function(){}){\n    this.liveSocket.transition(time, onStart, onDone)\n  }\n\n  // calls the callback with the view and target element for the given phxTarget\n  // targets can be:\n  //  * an element itself, then it is simply passed to liveSocket.owner;\n  //  * a CID (Component ID), then we first search the component's element in the DOM\n  //  * a selector, then we search the selector in the DOM and call the callback\n  //    for each element found with the corresponding owner view\n  withinTargets(phxTarget, callback, dom = document, viewEl){\n    // in the form recovery case we search in a template fragment instead of\n    // the real dom, therefore we optionally pass dom and viewEl\n\n    if(phxTarget instanceof HTMLElement || phxTarget instanceof SVGElement){\n      return this.liveSocket.owner(phxTarget, view => callback(view, phxTarget))\n    }\n\n    if(isCid(phxTarget)){\n      let targets = DOM.findComponentNodeList(viewEl || this.el, phxTarget)\n      if(targets.length === 0){\n        logError(`no component found matching phx-target of ${phxTarget}`)\n      } else {\n        callback(this, parseInt(phxTarget))\n      }\n    } else {\n      let targets = Array.from(dom.querySelectorAll(phxTarget))\n      if(targets.length === 0){ logError(`nothing found matching the phx-target selector \"${phxTarget}\"`) }\n      targets.forEach(target => this.liveSocket.owner(target, view => callback(view, target)))\n    }\n  }\n\n  applyDiff(type, rawDiff, callback){\n    this.log(type, () => [\"\", clone(rawDiff)])\n    let {diff, reply, events, title} = Rendered.extract(rawDiff)\n    callback({diff, reply, events})\n    if(typeof title === \"string\" || type == \"mount\"){ window.requestAnimationFrame(() => DOM.putTitle(title)) }\n  }\n\n  onJoin(resp){\n    let {rendered, container, liveview_version} = resp\n    if(container){\n      let [tag, attrs] = container\n      this.el = DOM.replaceRootContainer(this.el, tag, attrs)\n    }\n    this.childJoins = 0\n    this.joinPending = true\n    this.flash = null\n    if(this.root === this){\n      this.formsForRecovery = this.getFormsForRecovery()\n    }\n    if(this.isMain() && window.history.state === null){\n      // set initial history entry if this is the first page load (no history)\n      Browser.pushState(\"replace\", {\n        type: \"patch\",\n        id: this.id,\n        position: this.liveSocket.currentHistoryPosition\n      })\n    }\n\n    if(liveview_version !== this.liveSocket.version()){\n      console.error(`LiveView asset version mismatch. JavaScript version ${this.liveSocket.version()} vs. server ${liveview_version}. To avoid issues, please ensure that your assets use the same version as the server.`)\n    }\n\n    Browser.dropLocal(this.liveSocket.localStorage, window.location.pathname, CONSECUTIVE_RELOADS)\n    this.applyDiff(\"mount\", rendered, ({diff, events}) => {\n      this.rendered = new Rendered(this.id, diff)\n      let [html, streams] = this.renderContainer(null, \"join\")\n      this.dropPendingRefs()\n      this.joinCount++\n      this.joinAttempts = 0\n\n      this.maybeRecoverForms(html, () => {\n        this.onJoinComplete(resp, html, streams, events)\n      })\n    })\n  }\n\n  dropPendingRefs(){\n    DOM.all(document, `[${PHX_REF_SRC}=\"${this.refSrc()}\"]`, el => {\n      el.removeAttribute(PHX_REF_LOADING)\n      el.removeAttribute(PHX_REF_SRC)\n      el.removeAttribute(PHX_REF_LOCK)\n    })\n  }\n\n  onJoinComplete({live_patch}, html, streams, events){\n    // In order to provide a better experience, we want to join\n    // all LiveViews first and only then apply their patches.\n    if(this.joinCount > 1 || (this.parent && !this.parent.isJoinPending())){\n      return this.applyJoinPatch(live_patch, html, streams, events)\n    }\n\n    // One downside of this approach is that we need to find phxChildren\n    // in the html fragment, instead of directly on the DOM. The fragment\n    // also does not include PHX_STATIC, so we need to copy it over from\n    // the DOM.\n    let newChildren = DOM.findPhxChildrenInFragment(html, this.id).filter(toEl => {\n      let fromEl = toEl.id && this.el.querySelector(`[id=\"${toEl.id}\"]`)\n      let phxStatic = fromEl && fromEl.getAttribute(PHX_STATIC)\n      if(phxStatic){ toEl.setAttribute(PHX_STATIC, phxStatic) }\n      // set PHX_ROOT_ID to prevent events from being dispatched to the root view\n      // while the child join is still pending\n      if(fromEl){ fromEl.setAttribute(PHX_ROOT_ID, this.root.id) }\n      return this.joinChild(toEl)\n    })\n\n    if(newChildren.length === 0){\n      if(this.parent){\n        this.root.pendingJoinOps.push([this, () => this.applyJoinPatch(live_patch, html, streams, events)])\n        this.parent.ackJoin(this)\n      } else {\n        this.onAllChildJoinsComplete()\n        this.applyJoinPatch(live_patch, html, streams, events)\n      }\n    } else {\n      this.root.pendingJoinOps.push([this, () => this.applyJoinPatch(live_patch, html, streams, events)])\n    }\n  }\n\n  attachTrueDocEl(){\n    this.el = DOM.byId(this.id)\n    this.el.setAttribute(PHX_ROOT_ID, this.root.id)\n  }\n\n  // this is invoked for dead and live views, so we must filter by\n  // by owner to ensure we aren't duplicating hooks across disconnect\n  // and connected states. This also handles cases where hooks exist\n  // in a root layout with a LV in the body\n  execNewMounted(parent = this.el){\n    let phxViewportTop = this.binding(PHX_VIEWPORT_TOP)\n    let phxViewportBottom = this.binding(PHX_VIEWPORT_BOTTOM)\n    DOM.all(parent, `[${phxViewportTop}], [${phxViewportBottom}]`, hookEl => {\n      if(this.ownsElement(hookEl)){\n        DOM.maintainPrivateHooks(hookEl, hookEl, phxViewportTop, phxViewportBottom)\n        this.maybeAddNewHook(hookEl)\n      }\n    })\n    DOM.all(parent, `[${this.binding(PHX_HOOK)}], [data-phx-${PHX_HOOK}]`, hookEl => {\n      if(this.ownsElement(hookEl)){\n        this.maybeAddNewHook(hookEl)\n      }\n    })\n    DOM.all(parent, `[${this.binding(PHX_MOUNTED)}]`, el => {\n      if(this.ownsElement(el)){\n        this.maybeMounted(el)\n      }\n    })\n  }\n\n  applyJoinPatch(live_patch, html, streams, events){\n    this.attachTrueDocEl()\n    let patch = new DOMPatch(this, this.el, this.id, html, streams, null)\n    patch.markPrunableContentForRemoval()\n    this.performPatch(patch, false, true)\n    this.joinNewChildren()\n    this.execNewMounted()\n\n    this.joinPending = false\n    this.liveSocket.dispatchEvents(events)\n    this.applyPendingUpdates()\n\n    if(live_patch){\n      let {kind, to} = live_patch\n      this.liveSocket.historyPatch(to, kind)\n    }\n    this.hideLoader()\n    if(this.joinCount > 1){ this.triggerReconnected() }\n    this.stopCallback()\n  }\n\n  triggerBeforeUpdateHook(fromEl, toEl){\n    this.liveSocket.triggerDOM(\"onBeforeElUpdated\", [fromEl, toEl])\n    let hook = this.getHook(fromEl)\n    let isIgnored = hook && DOM.isIgnored(fromEl, this.binding(PHX_UPDATE))\n    if(hook && !fromEl.isEqualNode(toEl) && !(isIgnored && isEqualObj(fromEl.dataset, toEl.dataset))){\n      hook.__beforeUpdate()\n      return hook\n    }\n  }\n\n  maybeMounted(el){\n    let phxMounted = el.getAttribute(this.binding(PHX_MOUNTED))\n    let hasBeenInvoked = phxMounted && DOM.private(el, \"mounted\")\n    if(phxMounted && !hasBeenInvoked){\n      this.liveSocket.execJS(el, phxMounted)\n      DOM.putPrivate(el, \"mounted\", true)\n    }\n  }\n\n  maybeAddNewHook(el){\n    let newHook = this.addHook(el)\n    if(newHook){ newHook.__mounted() }\n  }\n\n  performPatch(patch, pruneCids, isJoinPatch = false){\n    let removedEls = []\n    let phxChildrenAdded = false\n    let updatedHookIds = new Set()\n\n    this.liveSocket.triggerDOM(\"onPatchStart\", [patch.targetContainer])\n\n    patch.after(\"added\", el => {\n      this.liveSocket.triggerDOM(\"onNodeAdded\", [el])\n      let phxViewportTop = this.binding(PHX_VIEWPORT_TOP)\n      let phxViewportBottom = this.binding(PHX_VIEWPORT_BOTTOM)\n      DOM.maintainPrivateHooks(el, el, phxViewportTop, phxViewportBottom)\n      this.maybeAddNewHook(el)\n      if(el.getAttribute){ this.maybeMounted(el) }\n    })\n\n    patch.after(\"phxChildAdded\", el => {\n      if(DOM.isPhxSticky(el)){\n        this.liveSocket.joinRootViews()\n      } else {\n        phxChildrenAdded = true\n      }\n    })\n\n    patch.before(\"updated\", (fromEl, toEl) => {\n      let hook = this.triggerBeforeUpdateHook(fromEl, toEl)\n      if(hook){ updatedHookIds.add(fromEl.id) }\n    })\n\n    patch.after(\"updated\", el => {\n      if(updatedHookIds.has(el.id)){ this.getHook(el).__updated() }\n    })\n\n    patch.after(\"discarded\", (el) => {\n      if(el.nodeType === Node.ELEMENT_NODE){ removedEls.push(el) }\n    })\n\n    patch.after(\"transitionsDiscarded\", els => this.afterElementsRemoved(els, pruneCids))\n    patch.perform(isJoinPatch)\n    this.afterElementsRemoved(removedEls, pruneCids)\n\n    this.liveSocket.triggerDOM(\"onPatchEnd\", [patch.targetContainer])\n    return phxChildrenAdded\n  }\n\n  afterElementsRemoved(elements, pruneCids){\n    let destroyedCIDs = []\n    elements.forEach(parent => {\n      let components = DOM.all(parent, `[${PHX_COMPONENT}]`)\n      let hooks = DOM.all(parent, `[${this.binding(PHX_HOOK)}], [data-phx-hook]`)\n      components.concat(parent).forEach(el => {\n        let cid = this.componentID(el)\n        if(isCid(cid) && destroyedCIDs.indexOf(cid) === -1){ destroyedCIDs.push(cid) }\n      })\n      hooks.concat(parent).forEach(hookEl => {\n        let hook = this.getHook(hookEl)\n        hook && this.destroyHook(hook)\n      })\n    })\n    // We should not pruneCids on joins. Otherwise, in case of\n    // rejoins, we may notify cids that no longer belong to the\n    // current LiveView to be removed.\n    if(pruneCids){\n      this.maybePushComponentsDestroyed(destroyedCIDs)\n    }\n  }\n\n  joinNewChildren(){\n    DOM.findPhxChildren(this.el, this.id).forEach(el => this.joinChild(el))\n  }\n\n  maybeRecoverForms(html, callback){\n    const phxChange = this.binding(\"change\")\n    const oldForms = this.root.formsForRecovery\n    // So why do we create a template element here?\n    // One way to recover forms would be to immediately apply the mount\n    // patch and then afterwards recover the forms. However, this would\n    // cause a flicker, because the mount patch would remove the form content\n    // until it is restored. Therefore LV decided to do form recovery with the\n    // raw HTML before it is applied and delay the mount patch until the form\n    // recovery events are done.\n    let template = document.createElement(\"template\")\n    template.innerHTML = html\n    // because we work with a template element, we must manually copy the attributes\n    // otherwise the owner / target helpers don't work properly\n    const rootEl = template.content.firstElementChild\n    rootEl.id = this.id\n    rootEl.setAttribute(PHX_ROOT_ID, this.root.id)\n    rootEl.setAttribute(PHX_SESSION, this.getSession())\n    rootEl.setAttribute(PHX_STATIC, this.getStatic())\n    rootEl.setAttribute(PHX_PARENT_ID, this.parent ? this.parent.id : null)\n\n    // we go over all form elements in the new HTML for the LV\n    // and look for old forms in the `formsForRecovery` object;\n    // the formsForRecovery can also contain forms from child views\n    const formsToRecover =\n      // we go over all forms in the new DOM; because this is only the HTML for the current\n      // view, we can be sure that all forms are owned by this view:\n      DOM.all(template.content, \"form\")\n        // only recover forms that have an id and are in the old DOM\n        .filter(newForm => newForm.id && oldForms[newForm.id])\n        // abandon forms we already tried to recover to prevent looping a failed state\n        .filter(newForm => !this.pendingForms.has(newForm.id))\n        // only recover if the form has the same phx-change value\n        .filter(newForm => oldForms[newForm.id].getAttribute(phxChange) === newForm.getAttribute(phxChange))\n        .map(newForm => {\n          return [oldForms[newForm.id], newForm]\n        })\n\n    if(formsToRecover.length === 0){\n      return callback()\n    }\n\n    formsToRecover.forEach(([oldForm, newForm], i) => {\n      this.pendingForms.add(newForm.id)\n      // it is important to use the firstElementChild of the template content\n      // because when traversing a documentFragment using parentNode, we won't ever arrive at\n      // the fragment; as the template is always a LiveView, we can be sure that there is only\n      // one child on the root level\n      this.pushFormRecovery(oldForm, newForm, template.content.firstElementChild, () => {\n        this.pendingForms.delete(newForm.id)\n        // we only call the callback once all forms have been recovered\n        if(i === formsToRecover.length - 1){\n          callback()\n        }\n      })\n    })\n  }\n\n  getChildById(id){ return this.root.children[this.id][id] }\n\n  getDescendentByEl(el){\n    if(el.id === this.id){\n      return this\n    } else {\n      return this.children[el.getAttribute(PHX_PARENT_ID)]?.[el.id]\n    }\n  }\n\n  destroyDescendent(id){\n    for(let parentId in this.root.children){\n      for(let childId in this.root.children[parentId]){\n        if(childId === id){ return this.root.children[parentId][childId].destroy() }\n      }\n    }\n  }\n\n  joinChild(el){\n    let child = this.getChildById(el.id)\n    if(!child){\n      let view = new View(el, this.liveSocket, this)\n      this.root.children[this.id][view.id] = view\n      view.join()\n      this.childJoins++\n      return true\n    }\n  }\n\n  isJoinPending(){ return this.joinPending }\n\n  ackJoin(_child){\n    this.childJoins--\n\n    if(this.childJoins === 0){\n      if(this.parent){\n        this.parent.ackJoin(this)\n      } else {\n        this.onAllChildJoinsComplete()\n      }\n    }\n  }\n\n  onAllChildJoinsComplete(){\n    // we can clear pending form recoveries now that we've joined.\n    // They either all resolved or were abandoned\n    this.pendingForms.clear()\n    // we can also clear the formsForRecovery object to not keep old form elements around\n    this.formsForRecovery = {}\n    this.joinCallback(() => {\n      this.pendingJoinOps.forEach(([view, op]) => {\n        if(!view.isDestroyed()){ op() }\n      })\n      this.pendingJoinOps = []\n    })\n  }\n\n  update(diff, events){\n    if(this.isJoinPending() || (this.liveSocket.hasPendingLink() && this.root.isMain())){\n      return this.pendingDiffs.push({diff, events})\n    }\n\n    this.rendered.mergeDiff(diff)\n    let phxChildrenAdded = false\n\n    // When the diff only contains component diffs, then walk components\n    // and patch only the parent component containers found in the diff.\n    // Otherwise, patch entire LV container.\n    if(this.rendered.isComponentOnlyDiff(diff)){\n      this.liveSocket.time(\"component patch complete\", () => {\n        let parentCids = DOM.findExistingParentCIDs(this.el, this.rendered.componentCIDs(diff))\n        parentCids.forEach(parentCID => {\n          if(this.componentPatch(this.rendered.getComponent(diff, parentCID), parentCID)){ phxChildrenAdded = true }\n        })\n      })\n    } else if(!isEmpty(diff)){\n      this.liveSocket.time(\"full patch complete\", () => {\n        let [html, streams] = this.renderContainer(diff, \"update\")\n        let patch = new DOMPatch(this, this.el, this.id, html, streams, null)\n        phxChildrenAdded = this.performPatch(patch, true)\n      })\n    }\n\n    this.liveSocket.dispatchEvents(events)\n    if(phxChildrenAdded){ this.joinNewChildren() }\n  }\n\n  renderContainer(diff, kind){\n    return this.liveSocket.time(`toString diff (${kind})`, () => {\n      let tag = this.el.tagName\n      // Don't skip any component in the diff nor any marked as pruned\n      // (as they may have been added back)\n      let cids = diff ? this.rendered.componentCIDs(diff) : null\n      let [html, streams] = this.rendered.toString(cids)\n      return [`<${tag}>${html}</${tag}>`, streams]\n    })\n  }\n\n  componentPatch(diff, cid){\n    if(isEmpty(diff)) return false\n    let [html, streams] = this.rendered.componentToString(cid)\n    let patch = new DOMPatch(this, this.el, this.id, html, streams, cid)\n    let childrenAdded = this.performPatch(patch, true)\n    return childrenAdded\n  }\n\n  getHook(el){ return this.viewHooks[ViewHook.elementID(el)] }\n\n  addHook(el){\n    let hookElId = ViewHook.elementID(el)\n\n    // only ever try to add hooks to elements owned by this view\n    if(el.getAttribute && !this.ownsElement(el)){ return }\n\n    if(hookElId && !this.viewHooks[hookElId]){\n      // hook created, but not attached (createHook for web component)\n      let hook = DOM.getCustomElHook(el) || logError(`no hook found for custom element: ${el.id}`)\n      this.viewHooks[hookElId] = hook\n      hook.__attachView(this)\n      return hook\n    }\n    else if(hookElId || !el.getAttribute){\n      // no hook found\n      return\n    } else {\n      // new hook found with phx-hook attribute\n      let hookName = el.getAttribute(`data-phx-${PHX_HOOK}`) || el.getAttribute(this.binding(PHX_HOOK))\n      let callbacks = this.liveSocket.getHookCallbacks(hookName)\n\n      if(callbacks){\n        if(!el.id){ logError(`no DOM ID for hook \"${hookName}\". Hooks require a unique ID on each element.`, el) }\n        let hook = new ViewHook(this, el, callbacks)\n        this.viewHooks[ViewHook.elementID(hook.el)] = hook\n        return hook\n      } else if(hookName !== null){\n        logError(`unknown hook found for \"${hookName}\"`, el)\n      }\n    }\n  }\n\n  destroyHook(hook){\n    // __destroyed clears the elementID from the hook, therefore\n    // we need to get it before calling __destroyed\n    const hookId = ViewHook.elementID(hook.el)\n    hook.__destroyed()\n    hook.__cleanup__()\n    delete this.viewHooks[hookId]\n  }\n\n  applyPendingUpdates(){\n    // prevent race conditions where we might still be pending a new\n    // navigation after applying the current one;\n    // if we call update and a pendingDiff is not applied, it would\n    // be silently dropped otherwise, as update would push it back to\n    // pendingDiffs, but we clear it immediately after\n    if(this.liveSocket.hasPendingLink() && this.root.isMain()){ return }\n    this.pendingDiffs.forEach(({diff, events}) => this.update(diff, events))\n    this.pendingDiffs = []\n    this.eachChild(child => child.applyPendingUpdates())\n  }\n\n  eachChild(callback){\n    let children = this.root.children[this.id] || {}\n    for(let id in children){ callback(this.getChildById(id)) }\n  }\n\n  onChannel(event, cb){\n    this.liveSocket.onChannel(this.channel, event, resp => {\n      if(this.isJoinPending()){\n        this.root.pendingJoinOps.push([this, () => cb(resp)])\n      } else {\n        this.liveSocket.requestDOMUpdate(() => cb(resp))\n      }\n    })\n  }\n\n  bindChannel(){\n    // The diff event should be handled by the regular update operations.\n    // All other operations are queued to be applied only after join.\n    this.liveSocket.onChannel(this.channel, \"diff\", (rawDiff) => {\n      this.liveSocket.requestDOMUpdate(() => {\n        this.applyDiff(\"update\", rawDiff, ({diff, events}) => this.update(diff, events))\n      })\n    })\n    this.onChannel(\"redirect\", ({to, flash}) => this.onRedirect({to, flash}))\n    this.onChannel(\"live_patch\", (redir) => this.onLivePatch(redir))\n    this.onChannel(\"live_redirect\", (redir) => this.onLiveRedirect(redir))\n    this.channel.onError(reason => this.onError(reason))\n    this.channel.onClose(reason => this.onClose(reason))\n  }\n\n  destroyAllChildren(){ this.eachChild(child => child.destroy()) }\n\n  onLiveRedirect(redir){\n    let {to, kind, flash} = redir\n    let url = this.expandURL(to)\n    let e = new CustomEvent(\"phx:server-navigate\", {detail: {to, kind, flash}})\n    this.liveSocket.historyRedirect(e, url, kind, flash)\n  }\n\n  onLivePatch(redir){\n    let {to, kind} = redir\n    this.href = this.expandURL(to)\n    this.liveSocket.historyPatch(to, kind)\n  }\n\n  expandURL(to){\n    return to.startsWith(\"/\") ? `${window.location.protocol}//${window.location.host}${to}` : to\n  }\n\n  onRedirect({to, flash, reloadToken}){ this.liveSocket.redirect(to, flash, reloadToken) }\n\n  isDestroyed(){ return this.destroyed }\n\n  joinDead(){ this.isDead = true }\n\n  joinPush(){\n    this.joinPush = this.joinPush || this.channel.join()\n    return this.joinPush\n  }\n\n  join(callback){\n    this.showLoader(this.liveSocket.loaderTimeout)\n    this.bindChannel()\n    if(this.isMain()){\n      this.stopCallback = this.liveSocket.withPageLoading({to: this.href, kind: \"initial\"})\n    }\n    this.joinCallback = (onDone) => {\n      onDone = onDone || function(){}\n      callback ? callback(this.joinCount, onDone) : onDone()\n    }\n\n    this.wrapPush(() => this.channel.join(), {\n      ok: (resp) => this.liveSocket.requestDOMUpdate(() => this.onJoin(resp)),\n      error: (error) => this.onJoinError(error),\n      timeout: () => this.onJoinError({reason: \"timeout\"})\n    })\n  }\n\n  onJoinError(resp){\n    if(resp.reason === \"reload\"){\n      this.log(\"error\", () => [`failed mount with ${resp.status}. Falling back to page reload`, resp])\n      this.onRedirect({to: this.root.href, reloadToken: resp.token})\n      return\n    } else if(resp.reason === \"unauthorized\" || resp.reason === \"stale\"){\n      this.log(\"error\", () => [\"unauthorized live_redirect. Falling back to page request\", resp])\n      this.onRedirect({to: this.root.href, flash: this.flash})\n      return\n    }\n    if(resp.redirect || resp.live_redirect){\n      this.joinPending = false\n      this.channel.leave()\n    }\n    if(resp.redirect){ return this.onRedirect(resp.redirect) }\n    if(resp.live_redirect){ return this.onLiveRedirect(resp.live_redirect) }\n    this.log(\"error\", () => [\"unable to join\", resp])\n    if(this.isMain()){\n      this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS])\n      if(this.liveSocket.isConnected()){ this.liveSocket.reloadWithJitter(this) }\n    } else {\n      if(this.joinAttempts >= MAX_CHILD_JOIN_ATTEMPTS){\n        // put the root review into permanent error state, but don't destroy it as it can remain active\n        this.root.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS])\n        this.log(\"error\", () => [`giving up trying to mount after ${MAX_CHILD_JOIN_ATTEMPTS} tries`, resp])\n        this.destroy()\n      }\n      let trueChildEl = DOM.byId(this.el.id)\n      if(trueChildEl){\n        DOM.mergeAttrs(trueChildEl, this.el)\n        this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS])\n        this.el = trueChildEl\n      } else {\n        this.destroy()\n      }\n    }\n  }\n\n  onClose(reason){\n    if(this.isDestroyed()){ return }\n    if(this.isMain() && this.liveSocket.hasPendingLink() && reason !== \"leave\"){\n      return this.liveSocket.reloadWithJitter(this)\n    }\n    this.destroyAllChildren()\n    this.liveSocket.dropActiveElement(this)\n    // document.activeElement can be null in Internet Explorer 11\n    if(document.activeElement){ document.activeElement.blur() }\n    if(this.liveSocket.isUnloaded()){\n      this.showLoader(BEFORE_UNLOAD_LOADER_TIMEOUT)\n    }\n  }\n\n  onError(reason){\n    this.onClose(reason)\n    if(this.liveSocket.isConnected()){ this.log(\"error\", () => [\"view crashed\", reason]) }\n    if(!this.liveSocket.isUnloaded()){\n      if(this.liveSocket.isConnected()){\n        this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_SERVER_ERROR_CLASS])\n      } else {\n        this.displayError([PHX_LOADING_CLASS, PHX_ERROR_CLASS, PHX_CLIENT_ERROR_CLASS])\n      }\n    }\n  }\n\n  displayError(classes){\n    if(this.isMain()){ DOM.dispatchEvent(window, \"phx:page-loading-start\", {detail: {to: this.href, kind: \"error\"}}) }\n    this.showLoader()\n    this.setContainerClasses(...classes)\n    this.delayedDisconnected()\n  }\n\n  delayedDisconnected(){\n    this.disconnectedTimer = setTimeout(() => {\n      this.execAll(this.binding(\"disconnected\"))\n    }, this.liveSocket.disconnectedTimeout)\n  }\n\n  wrapPush(callerPush, receives){\n    let latency = this.liveSocket.getLatencySim()\n    let withLatency = latency ?\n      (cb) => setTimeout(() => !this.isDestroyed() && cb(), latency) :\n      (cb) => !this.isDestroyed() && cb()\n\n    withLatency(() => {\n      callerPush()\n        .receive(\"ok\", resp => withLatency(() => receives.ok && receives.ok(resp)))\n        .receive(\"error\", reason => withLatency(() => receives.error && receives.error(reason)))\n        .receive(\"timeout\", () => withLatency(() => receives.timeout && receives.timeout()))\n    })\n  }\n\n  pushWithReply(refGenerator, event, payload){\n    if(!this.isConnected()){ return Promise.reject({error: \"noconnection\"}) }\n\n    let [ref, [el], opts] = refGenerator ? refGenerator() : [null, [], {}]\n    let oldJoinCount = this.joinCount\n    let onLoadingDone = function(){}\n    if(opts.page_loading){\n      onLoadingDone = this.liveSocket.withPageLoading({kind: \"element\", target: el})\n    }\n\n    if(typeof (payload.cid) !== \"number\"){ delete payload.cid }\n\n    return new Promise((resolve, reject) => {\n      this.wrapPush(() => this.channel.push(event, payload, PUSH_TIMEOUT), {\n        ok: (resp) => {\n          if(ref !== null){ this.lastAckRef = ref }\n          let finish = (hookReply) => {\n            if(resp.redirect){ this.onRedirect(resp.redirect) }\n            if(resp.live_patch){ this.onLivePatch(resp.live_patch) }\n            if(resp.live_redirect){ this.onLiveRedirect(resp.live_redirect) }\n            onLoadingDone()\n            resolve({resp: resp, reply: hookReply})\n          }\n          if(resp.diff){\n            this.liveSocket.requestDOMUpdate(() => {\n              this.applyDiff(\"update\", resp.diff, ({diff, reply, events}) => {\n                if(ref !== null){\n                  this.undoRefs(ref, payload.event)\n                }\n                this.update(diff, events)\n                finish(reply)\n              })\n            })\n          } else {\n            if(ref !== null){ this.undoRefs(ref, payload.event) }\n            finish(null)\n          }\n        },\n        error: (reason) => reject({error: reason}),\n        timeout: () => {\n          reject({timeout: true})\n          if(this.joinCount === oldJoinCount){\n            this.liveSocket.reloadWithJitter(this, () => {\n              this.log(\"timeout\", () => [\"received timeout while communicating with server. Falling back to hard refresh for recovery\"])\n            })\n          }\n        }\n      })\n    })\n  }\n\n  undoRefs(ref, phxEvent, onlyEls){\n    if(!this.isConnected()){ return } // exit if external form triggered\n    let selector = `[${PHX_REF_SRC}=\"${this.refSrc()}\"]`\n\n    if(onlyEls){\n      onlyEls = new Set(onlyEls)\n      DOM.all(document, selector, parent => {\n        if(onlyEls && !onlyEls.has(parent)){ return }\n        // undo any child refs within parent first\n        DOM.all(parent, selector, child => this.undoElRef(child, ref, phxEvent))\n        this.undoElRef(parent, ref, phxEvent)\n      })\n    } else {\n      DOM.all(document, selector, el => this.undoElRef(el, ref, phxEvent))\n    }\n  }\n\n  undoElRef(el, ref, phxEvent){\n    let elRef = new ElementRef(el)\n\n    elRef.maybeUndo(ref, phxEvent, clonedTree => {\n      // we need to perform a full patch on unlocked elements\n      // to perform all the necessary logic (like calling updated for hooks, etc.)\n      let patch = new DOMPatch(this, el, this.id, clonedTree, [], null, {undoRef: ref})\n      const phxChildrenAdded = this.performPatch(patch, true)\n      DOM.all(el, `[${PHX_REF_SRC}=\"${this.refSrc()}\"]`, child => this.undoElRef(child, ref, phxEvent))\n      if(phxChildrenAdded){ this.joinNewChildren() }\n    })\n  }\n\n  refSrc(){ return this.el.id }\n\n  putRef(elements, phxEvent, eventType, opts = {}){\n    let newRef = this.ref++\n    let disableWith = this.binding(PHX_DISABLE_WITH)\n    if(opts.loading){\n      let loadingEls = DOM.all(document, opts.loading).map(el => {\n        return {el, lock: true, loading: true}\n      })\n      elements = elements.concat(loadingEls)\n    }\n\n    for(let {el, lock, loading} of elements){\n      if(!lock && !loading){ throw new Error(\"putRef requires lock or loading\") }\n      el.setAttribute(PHX_REF_SRC, this.refSrc())\n      if(loading){ el.setAttribute(PHX_REF_LOADING, newRef) }\n      if(lock){ el.setAttribute(PHX_REF_LOCK, newRef) }\n\n      if(!loading || (opts.submitter && !(el === opts.submitter || el === opts.form))){ continue }\n\n      let lockCompletePromise = new Promise(resolve => {\n        el.addEventListener(`phx:undo-lock:${newRef}`, () => resolve(detail), {once: true})\n      })\n\n      let loadingCompletePromise = new Promise(resolve => {\n        el.addEventListener(`phx:undo-loading:${newRef}`, () => resolve(detail), {once: true})\n      })\n\n      el.classList.add(`phx-${eventType}-loading`)\n      let disableText = el.getAttribute(disableWith)\n      if(disableText !== null){\n        if(!el.getAttribute(PHX_DISABLE_WITH_RESTORE)){\n          el.setAttribute(PHX_DISABLE_WITH_RESTORE, el.innerText)\n        }\n        if(disableText !== \"\"){ el.innerText = disableText }\n        // PHX_DISABLED could have already been set in disableForm\n        el.setAttribute(PHX_DISABLED, el.getAttribute(PHX_DISABLED) || el.disabled)\n        el.setAttribute(\"disabled\", \"\")\n      }\n\n      let detail = {\n        event: phxEvent,\n        eventType: eventType,\n        ref: newRef,\n        isLoading: loading,\n        isLocked: lock,\n        lockElements: elements.filter(({lock}) => lock).map(({el}) => el),\n        loadingElements: elements.filter(({loading}) => loading).map(({el}) => el),\n        unlock: (els) => {\n          els = Array.isArray(els) ? els : [els]\n          this.undoRefs(newRef, phxEvent, els)\n        },\n        lockComplete: lockCompletePromise,\n        loadingComplete: loadingCompletePromise,\n        lock: (lockEl) => {\n          return new Promise(resolve => {\n            if(this.isAcked(newRef)){ return resolve(detail) }\n            lockEl.setAttribute(PHX_REF_LOCK, newRef)\n            lockEl.setAttribute(PHX_REF_SRC, this.refSrc())\n            lockEl.addEventListener(`phx:lock-stop:${newRef}`, () => resolve(detail), {once: true})\n          })\n        }\n      }\n      el.dispatchEvent(new CustomEvent(\"phx:push\", {\n        detail: detail,\n        bubbles: true,\n        cancelable: false\n      }))\n      if(phxEvent){\n        el.dispatchEvent(new CustomEvent(`phx:push:${phxEvent}`, {\n          detail: detail,\n          bubbles: true,\n          cancelable: false\n        }))\n      }\n    }\n    return [newRef, elements.map(({el}) => el), opts]\n  }\n\n  isAcked(ref){ return this.lastAckRef !== null && this.lastAckRef >= ref }\n\n  componentID(el){\n    let cid = el.getAttribute && el.getAttribute(PHX_COMPONENT)\n    return cid ? parseInt(cid) : null\n  }\n\n  targetComponentID(target, targetCtx, opts = {}){\n    if(isCid(targetCtx)){ return targetCtx }\n\n    let cidOrSelector = opts.target || target.getAttribute(this.binding(\"target\"))\n    if(isCid(cidOrSelector)){\n      return parseInt(cidOrSelector)\n    } else if(targetCtx && (cidOrSelector !== null || opts.target)){\n      return this.closestComponentID(targetCtx)\n    } else {\n      return null\n    }\n  }\n\n  closestComponentID(targetCtx){\n    if(isCid(targetCtx)){\n      return targetCtx\n    } else if(targetCtx){\n      return maybe(targetCtx.closest(`[${PHX_COMPONENT}]`), el => this.ownsElement(el) && this.componentID(el))\n    } else {\n      return null\n    }\n  }\n\n  pushHookEvent(el, targetCtx, event, payload, onReply){\n    if(!this.isConnected()){\n      this.log(\"hook\", () => [\"unable to push hook event. LiveView not connected\", event, payload])\n      return false\n    }\n    let [ref, els, opts] = this.putRef([{el, loading: true, lock: true}], event, \"hook\")\n    this.pushWithReply(() => [ref, els, opts], \"event\", {\n      type: \"hook\",\n      event: event,\n      value: payload,\n      cid: this.closestComponentID(targetCtx)\n    }).then(({resp: _resp, reply: hookReply}) => onReply(hookReply, ref))\n\n    return ref\n  }\n\n  extractMeta(el, meta, value){\n    let prefix = this.binding(\"value-\")\n    for(let i = 0; i < el.attributes.length; i++){\n      if(!meta){ meta = {} }\n      let name = el.attributes[i].name\n      if(name.startsWith(prefix)){ meta[name.replace(prefix, \"\")] = el.getAttribute(name) }\n    }\n    if(el.value !== undefined && !(el instanceof HTMLFormElement)){\n      if(!meta){ meta = {} }\n      meta.value = el.value\n\n      if(el.tagName === \"INPUT\" && CHECKABLE_INPUTS.indexOf(el.type) >= 0 && !el.checked){\n        delete meta.value\n      }\n    }\n    if(value){\n      if(!meta){ meta = {} }\n      for(let key in value){ meta[key] = value[key] }\n    }\n    return meta\n  }\n\n  pushEvent(type, el, targetCtx, phxEvent, meta, opts = {}, onReply){\n    this.pushWithReply(() => this.putRef([{el, loading: true, lock: true}], phxEvent, type, opts), \"event\", {\n      type: type,\n      event: phxEvent,\n      value: this.extractMeta(el, meta, opts.value),\n      cid: this.targetComponentID(el, targetCtx, opts)\n    })\n      .then(({reply}) => onReply && onReply(reply))\n      .catch((error) => logError(\"Failed to push event\", error))\n  }\n\n  pushFileProgress(fileEl, entryRef, progress, onReply = function (){ }){\n    this.liveSocket.withinOwners(fileEl.form, (view, targetCtx) => {\n      view.pushWithReply(null, \"progress\", {\n        event: fileEl.getAttribute(view.binding(PHX_PROGRESS)),\n        ref: fileEl.getAttribute(PHX_UPLOAD_REF),\n        entry_ref: entryRef,\n        progress: progress,\n        cid: view.targetComponentID(fileEl.form, targetCtx)\n      })\n        .then(({resp}) => onReply(resp))\n        .catch((error) => logError(\"Failed to push file progress\", error))\n    })\n  }\n\n  pushInput(inputEl, targetCtx, forceCid, phxEvent, opts, callback){\n    if(!inputEl.form){\n      throw new Error(\"form events require the input to be inside a form\")\n    }\n\n    let uploads\n    let cid = isCid(forceCid) ? forceCid : this.targetComponentID(inputEl.form, targetCtx, opts)\n    let refGenerator = () => {\n      return this.putRef([\n        {el: inputEl, loading: true, lock: true},\n        {el: inputEl.form, loading: true, lock: true}\n      ], phxEvent, \"change\", opts)\n    }\n    let formData\n    let meta = this.extractMeta(inputEl.form, {}, opts.value)\n    let serializeOpts = {}\n    if(inputEl instanceof HTMLButtonElement){ serializeOpts.submitter = inputEl }\n    if(inputEl.getAttribute(this.binding(\"change\"))){\n      formData = serializeForm(inputEl.form, serializeOpts, [inputEl.name])\n    } else {\n      formData = serializeForm(inputEl.form, serializeOpts)\n    }\n    if(DOM.isUploadInput(inputEl) && inputEl.files && inputEl.files.length > 0){\n      LiveUploader.trackFiles(inputEl, Array.from(inputEl.files))\n    }\n    uploads = LiveUploader.serializeUploads(inputEl)\n\n    let event = {\n      type: \"form\",\n      event: phxEvent,\n      value: formData,\n      meta: {\n        // no target was implicitly sent as \"undefined\" in LV <= 1.0.5, therefore\n        // we have to keep it. In 1.0.6 we switched from passing meta as URL encoded data\n        // to passing it directly in the event, but the JSON encode would drop keys with\n        // undefined values.\n        _target: opts._target || \"undefined\",\n        ...meta\n      },\n      uploads: uploads,\n      cid: cid\n    }\n    this.pushWithReply(refGenerator, \"event\", event).then(({resp}) => {\n      if(DOM.isUploadInput(inputEl) && DOM.isAutoUpload(inputEl)){\n        // the element could be inside a locked parent for other unrelated changes;\n        // we can only start uploads when the tree is unlocked and the\n        // necessary data attributes are set in the real DOM\n        ElementRef.onUnlock(inputEl, () => {\n          if(LiveUploader.filesAwaitingPreflight(inputEl).length > 0){\n            let [ref, _els] = refGenerator()\n            this.undoRefs(ref, phxEvent, [inputEl.form])\n            this.uploadFiles(inputEl.form, phxEvent, targetCtx, ref, cid, (_uploads) => {\n              callback && callback(resp)\n              this.triggerAwaitingSubmit(inputEl.form, phxEvent)\n              this.undoRefs(ref, phxEvent)\n            })\n          }\n        })\n      } else {\n        callback && callback(resp)\n      }\n    }).catch((error) => logError(\"Failed to push input event\", error))\n  }\n\n  triggerAwaitingSubmit(formEl, phxEvent){\n    let awaitingSubmit = this.getScheduledSubmit(formEl)\n    if(awaitingSubmit){\n      let [_el, _ref, _opts, callback] = awaitingSubmit\n      this.cancelSubmit(formEl, phxEvent)\n      callback()\n    }\n  }\n\n  getScheduledSubmit(formEl){\n    return this.formSubmits.find(([el, _ref, _opts, _callback]) => el.isSameNode(formEl))\n  }\n\n  scheduleSubmit(formEl, ref, opts, callback){\n    if(this.getScheduledSubmit(formEl)){ return true }\n    this.formSubmits.push([formEl, ref, opts, callback])\n  }\n\n  cancelSubmit(formEl, phxEvent){\n    this.formSubmits = this.formSubmits.filter(([el, ref, _opts, _callback]) => {\n      if(el.isSameNode(formEl)){\n        this.undoRefs(ref, phxEvent)\n        return false\n      } else {\n        return true\n      }\n    })\n  }\n\n  disableForm(formEl, phxEvent, opts = {}){\n    let filterIgnored = el => {\n      let userIgnored = closestPhxBinding(el, `${this.binding(PHX_UPDATE)}=ignore`, el.form)\n      return !(userIgnored || closestPhxBinding(el, \"data-phx-update=ignore\", el.form))\n    }\n    let filterDisables = el => {\n      return el.hasAttribute(this.binding(PHX_DISABLE_WITH))\n    }\n    let filterButton = el => el.tagName == \"BUTTON\"\n\n    let filterInput = el => [\"INPUT\", \"TEXTAREA\", \"SELECT\"].includes(el.tagName)\n\n    let formElements = Array.from(formEl.elements)\n    let disables = formElements.filter(filterDisables)\n    let buttons = formElements.filter(filterButton).filter(filterIgnored)\n    let inputs = formElements.filter(filterInput).filter(filterIgnored)\n\n    buttons.forEach(button => {\n      button.setAttribute(PHX_DISABLED, button.disabled)\n      button.disabled = true\n    })\n    inputs.forEach(input => {\n      input.setAttribute(PHX_READONLY, input.readOnly)\n      input.readOnly = true\n      if(input.files){\n        input.setAttribute(PHX_DISABLED, input.disabled)\n        input.disabled = true\n      }\n    })\n    let formEls = disables.concat(buttons).concat(inputs).map(el => {\n      return {el, loading: true, lock: true}\n    })\n\n    // we reverse the order so form children are already locked by the time\n    // the form is locked\n    let els = [{el: formEl, loading: true, lock: false}].concat(formEls).reverse()\n    return this.putRef(els, phxEvent, \"submit\", opts)\n  }\n\n  pushFormSubmit(formEl, targetCtx, phxEvent, submitter, opts, onReply){\n    let refGenerator = () => this.disableForm(formEl, phxEvent, {\n      ...opts,\n      form: formEl,\n      submitter: submitter\n    })\n    // store the submitter in the form element in order to trigger it\n    // for phx-trigger-action\n    DOM.putPrivate(formEl, \"submitter\", submitter)\n    let cid = this.targetComponentID(formEl, targetCtx)\n    if(LiveUploader.hasUploadsInProgress(formEl)){\n      let [ref, _els] = refGenerator()\n      let push = () => this.pushFormSubmit(formEl, targetCtx, phxEvent, submitter, opts, onReply)\n      return this.scheduleSubmit(formEl, ref, opts, push)\n    } else if(LiveUploader.inputsAwaitingPreflight(formEl).length > 0){\n      let [ref, els] = refGenerator()\n      let proxyRefGen = () => [ref, els, opts]\n      this.uploadFiles(formEl, phxEvent, targetCtx, ref, cid, (_uploads) => {\n        // if we still having pending preflights it means we have invalid entries\n        // and the phx-submit cannot be completed\n        if(LiveUploader.inputsAwaitingPreflight(formEl).length > 0){\n          return this.undoRefs(ref, phxEvent)\n        }\n        let meta = this.extractMeta(formEl, {}, opts.value)\n        let formData = serializeForm(formEl, {submitter})\n        this.pushWithReply(proxyRefGen, \"event\", {\n          type: \"form\",\n          event: phxEvent,\n          value: formData,\n          meta: meta,\n          cid: cid\n        })\n          .then(({resp}) => onReply(resp))\n          .catch((error) => logError(\"Failed to push form submit\", error))\n      })\n    } else if(!(formEl.hasAttribute(PHX_REF_SRC) && formEl.classList.contains(\"phx-submit-loading\"))){\n      let meta = this.extractMeta(formEl, {}, opts.value)\n      let formData = serializeForm(formEl, {submitter})\n      this.pushWithReply(refGenerator, \"event\", {\n        type: \"form\",\n        event: phxEvent,\n        value: formData,\n        meta: meta,\n        cid: cid\n      })\n        .then(({resp}) => onReply(resp))\n        .catch((error) => logError(\"Failed to push form submit\", error))\n    }\n  }\n\n  uploadFiles(formEl, phxEvent, targetCtx, ref, cid, onComplete){\n    let joinCountAtUpload = this.joinCount\n    let inputEls = LiveUploader.activeFileInputs(formEl)\n    let numFileInputsInProgress = inputEls.length\n\n    // get each file input\n    inputEls.forEach(inputEl => {\n      let uploader = new LiveUploader(inputEl, this, () => {\n        numFileInputsInProgress--\n        if(numFileInputsInProgress === 0){ onComplete() }\n      })\n\n      let entries = uploader.entries().map(entry => entry.toPreflightPayload())\n\n      if(entries.length === 0){\n        numFileInputsInProgress--\n        return\n      }\n\n      let payload = {\n        ref: inputEl.getAttribute(PHX_UPLOAD_REF),\n        entries: entries,\n        cid: this.targetComponentID(inputEl.form, targetCtx)\n      }\n\n      this.log(\"upload\", () => [\"sending preflight request\", payload])\n\n      this.pushWithReply(null, \"allow_upload\", payload).then(({resp}) => {\n        this.log(\"upload\", () => [\"got preflight response\", resp])\n        // the preflight will reject entries beyond the max entries\n        // so we error and cancel entries on the client that are missing from the response\n        uploader.entries().forEach(entry => {\n          if(resp.entries && !resp.entries[entry.ref]){\n            this.handleFailedEntryPreflight(entry.ref, \"failed preflight\", uploader)\n          }\n        })\n        // for auto uploads, we may have an empty entries response from the server\n        // for form submits that contain invalid entries\n        if(resp.error || Object.keys(resp.entries).length === 0){\n          this.undoRefs(ref, phxEvent)\n          let errors = resp.error || []\n          errors.map(([entry_ref, reason]) => {\n            this.handleFailedEntryPreflight(entry_ref, reason, uploader)\n          })\n        } else {\n          let onError = (callback) => {\n            this.channel.onError(() => {\n              if(this.joinCount === joinCountAtUpload){ callback() }\n            })\n          }\n          uploader.initAdapterUpload(resp, onError, this.liveSocket)\n        }\n      }).catch((error) => logError(\"Failed to push upload\", error))\n    })\n  }\n\n  handleFailedEntryPreflight(uploadRef, reason, uploader){\n    if(uploader.isAutoUpload()){\n      // uploadRef may be top level upload config ref or entry ref\n      let entry = uploader.entries().find(entry => entry.ref === uploadRef.toString())\n      if(entry){ entry.cancel() }\n    } else {\n      uploader.entries().map(entry => entry.cancel())\n    }\n    this.log(\"upload\", () => [`error for entry ${uploadRef}`, reason])\n  }\n\n  dispatchUploads(targetCtx, name, filesOrBlobs){\n    let targetElement = this.targetCtxElement(targetCtx) || this.el\n    let inputs = DOM.findUploadInputs(targetElement).filter(el => el.name === name)\n    if(inputs.length === 0){ logError(`no live file inputs found matching the name \"${name}\"`) }\n    else if(inputs.length > 1){ logError(`duplicate live file inputs found matching the name \"${name}\"`) }\n    else { DOM.dispatchEvent(inputs[0], PHX_TRACK_UPLOADS, {detail: {files: filesOrBlobs}}) }\n  }\n\n  targetCtxElement(targetCtx){\n    if(isCid(targetCtx)){\n      let [target] = DOM.findComponentNodeList(this.el, targetCtx)\n      return target\n    } else if(targetCtx){\n      return targetCtx\n    } else {\n      return null\n    }\n  }\n\n  pushFormRecovery(oldForm, newForm, templateDom, callback){\n    // we are only recovering forms inside the current view, therefore it is safe to\n    // skip withinOwners here and always use this when referring to the view\n    const phxChange = this.binding(\"change\")\n    const phxTarget = newForm.getAttribute(this.binding(\"target\")) || newForm\n    const phxEvent = newForm.getAttribute(this.binding(PHX_AUTO_RECOVER)) || newForm.getAttribute(this.binding(\"change\"))\n    const inputs = Array.from(oldForm.elements).filter(el => DOM.isFormInput(el) && el.name && !el.hasAttribute(phxChange))\n    if(inputs.length === 0){\n      callback()\n      return\n    }\n\n    // we must clear tracked uploads before recovery as they no longer have valid refs\n    inputs.forEach(input => input.hasAttribute(PHX_UPLOAD_REF) && LiveUploader.clearFiles(input))\n    // pushInput assumes that there is a source element that initiated the change;\n    // because this is not the case when we recover forms, we provide the first input we find\n    let input = inputs.find(el => el.type !== \"hidden\") || inputs[0]\n\n    // in the case that there are multiple targets, we count the number of pending recovery events\n    // and only call the callback once all events have been processed\n    let pending = 0\n    // withinTargets(phxTarget, callback, dom, viewEl)\n    this.withinTargets(phxTarget, (targetView, targetCtx) => {\n      const cid = this.targetComponentID(newForm, targetCtx)\n      pending++\n      let e = new CustomEvent(\"phx:form-recovery\", {detail: {sourceElement: oldForm}})\n      JS.exec(e, \"change\", phxEvent, this, input, [\"push\", {\n        _target: input.name,\n        targetView,\n        targetCtx,\n        newCid: cid,\n        callback: () => {\n          pending--\n          if(pending === 0){ callback() }\n        }\n      }])\n    }, templateDom, templateDom)\n  }\n\n  pushLinkPatch(e, href, targetEl, callback){\n    let linkRef = this.liveSocket.setPendingLink(href)\n    // only add loading states if event is trusted (it was triggered by user, such as click) and\n    // it's not a forward/back navigation from popstate\n    let loading = e.isTrusted && e.type !== \"popstate\"\n    let refGen = targetEl ? () => this.putRef([{el: targetEl, loading: loading, lock: true}], null, \"click\") : null\n    let fallback = () => this.liveSocket.redirect(window.location.href)\n    let url = href.startsWith(\"/\") ? `${location.protocol}//${location.host}${href}` : href\n\n    this.pushWithReply(refGen, \"live_patch\", {url}).then(\n      ({resp}) => {\n        this.liveSocket.requestDOMUpdate(() => {\n          if(resp.link_redirect){\n            this.liveSocket.replaceMain(href, null, callback, linkRef)\n          } else {\n            if(this.liveSocket.commitPendingLink(linkRef)){\n              this.href = href\n            }\n            this.applyPendingUpdates()\n            callback && callback(linkRef)\n          }\n        })\n      },\n      ({error: _error, timeout: _timeout}) => fallback()\n    )\n  }\n\n  getFormsForRecovery(){\n    if(this.joinCount === 0){ return {} }\n\n    let phxChange = this.binding(\"change\")\n\n    return DOM.all(this.el, `form[${phxChange}]`)\n      .filter(form => form.id)\n      .filter(form => form.elements.length > 0)\n      .filter(form => form.getAttribute(this.binding(PHX_AUTO_RECOVER)) !== \"ignore\")\n      .map(form => {\n        // we perform a shallow clone and manually copy all elements\n        const clonedForm = form.cloneNode(false)\n        // we need to copy the private data as it contains\n        // the information about touched fields\n        DOM.copyPrivates(clonedForm, form)\n        Array.from(form.elements).forEach((el) => {\n          // we need to clone all child nodes as well,\n          // because those could also be selects\n          const clonedEl = el.cloneNode(true)\n          // we call morphdom to copy any special state\n          // like the selected option of a <select> element;\n          // this should be plenty fast as we call it on a small subset of the DOM,\n          // single inputs or a select with children\n          morphdom(clonedEl, el)\n          DOM.copyPrivates(clonedEl, el)\n          clonedForm.appendChild(clonedEl)\n        })\n        return clonedForm\n      })\n      .reduce((acc, form) => {\n        acc[form.id] = form\n        return acc\n      }, {})\n  }\n\n  maybePushComponentsDestroyed(destroyedCIDs){\n    let willDestroyCIDs = destroyedCIDs.filter(cid => {\n      return DOM.findComponentNodeList(this.el, cid).length === 0\n    })\n\n    if(willDestroyCIDs.length > 0){\n      // we must reset the render change tracking for cids that\n      // could be added back from the server so we don't skip them\n      willDestroyCIDs.forEach(cid => this.rendered.resetRender(cid))\n\n      this.pushWithReply(null, \"cids_will_destroy\", {cids: willDestroyCIDs}).then(() => {\n        // we must wait for pending transitions to complete before determining\n        // if the cids were added back to the DOM in the meantime (#3139)\n        this.liveSocket.requestDOMUpdate(() => {\n          // See if any of the cids we wanted to destroy were added back,\n          // if they were added back, we don't actually destroy them.\n          let completelyDestroyCIDs = willDestroyCIDs.filter(cid => {\n            return DOM.findComponentNodeList(this.el, cid).length === 0\n          })\n\n          if(completelyDestroyCIDs.length > 0){\n            this.pushWithReply(null, \"cids_destroyed\", {cids: completelyDestroyCIDs}).then(({resp}) => {\n              this.rendered.pruneCIDs(resp.cids)\n            }).catch((error) => logError(\"Failed to push components destroyed\", error))\n          }\n        })\n      }).catch((error) => logError(\"Failed to push components destroyed\", error))\n    }\n  }\n\n  ownsElement(el){\n    let parentViewEl = el.closest(PHX_VIEW_SELECTOR)\n    return el.getAttribute(PHX_PARENT_ID) === this.id ||\n      (parentViewEl && parentViewEl.id === this.id) ||\n      (!parentViewEl && this.isDead)\n  }\n\n  submitForm(form, targetCtx, phxEvent, submitter, opts = {}){\n    DOM.putPrivate(form, PHX_HAS_SUBMITTED, true)\n    const inputs = Array.from(form.elements)\n    inputs.forEach(input => DOM.putPrivate(input, PHX_HAS_SUBMITTED, true))\n    this.liveSocket.blurActiveElement(this)\n    this.pushFormSubmit(form, targetCtx, phxEvent, submitter, opts, () => {\n      this.liveSocket.restorePreviouslyActiveFocus()\n    })\n  }\n\n  binding(kind){ return this.liveSocket.binding(kind) }\n}\n", "/** Initializes the LiveSocket\n *\n *\n * @param {string} endPoint - The string WebSocket endpoint, ie, `\"wss://example.com/live\"`,\n *                                               `\"/live\"` (inherited host & protocol)\n * @param {Phoenix.Socket} socket - the required Phoenix Socket class imported from \"phoenix\". For example:\n *\n *     import {Socket} from \"phoenix\"\n *     import {LiveSocket} from \"phoenix_live_view\"\n *     let liveSocket = new LiveSocket(\"/live\", Socket, {...})\n *\n * @param {Object} [opts] - Optional configuration. Outside of keys listed below, all\n * configuration is passed directly to the Phoenix Socket constructor.\n * @param {Object} [opts.defaults] - The optional defaults to use for various bindings,\n * such as `phx-debounce`. Supports the following keys:\n *\n *   - debounce - the millisecond phx-debounce time. Defaults 300\n *   - throttle - the millisecond phx-throttle time. Defaults 300\n *\n * @param {Function} [opts.params] - The optional function for passing connect params.\n * The function receives the element associated with a given LiveView. For example:\n *\n *     (el) => {view: el.getAttribute(\"data-my-view-name\", token: window.myToken}\n *\n * @param {string} [opts.bindingPrefix] - The optional prefix to use for all phx DOM annotations.\n * Defaults to \"phx-\".\n * @param {Object} [opts.hooks] - The optional object for referencing LiveView hook callbacks.\n * @param {Object} [opts.uploaders] - The optional object for referencing LiveView uploader callbacks.\n * @param {integer} [opts.loaderTimeout] - The optional delay in milliseconds to wait before apply\n * loading states.\n * @param {integer} [opts.disconnectedTimeout] - The delay in milliseconds to wait before\n * executing phx-disconnected commands. Defaults to 500.\n * @param {integer} [opts.maxReloads] - The maximum reloads before entering failsafe mode.\n * @param {integer} [opts.reloadJitterMin] - The minimum time between normal reload attempts.\n * @param {integer} [opts.reloadJitterMax] - The maximum time between normal reload attempts.\n * @param {integer} [opts.failsafeJitter] - The time between reload attempts in failsafe mode.\n * @param {Function} [opts.viewLogger] - The optional function to log debug information. For example:\n *\n *     (view, kind, msg, obj) => console.log(`${view.id} ${kind}: ${msg} - `, obj)\n *\n * @param {Object} [opts.metadata] - The optional object mapping event names to functions for\n * populating event metadata. For example:\n *\n *     metadata: {\n *       click: (e, el) => {\n *         return {\n *           ctrlKey: e.ctrlKey,\n *           metaKey: e.metaKey,\n *           detail: e.detail || 1,\n *         }\n *       },\n *       keydown: (e, el) => {\n *         return {\n *           key: e.key,\n *           ctrlKey: e.ctrlKey,\n *           metaKey: e.metaKey,\n *           shiftKey: e.shiftKey\n *         }\n *       }\n *     }\n * @param {Object} [opts.sessionStorage] - An optional Storage compatible object\n * Useful when LiveView won't have access to `sessionStorage`.  For example, This could\n * happen if a site loads a cross-domain LiveView in an iframe.  Example usage:\n *\n *     class InMemoryStorage {\n *       constructor() { this.storage = {} }\n *       getItem(keyName) { return this.storage[keyName] || null }\n *       removeItem(keyName) { delete this.storage[keyName] }\n *       setItem(keyName, keyValue) { this.storage[keyName] = keyValue }\n *     }\n *\n * @param {Object} [opts.localStorage] - An optional Storage compatible object\n * Useful for when LiveView won't have access to `localStorage`.\n * See `opts.sessionStorage` for examples.\n*/\n\nimport {\n  BINDING_PREFIX,\n  CONSECUTIVE_RELOADS,\n  DEFAULTS,\n  FAILSAFE_JITTER,\n  LOADER_TIMEOUT,\n  DISCONNECTED_TIMEOUT,\n  MAX_RELOADS,\n  PHX_DEBOUNCE,\n  PHX_DROP_TARGET,\n  PHX_HAS_FOCUSED,\n  PHX_KEY,\n  PHX_LINK_STATE,\n  PHX_LIVE_LINK,\n  PHX_LV_DEBUG,\n  PHX_LV_LATENCY_SIM,\n  PHX_LV_PROFILE,\n  PHX_LV_HISTORY_POSITION,\n  PHX_MAIN,\n  PHX_PARENT_ID,\n  PHX_VIEW_SELECTOR,\n  PHX_ROOT_ID,\n  PHX_THROTTLE,\n  PHX_TRACK_UPLOADS,\n  PHX_SESSION,\n  RELOAD_JITTER_MIN,\n  RELOAD_JITTER_MAX,\n  PHX_REF_SRC,\n  PHX_RELOAD_STATUS\n} from \"./constants\"\n\nimport {\n  clone,\n  closestPhxBinding,\n  closure,\n  debug,\n  maybe\n} from \"./utils\"\n\nimport Browser from \"./browser\"\nimport DOM from \"./dom\"\nimport Hooks from \"./hooks\"\nimport LiveUploader from \"./live_uploader\"\nimport View from \"./view\"\nimport JS from \"./js\"\n\nexport let isUsedInput = (el) => DOM.isUsedInput(el)\n\nexport default class LiveSocket {\n  constructor(url, phxSocket, opts = {}){\n    this.unloaded = false\n    if(!phxSocket || phxSocket.constructor.name === \"Object\"){\n      throw new Error(`\n      a phoenix Socket must be provided as the second argument to the LiveSocket constructor. For example:\n\n          import {Socket} from \"phoenix\"\n          import {LiveSocket} from \"phoenix_live_view\"\n          let liveSocket = new LiveSocket(\"/live\", Socket, {...})\n      `)\n    }\n    this.socket = new phxSocket(url, opts)\n    this.bindingPrefix = opts.bindingPrefix || BINDING_PREFIX\n    this.opts = opts\n    this.params = closure(opts.params || {})\n    this.viewLogger = opts.viewLogger\n    this.metadataCallbacks = opts.metadata || {}\n    this.defaults = Object.assign(clone(DEFAULTS), opts.defaults || {})\n    this.activeElement = null\n    this.prevActive = null\n    this.silenced = false\n    this.main = null\n    this.outgoingMainEl = null\n    this.clickStartedAtTarget = null\n    this.linkRef = 1\n    this.roots = {}\n    this.href = window.location.href\n    this.pendingLink = null\n    this.currentLocation = clone(window.location)\n    this.hooks = opts.hooks || {}\n    this.uploaders = opts.uploaders || {}\n    this.loaderTimeout = opts.loaderTimeout || LOADER_TIMEOUT\n    this.disconnectedTimeout = opts.disconnectedTimeout || DISCONNECTED_TIMEOUT\n    this.reloadWithJitterTimer = null\n    this.maxReloads = opts.maxReloads || MAX_RELOADS\n    this.reloadJitterMin = opts.reloadJitterMin || RELOAD_JITTER_MIN\n    this.reloadJitterMax = opts.reloadJitterMax || RELOAD_JITTER_MAX\n    this.failsafeJitter = opts.failsafeJitter || FAILSAFE_JITTER\n    this.localStorage = opts.localStorage || window.localStorage\n    this.sessionStorage = opts.sessionStorage || window.sessionStorage\n    this.boundTopLevelEvents = false\n    this.boundEventNames = new Set()\n    this.serverCloseRef = null\n    this.domCallbacks = Object.assign({\n      jsQuerySelectorAll: null,\n      onPatchStart: closure(),\n      onPatchEnd: closure(),\n      onNodeAdded: closure(),\n      onBeforeElUpdated: closure()},\n    opts.dom || {})\n    this.transitions = new TransitionSet()\n    this.currentHistoryPosition = parseInt(this.sessionStorage.getItem(PHX_LV_HISTORY_POSITION)) || 0\n    window.addEventListener(\"pagehide\", _e => {\n      this.unloaded = true\n    })\n    this.socket.onOpen(() => {\n      if(this.isUnloaded()){\n        // reload page if being restored from back/forward cache and browser does not emit \"pageshow\"\n        window.location.reload()\n      }\n    })\n  }\n\n  // public\n\n  version(){ return LV_VSN }\n\n  isProfileEnabled(){ return this.sessionStorage.getItem(PHX_LV_PROFILE) === \"true\" }\n\n  isDebugEnabled(){ return this.sessionStorage.getItem(PHX_LV_DEBUG) === \"true\" }\n\n  isDebugDisabled(){ return this.sessionStorage.getItem(PHX_LV_DEBUG) === \"false\" }\n\n  enableDebug(){ this.sessionStorage.setItem(PHX_LV_DEBUG, \"true\") }\n\n  enableProfiling(){ this.sessionStorage.setItem(PHX_LV_PROFILE, \"true\") }\n\n  disableDebug(){ this.sessionStorage.setItem(PHX_LV_DEBUG, \"false\") }\n\n  disableProfiling(){ this.sessionStorage.removeItem(PHX_LV_PROFILE) }\n\n  enableLatencySim(upperBoundMs){\n    this.enableDebug()\n    console.log(\"latency simulator enabled for the duration of this browser session. Call disableLatencySim() to disable\")\n    this.sessionStorage.setItem(PHX_LV_LATENCY_SIM, upperBoundMs)\n  }\n\n  disableLatencySim(){ this.sessionStorage.removeItem(PHX_LV_LATENCY_SIM) }\n\n  getLatencySim(){\n    let str = this.sessionStorage.getItem(PHX_LV_LATENCY_SIM)\n    return str ? parseInt(str) : null\n  }\n\n  getSocket(){ return this.socket }\n\n  connect(){\n    // enable debug by default if on localhost and not explicitly disabled\n    if(window.location.hostname === \"localhost\" && !this.isDebugDisabled()){ this.enableDebug() }\n    let doConnect = () => {\n      this.resetReloadStatus()\n      if(this.joinRootViews()){\n        this.bindTopLevelEvents()\n        this.socket.connect()\n      } else if(this.main){\n        this.socket.connect()\n      } else {\n        this.bindTopLevelEvents({dead: true})\n      }\n      this.joinDeadView()\n    }\n    if([\"complete\", \"loaded\", \"interactive\"].indexOf(document.readyState) >= 0){\n      doConnect()\n    } else {\n      document.addEventListener(\"DOMContentLoaded\", () => doConnect())\n    }\n  }\n\n  disconnect(callback){\n    clearTimeout(this.reloadWithJitterTimer)\n    // remove the socket close listener to avoid trying to handle\n    // a server close event when it is actually caused by us disconnecting\n    if(this.serverCloseRef){\n      this.socket.off(this.serverCloseRef)\n      this.serverCloseRef = null\n    }\n    this.socket.disconnect(callback)\n  }\n\n  replaceTransport(transport){\n    clearTimeout(this.reloadWithJitterTimer)\n    this.socket.replaceTransport(transport)\n    this.connect()\n  }\n\n  execJS(el, encodedJS, eventType = null){\n    let e = new CustomEvent(\"phx:exec\", {detail: {sourceElement: el}})\n    this.owner(el, view => JS.exec(e, eventType, encodedJS, view, el))\n  }\n\n  // private\n\n  execJSHookPush(el, phxEvent, data, callback){\n    this.withinOwners(el, view => {\n      let e = new CustomEvent(\"phx:exec\", {detail: {sourceElement: el}})\n      JS.exec(e, \"hook\", phxEvent, view, el, [\"push\", {data, callback}])\n    })\n  }\n\n  unload(){\n    if(this.unloaded){ return }\n    if(this.main && this.isConnected()){ this.log(this.main, \"socket\", () => [\"disconnect for page nav\"]) }\n    this.unloaded = true\n    this.destroyAllViews()\n    this.disconnect()\n  }\n\n  triggerDOM(kind, args){ this.domCallbacks[kind](...args) }\n\n  time(name, func){\n    if(!this.isProfileEnabled() || !console.time){ return func() }\n    console.time(name)\n    let result = func()\n    console.timeEnd(name)\n    return result\n  }\n\n  log(view, kind, msgCallback){\n    if(this.viewLogger){\n      let [msg, obj] = msgCallback()\n      this.viewLogger(view, kind, msg, obj)\n    } else if(this.isDebugEnabled()){\n      let [msg, obj] = msgCallback()\n      debug(view, kind, msg, obj)\n    }\n  }\n\n  requestDOMUpdate(callback){\n    this.transitions.after(callback)\n  }\n\n  transition(time, onStart, onDone = function(){}){\n    this.transitions.addTransition(time, onStart, onDone)\n  }\n\n  onChannel(channel, event, cb){\n    channel.on(event, data => {\n      let latency = this.getLatencySim()\n      if(!latency){\n        cb(data)\n      } else {\n        setTimeout(() => cb(data), latency)\n      }\n    })\n  }\n\n  reloadWithJitter(view, log){\n    clearTimeout(this.reloadWithJitterTimer)\n    this.disconnect()\n    let minMs = this.reloadJitterMin\n    let maxMs = this.reloadJitterMax\n    let afterMs = Math.floor(Math.random() * (maxMs - minMs + 1)) + minMs\n    let tries = Browser.updateLocal(this.localStorage, window.location.pathname, CONSECUTIVE_RELOADS, 0, count => count + 1)\n    if(tries >= this.maxReloads){\n      afterMs = this.failsafeJitter\n    }\n    this.reloadWithJitterTimer = setTimeout(() => {\n      // if view has recovered, such as transport replaced, then cancel\n      if(view.isDestroyed() || view.isConnected()){ return }\n      view.destroy()\n      log ? log() : this.log(view, \"join\", () => [`encountered ${tries} consecutive reloads`])\n      if(tries >= this.maxReloads){\n        this.log(view, \"join\", () => [`exceeded ${this.maxReloads} consecutive reloads. Entering failsafe mode`])\n      }\n      if(this.hasPendingLink()){\n        window.location = this.pendingLink\n      } else {\n        window.location.reload()\n      }\n    }, afterMs)\n  }\n\n  getHookCallbacks(name){\n    return name && name.startsWith(\"Phoenix.\") ? Hooks[name.split(\".\")[1]] : this.hooks[name]\n  }\n\n  isUnloaded(){ return this.unloaded }\n\n  isConnected(){ return this.socket.isConnected() }\n\n  getBindingPrefix(){ return this.bindingPrefix }\n\n  binding(kind){ return `${this.getBindingPrefix()}${kind}` }\n\n  channel(topic, params){ return this.socket.channel(topic, params) }\n\n  joinDeadView(){\n    let body = document.body\n    if(body && !this.isPhxView(body) && !this.isPhxView(document.firstElementChild)){\n      let view = this.newRootView(body)\n      view.setHref(this.getHref())\n      view.joinDead()\n      if(!this.main){ this.main = view }\n      window.requestAnimationFrame(() => {\n        view.execNewMounted()\n        // restore scroll position when navigating from an external / non-live page\n        this.maybeScroll(history.state?.scroll)\n      })\n    }\n  }\n\n  joinRootViews(){\n    let rootsFound = false\n    DOM.all(document, `${PHX_VIEW_SELECTOR}:not([${PHX_PARENT_ID}])`, rootEl => {\n      if(!this.getRootById(rootEl.id)){\n        let view = this.newRootView(rootEl)\n        // stickies cannot be mounted at the router and therefore should not\n        // get a href set on them\n        if(!DOM.isPhxSticky(rootEl)){ view.setHref(this.getHref()) }\n        view.join()\n        if(rootEl.hasAttribute(PHX_MAIN)){ this.main = view }\n      }\n      rootsFound = true\n    })\n    return rootsFound\n  }\n\n  redirect(to, flash, reloadToken){\n    if(reloadToken){ Browser.setCookie(PHX_RELOAD_STATUS, reloadToken, 60) }\n    this.unload()\n    Browser.redirect(to, flash)\n  }\n\n  replaceMain(href, flash, callback = null, linkRef = this.setPendingLink(href)){\n    const liveReferer = this.currentLocation.href\n    this.outgoingMainEl = this.outgoingMainEl || this.main.el\n\n    const stickies = DOM.findPhxSticky(document) || []\n    const removeEls = DOM.all(this.outgoingMainEl, `[${this.binding(\"remove\")}]`)\n      .filter(el => !DOM.isChildOfAny(el, stickies))\n\n    const newMainEl = DOM.cloneNode(this.outgoingMainEl, \"\")\n    this.main.showLoader(this.loaderTimeout)\n    this.main.destroy()\n\n    this.main = this.newRootView(newMainEl, flash, liveReferer)\n    this.main.setRedirect(href)\n    this.transitionRemoves(removeEls)\n    this.main.join((joinCount, onDone) => {\n      if(joinCount === 1 && this.commitPendingLink(linkRef)){\n        this.requestDOMUpdate(() => {\n          // remove phx-remove els right before we replace the main element\n          removeEls.forEach(el => el.remove())\n          stickies.forEach(el => newMainEl.appendChild(el))\n          this.outgoingMainEl.replaceWith(newMainEl)\n          this.outgoingMainEl = null\n          callback && callback(linkRef)\n          onDone()\n        })\n      }\n    })\n  }\n\n  transitionRemoves(elements, callback){\n    let removeAttr = this.binding(\"remove\")\n    let silenceEvents = (e) => {\n      e.preventDefault()\n      e.stopImmediatePropagation()\n    }\n    elements.forEach(el => {\n      // prevent all listeners we care about from bubbling to window\n      // since we are removing the element\n      for(let event of this.boundEventNames){\n        el.addEventListener(event, silenceEvents, true)\n      }\n      this.execJS(el, el.getAttribute(removeAttr), \"remove\")\n    })\n    // remove the silenced listeners when transitions are done incase the element is re-used\n    // and call caller's callback as soon as we are done with transitions\n    this.requestDOMUpdate(() => {\n      elements.forEach(el => {\n        for(let event of this.boundEventNames){\n          el.removeEventListener(event, silenceEvents, true)\n        }\n      })\n      callback && callback()\n    })\n  }\n\n  isPhxView(el){ return el.getAttribute && el.getAttribute(PHX_SESSION) !== null }\n\n  newRootView(el, flash, liveReferer){\n    let view = new View(el, this, null, flash, liveReferer)\n    this.roots[view.id] = view\n    return view\n  }\n\n  owner(childEl, callback){\n    let view\n    const closestViewEl = childEl.closest(PHX_VIEW_SELECTOR)\n    if(closestViewEl){\n      // it can happen that we find a view that is already destroyed;\n      // in that case we DO NOT want to fallback to the main element\n      view = this.getViewByEl(closestViewEl)\n    } else {\n      view = this.main\n    }\n    return view && callback ? callback(view) : view\n  }\n\n  withinOwners(childEl, callback){\n    this.owner(childEl, view => callback(view, childEl))\n  }\n\n  getViewByEl(el){\n    let rootId = el.getAttribute(PHX_ROOT_ID)\n    return maybe(this.getRootById(rootId), root => root.getDescendentByEl(el))\n  }\n\n  getRootById(id){ return this.roots[id] }\n\n  destroyAllViews(){\n    for(let id in this.roots){\n      this.roots[id].destroy()\n      delete this.roots[id]\n    }\n    this.main = null\n  }\n\n  destroyViewByEl(el){\n    let root = this.getRootById(el.getAttribute(PHX_ROOT_ID))\n    if(root && root.id === el.id){\n      root.destroy()\n      delete this.roots[root.id]\n    } else if(root){\n      root.destroyDescendent(el.id)\n    }\n  }\n\n  getActiveElement(){\n    return document.activeElement\n  }\n\n  dropActiveElement(view){\n    if(this.prevActive && view.ownsElement(this.prevActive)){\n      this.prevActive = null\n    }\n  }\n\n  restorePreviouslyActiveFocus(){\n    if(this.prevActive && this.prevActive !== document.body){\n      this.prevActive.focus()\n    }\n  }\n\n  blurActiveElement(){\n    this.prevActive = this.getActiveElement()\n    if(this.prevActive !== document.body){ this.prevActive.blur() }\n  }\n\n  bindTopLevelEvents({dead} = {}){\n    if(this.boundTopLevelEvents){ return }\n\n    this.boundTopLevelEvents = true\n    // enter failsafe reload if server has gone away intentionally, such as \"disconnect\" broadcast\n    this.serverCloseRef = this.socket.onClose(event => {\n      // failsafe reload if normal closure and we still have a main LV\n      if(event && event.code === 1000 && this.main){ return this.reloadWithJitter(this.main) }\n    })\n    document.body.addEventListener(\"click\", function (){ }) // ensure all click events bubble for mobile Safari\n    window.addEventListener(\"pageshow\", e => {\n      if(e.persisted){ // reload page if being restored from back/forward cache\n        this.getSocket().disconnect()\n        this.withPageLoading({to: window.location.href, kind: \"redirect\"})\n        window.location.reload()\n      }\n    }, true)\n    if(!dead){ this.bindNav() }\n    this.bindClicks()\n    if(!dead){ this.bindForms() }\n    this.bind({keyup: \"keyup\", keydown: \"keydown\"}, (e, type, view, targetEl, phxEvent, _phxTarget) => {\n      let matchKey = targetEl.getAttribute(this.binding(PHX_KEY))\n      let pressedKey = e.key && e.key.toLowerCase() // chrome clicked autocompletes send a keydown without key\n      if(matchKey && matchKey.toLowerCase() !== pressedKey){ return }\n\n      let data = {key: e.key, ...this.eventMeta(type, e, targetEl)}\n      JS.exec(e, type, phxEvent, view, targetEl, [\"push\", {data}])\n    })\n    this.bind({blur: \"focusout\", focus: \"focusin\"}, (e, type, view, targetEl, phxEvent, phxTarget) => {\n      if(!phxTarget){\n        let data = {key: e.key, ...this.eventMeta(type, e, targetEl)}\n        JS.exec(e, type, phxEvent, view, targetEl, [\"push\", {data}])\n      }\n    })\n    this.bind({blur: \"blur\", focus: \"focus\"}, (e, type, view, targetEl, phxEvent, phxTarget) => {\n      // blur and focus are triggered on document and window. Discard one to avoid dups\n      if(phxTarget === \"window\"){\n        let data = this.eventMeta(type, e, targetEl)\n        JS.exec(e, type, phxEvent, view, targetEl, [\"push\", {data}])\n      }\n    })\n    this.on(\"dragover\", e => e.preventDefault())\n    this.on(\"drop\", e => {\n      e.preventDefault()\n      let dropTargetId = maybe(closestPhxBinding(e.target, this.binding(PHX_DROP_TARGET)), trueTarget => {\n        return trueTarget.getAttribute(this.binding(PHX_DROP_TARGET))\n      })\n      let dropTarget = dropTargetId && document.getElementById(dropTargetId)\n      let files = Array.from(e.dataTransfer.files || [])\n      if(!dropTarget || dropTarget.disabled || files.length === 0 || !(dropTarget.files instanceof FileList)){ return }\n\n      LiveUploader.trackFiles(dropTarget, files, e.dataTransfer)\n      dropTarget.dispatchEvent(new Event(\"input\", {bubbles: true}))\n    })\n    this.on(PHX_TRACK_UPLOADS, e => {\n      let uploadTarget = e.target\n      if(!DOM.isUploadInput(uploadTarget)){ return }\n      let files = Array.from(e.detail.files || []).filter(f => f instanceof File || f instanceof Blob)\n      LiveUploader.trackFiles(uploadTarget, files)\n      uploadTarget.dispatchEvent(new Event(\"input\", {bubbles: true}))\n    })\n  }\n\n  eventMeta(eventName, e, targetEl){\n    let callback = this.metadataCallbacks[eventName]\n    return callback ? callback(e, targetEl) : {}\n  }\n\n  setPendingLink(href){\n    this.linkRef++\n    this.pendingLink = href\n    this.resetReloadStatus()\n    return this.linkRef\n  }\n\n  // anytime we are navigating or connecting, drop reload cookie in case\n  // we issue the cookie but the next request was interrupted and the server never dropped it\n  resetReloadStatus(){ Browser.deleteCookie(PHX_RELOAD_STATUS) }\n\n  commitPendingLink(linkRef){\n    if(this.linkRef !== linkRef){\n      return false\n    } else {\n      this.href = this.pendingLink\n      this.pendingLink = null\n      return true\n    }\n  }\n\n  getHref(){ return this.href }\n\n  hasPendingLink(){ return !!this.pendingLink }\n\n  bind(events, callback){\n    for(let event in events){\n      let browserEventName = events[event]\n\n      this.on(browserEventName, e => {\n        let binding = this.binding(event)\n        let windowBinding = this.binding(`window-${event}`)\n        let targetPhxEvent = e.target.getAttribute && e.target.getAttribute(binding)\n        if(targetPhxEvent){\n          this.debounce(e.target, e, browserEventName, () => {\n            this.withinOwners(e.target, view => {\n              callback(e, event, view, e.target, targetPhxEvent, null)\n            })\n          })\n        } else {\n          DOM.all(document, `[${windowBinding}]`, el => {\n            let phxEvent = el.getAttribute(windowBinding)\n            this.debounce(el, e, browserEventName, () => {\n              this.withinOwners(el, view => {\n                callback(e, event, view, el, phxEvent, \"window\")\n              })\n            })\n          })\n        }\n      })\n    }\n  }\n\n  bindClicks(){\n    this.on(\"mousedown\", e => this.clickStartedAtTarget = e.target)\n    this.bindClick(\"click\", \"click\")\n  }\n\n  bindClick(eventName, bindingName){\n    let click = this.binding(bindingName)\n    window.addEventListener(eventName, e => {\n      let target = null\n      // a synthetic click event (detail 0) will not have caused a mousedown event,\n      // therefore the clickStartedAtTarget is stale\n      if(e.detail === 0) this.clickStartedAtTarget = e.target\n      let clickStartedAtTarget = this.clickStartedAtTarget || e.target\n      // when searching the target for the click event, we always want to\n      // use the actual event target, see #3372\n      target = closestPhxBinding(e.target, click)\n      this.dispatchClickAway(e, clickStartedAtTarget)\n      this.clickStartedAtTarget = null\n      let phxEvent = target && target.getAttribute(click)\n      if(!phxEvent){\n        if(DOM.isNewPageClick(e, window.location)){ this.unload() }\n        return\n      }\n\n      if(target.getAttribute(\"href\") === \"#\"){ e.preventDefault() }\n\n      // noop if we are in the middle of awaiting an ack for this el already\n      if(target.hasAttribute(PHX_REF_SRC)){ return }\n\n      this.debounce(target, e, \"click\", () => {\n        this.withinOwners(target, view => {\n          JS.exec(e, \"click\", phxEvent, view, target, [\"push\", {data: this.eventMeta(\"click\", e, target)}])\n        })\n      })\n    }, false)\n  }\n\n  dispatchClickAway(e, clickStartedAt){\n    let phxClickAway = this.binding(\"click-away\")\n    DOM.all(document, `[${phxClickAway}]`, el => {\n      if(!(el.isSameNode(clickStartedAt) || el.contains(clickStartedAt))){\n        this.withinOwners(el, view => {\n          let phxEvent = el.getAttribute(phxClickAway)\n          if(JS.isVisible(el) && JS.isInViewport(el)){\n            JS.exec(e, \"click\", phxEvent, view, el, [\"push\", {data: this.eventMeta(\"click\", e, e.target)}])\n          }\n        })\n      }\n    })\n  }\n\n  bindNav(){\n    if(!Browser.canPushState()){ return }\n    if(history.scrollRestoration){ history.scrollRestoration = \"manual\" }\n    let scrollTimer = null\n    window.addEventListener(\"scroll\", _e => {\n      clearTimeout(scrollTimer)\n      scrollTimer = setTimeout(() => {\n        Browser.updateCurrentState(state => Object.assign(state, {scroll: window.scrollY}))\n      }, 100)\n    })\n    window.addEventListener(\"popstate\", event => {\n      if(!this.registerNewLocation(window.location)){ return }\n      let {type, backType, id, scroll, position} = event.state || {}\n      let href = window.location.href\n\n      // Compare positions to determine direction\n      let isForward = position > this.currentHistoryPosition\n\n      type = isForward ? type : (backType || type)\n\n      // Update current position\n      this.currentHistoryPosition = position || 0\n      this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString())\n\n      DOM.dispatchEvent(window, \"phx:navigate\", {detail: {href, patch: type === \"patch\", pop: true, direction: isForward ? \"forward\" : \"backward\"}})\n      this.requestDOMUpdate(() => {\n        const callback = () => { this.maybeScroll(scroll) }\n        if(this.main.isConnected() && (type === \"patch\" && id === this.main.id)){\n          this.main.pushLinkPatch(event, href, null, callback)\n        } else {\n          this.replaceMain(href, null, callback)\n        }\n      })\n    }, false)\n    window.addEventListener(\"click\", e => {\n      let target = closestPhxBinding(e.target, PHX_LIVE_LINK)\n      let type = target && target.getAttribute(PHX_LIVE_LINK)\n      if(!type || !this.isConnected() || !this.main || DOM.wantsNewTab(e)){ return }\n\n      // When wrapping an SVG element in an anchor tag, the href can be an SVGAnimatedString\n      let href = target.href instanceof SVGAnimatedString ? target.href.baseVal : target.href\n\n      let linkState = target.getAttribute(PHX_LINK_STATE)\n      e.preventDefault()\n      e.stopImmediatePropagation() // do not bubble click to regular phx-click bindings\n      if(this.pendingLink === href){ return }\n\n      this.requestDOMUpdate(() => {\n        if(type === \"patch\"){\n          this.pushHistoryPatch(e, href, linkState, target)\n        } else if(type === \"redirect\"){\n          this.historyRedirect(e, href, linkState, null, target)\n        } else {\n          throw new Error(`expected ${PHX_LIVE_LINK} to be \"patch\" or \"redirect\", got: ${type}`)\n        }\n        let phxClick = target.getAttribute(this.binding(\"click\"))\n        if(phxClick){\n          this.requestDOMUpdate(() => this.execJS(target, phxClick, \"click\"))\n        }\n      })\n    }, false)\n  }\n\n  maybeScroll(scroll){\n    if(typeof(scroll) === \"number\"){\n      requestAnimationFrame(() => {\n        window.scrollTo(0, scroll)\n      }) // the body needs to render before we scroll.\n    }\n  }\n\n  dispatchEvent(event, payload = {}){\n    DOM.dispatchEvent(window, `phx:${event}`, {detail: payload})\n  }\n\n  dispatchEvents(events){\n    events.forEach(([event, payload]) => this.dispatchEvent(event, payload))\n  }\n\n  withPageLoading(info, callback){\n    DOM.dispatchEvent(window, \"phx:page-loading-start\", {detail: info})\n    let done = () => DOM.dispatchEvent(window, \"phx:page-loading-stop\", {detail: info})\n    return callback ? callback(done) : done\n  }\n\n  pushHistoryPatch(e, href, linkState, targetEl){\n    if(!this.isConnected() || !this.main.isMain()){ return Browser.redirect(href) }\n\n    this.withPageLoading({to: href, kind: \"patch\"}, done => {\n      this.main.pushLinkPatch(e, href, targetEl, linkRef => {\n        this.historyPatch(href, linkState, linkRef)\n        done()\n      })\n    })\n  }\n\n  historyPatch(href, linkState, linkRef = this.setPendingLink(href)){\n    if(!this.commitPendingLink(linkRef)){ return }\n\n    // Increment position for new state\n    this.currentHistoryPosition++\n    this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString())\n\n    // store the type for back navigation\n    Browser.updateCurrentState((state) => ({...state, backType: \"patch\"}))\n\n    Browser.pushState(linkState, {\n      type: \"patch\",\n      id: this.main.id,\n      position: this.currentHistoryPosition\n    }, href)\n\n    DOM.dispatchEvent(window, \"phx:navigate\", {detail: {patch: true, href, pop: false, direction: \"forward\"}})\n    this.registerNewLocation(window.location)\n  }\n\n  historyRedirect(e, href, linkState, flash, targetEl){\n    const clickLoading = targetEl && e.isTrusted && e.type !== \"popstate\"\n    if(clickLoading){ targetEl.classList.add(\"phx-click-loading\") }\n    if(!this.isConnected() || !this.main.isMain()){ return Browser.redirect(href, flash) }\n\n    // convert to full href if only path prefix\n    if(/^\\/$|^\\/[^\\/]+.*$/.test(href)){\n      let {protocol, host} = window.location\n      href = `${protocol}//${host}${href}`\n    }\n    let scroll = window.scrollY\n    this.withPageLoading({to: href, kind: \"redirect\"}, done => {\n      this.replaceMain(href, flash, (linkRef) => {\n        if(linkRef === this.linkRef){\n          // Increment position for new state\n          this.currentHistoryPosition++\n          this.sessionStorage.setItem(PHX_LV_HISTORY_POSITION, this.currentHistoryPosition.toString())\n\n          // store the type for back navigation\n          Browser.updateCurrentState((state) => ({...state, backType: \"redirect\"}))\n\n          Browser.pushState(linkState, {\n            type: \"redirect\",\n            id: this.main.id,\n            scroll: scroll,\n            position: this.currentHistoryPosition\n          }, href)\n\n          DOM.dispatchEvent(window, \"phx:navigate\", {detail: {href, patch: false, pop: false, direction: \"forward\"}})\n          this.registerNewLocation(window.location)\n        }\n        // explicitly undo click-loading class\n        // (in case it originated in a sticky live view, otherwise it would be removed anyway)\n        if(clickLoading){ targetEl.classList.remove(\"phx-click-loading\") }\n        done()\n      })\n    })\n  }\n\n  registerNewLocation(newLocation){\n    let {pathname, search} = this.currentLocation\n    if(pathname + search === newLocation.pathname + newLocation.search){\n      return false\n    } else {\n      this.currentLocation = clone(newLocation)\n      return true\n    }\n  }\n\n  bindForms(){\n    let iterations = 0\n    let externalFormSubmitted = false\n\n    // disable forms on submit that track phx-change but perform external submit\n    this.on(\"submit\", e => {\n      let phxSubmit = e.target.getAttribute(this.binding(\"submit\"))\n      let phxChange = e.target.getAttribute(this.binding(\"change\"))\n      if(!externalFormSubmitted && phxChange && !phxSubmit){\n        externalFormSubmitted = true\n        e.preventDefault()\n        this.withinOwners(e.target, view => {\n          view.disableForm(e.target)\n          // safari needs next tick\n          window.requestAnimationFrame(() => {\n            if(DOM.isUnloadableFormSubmit(e)){ this.unload() }\n            e.target.submit()\n          })\n        })\n      }\n    })\n\n    this.on(\"submit\", e => {\n      let phxEvent = e.target.getAttribute(this.binding(\"submit\"))\n      if(!phxEvent){\n        if(DOM.isUnloadableFormSubmit(e)){ this.unload() }\n        return\n      }\n      e.preventDefault()\n      e.target.disabled = true\n      this.withinOwners(e.target, view => {\n        JS.exec(e, \"submit\", phxEvent, view, e.target, [\"push\", {submitter: e.submitter}])\n      })\n    })\n\n    for(let type of [\"change\", \"input\"]){\n      this.on(type, e => {\n        if(e instanceof CustomEvent && e.target.form === undefined){\n          // throw on invalid JS.dispatch target and noop if CustomEvent triggered outside JS.dispatch\n          if(e.detail && e.detail.dispatcher){\n            throw new Error(`dispatching a custom ${type} event is only supported on input elements inside a form`)\n          }\n          return\n        }\n        let phxChange = this.binding(\"change\")\n        let input = e.target\n        // do not fire phx-change if we are in the middle of a composition session\n        // https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/isComposing\n        // Safari has issues if the input is updated while composing\n        // see https://github.com/phoenixframework/phoenix_live_view/issues/3322\n        if(e.isComposing){\n          const key = `composition-listener-${type}`\n          if(!DOM.private(input, key)){\n            DOM.putPrivate(input, key, true)\n            input.addEventListener(\"compositionend\", () => {\n              // trigger a new input/change event\n              input.dispatchEvent(new Event(type, {bubbles: true}))\n              DOM.deletePrivate(input, key)\n            }, {once: true})\n          }\n          return\n        }\n        let inputEvent = input.getAttribute(phxChange)\n        let formEvent = input.form && input.form.getAttribute(phxChange)\n        let phxEvent = inputEvent || formEvent\n        if(!phxEvent){ return }\n        if(input.type === \"number\" && input.validity && input.validity.badInput){ return }\n\n        let dispatcher = inputEvent ? input : input.form\n        let currentIterations = iterations\n        iterations++\n        let {at: at, type: lastType} = DOM.private(input, \"prev-iteration\") || {}\n        // Browsers should always fire at least one \"input\" event before every \"change\"\n        // Ignore \"change\" events, unless there was no prior \"input\" event.\n        // This could happen if user code triggers a \"change\" event, or if the browser is non-conforming.\n        if(at === currentIterations - 1 && type === \"change\" && lastType === \"input\"){ return }\n\n        DOM.putPrivate(input, \"prev-iteration\", {at: currentIterations, type: type})\n\n        this.debounce(input, e, type, () => {\n          this.withinOwners(dispatcher, view => {\n            DOM.putPrivate(input, PHX_HAS_FOCUSED, true)\n            JS.exec(e, \"change\", phxEvent, view, input, [\"push\", {_target: e.target.name, dispatcher: dispatcher}])\n          })\n        })\n      })\n    }\n    this.on(\"reset\", (e) => {\n      let form = e.target\n      DOM.resetForm(form)\n      let input = Array.from(form.elements).find(el => el.type === \"reset\")\n      if(input){\n        // wait until next tick to get updated input value\n        window.requestAnimationFrame(() => {\n          input.dispatchEvent(new Event(\"input\", {bubbles: true, cancelable: false}))\n        })\n      }\n    })\n  }\n\n  debounce(el, event, eventType, callback){\n    if(eventType === \"blur\" || eventType === \"focusout\"){ return callback() }\n\n    let phxDebounce = this.binding(PHX_DEBOUNCE)\n    let phxThrottle = this.binding(PHX_THROTTLE)\n    let defaultDebounce = this.defaults.debounce.toString()\n    let defaultThrottle = this.defaults.throttle.toString()\n\n    this.withinOwners(el, view => {\n      let asyncFilter = () => !view.isDestroyed() && document.body.contains(el)\n      DOM.debounce(el, event, phxDebounce, defaultDebounce, phxThrottle, defaultThrottle, asyncFilter, () => {\n        callback()\n      })\n    })\n  }\n\n  silenceEvents(callback){\n    this.silenced = true\n    callback()\n    this.silenced = false\n  }\n\n  on(event, callback){\n    this.boundEventNames.add(event)\n    window.addEventListener(event, e => {\n      if(!this.silenced){ callback(e) }\n    })\n  }\n\n  jsQuerySelectorAll(sourceEl, query, defaultQuery){\n    let all = this.domCallbacks.jsQuerySelectorAll\n    return all ? all(sourceEl, query, defaultQuery) : defaultQuery()\n  }\n}\n\nclass TransitionSet {\n  constructor(){\n    this.transitions = new Set()\n    this.pendingOps = []\n  }\n\n  reset(){\n    this.transitions.forEach(timer => {\n      clearTimeout(timer)\n      this.transitions.delete(timer)\n    })\n    this.flushPendingOps()\n  }\n\n  after(callback){\n    if(this.size() === 0){\n      callback()\n    } else {\n      this.pushPendingOp(callback)\n    }\n  }\n\n  addTransition(time, onStart, onDone){\n    onStart()\n    let timer = setTimeout(() => {\n      this.transitions.delete(timer)\n      onDone()\n      this.flushPendingOps()\n    }, time)\n    this.transitions.add(timer)\n  }\n\n  pushPendingOp(op){ this.pendingOps.push(op) }\n\n  size(){ return this.transitions.size }\n\n  flushPendingOps(){\n    if(this.size() > 0){ return }\n    let op = this.pendingOps.shift()\n    if(op){\n      op()\n      this.flushPendingOps()\n    }\n  }\n}\n", "/*\n================================================================================\nPhoenix LiveView JavaScript Client\n================================================================================\n\nSee the hexdocs at `https://hexdocs.pm/phoenix_live_view` for documentation.\n\n*/\n\nimport LiveSocket, {isUsedInput} from \"./live_socket\"\nimport DOM from \"./dom\"\nimport ViewHook from \"./view_hook\"\nimport View from \"./view\"\n\n/** Creates a ViewHook instance for the given element and callbacks.\n *\n * @param {HTMLElement} el - The element to associate with the hook.\n * @param {Object} [callbacks] - The list of hook callbacks, such as mounted,\n *   updated, destroyed, etc.\n *\n * @example\n *\n * class MyComponent extends HTMLElement {\n *   connectedCallback(){\n *     let onLiveViewMounted = () => this.hook.pushEvent(...))\n *     this.hook = createHook(this, {mounted: onLiveViewMounted})\n *   }\n * }\n *\n * *Note*: `createHook` must be called from the `connectedCallback` lifecycle\n * which is triggered after the element has been added to the DOM. If you try\n * to call `createHook` from the constructor, an error will be logged.\n *\n * @returns {ViewHook} Returns the ViewHook instance for the custom element.\n */\nlet createHook = (el, callbacks = {}) => {\n  let existingHook = DOM.getCustomElHook(el)\n  if(existingHook){ return existingHook }\n\n  let hook = new ViewHook(View.closestView(el), el, callbacks)\n  DOM.putCustomElHook(el, hook)\n  return hook\n}\n\nexport {\n  LiveSocket,\n  isUsedInput,\n  createHook\n}\n", "// If you want to use Phoenix channels, run `mix help phx.gen.channel`\n// to get started and then uncomment the line below.\n// import \"./user_socket.js\"\n\n// You can include dependencies in two ways.\n//\n// The simplest option is to put them in assets/vendor and\n// import them using relative paths:\n//\n//     import \"../vendor/some-package.js\"\n//\n// Alternatively, you can `npm install some-package --prefix assets` and import\n// them using a path starting with the package name:\n//\n//     import \"some-package\"\n//\n\n// Include phoenix_html to handle method=PUT/DELETE in forms and buttons.\nimport \"phoenix_html\"\n// Establish Phoenix Socket and LiveView configuration.\nimport {Socket} from \"phoenix\"\nimport {LiveSocket} from \"phoenix_live_view\"\nimport topbar from \"../vendor/topbar\"\n\nlet csrfToken = document.querySelector(\"meta[name='csrf-token']\").getAttribute(\"content\")\nlet liveSocket = new LiveSocket(\"/live\", Socket, {\n  longPollFallbackMs: 2500,\n  params: {_csrf_token: csrfToken}\n})\n\n// Show progress bar on live navigation and form submits\ntopbar.config({barColors: {0: \"#29d\"}, shadowColor: \"rgba(0, 0, 0, .3)\"})\nwindow.addEventListener(\"phx:page-loading-start\", _info => topbar.show(300))\nwindow.addEventListener(\"phx:page-loading-stop\", _info => topbar.hide())\n\n// connect if there are any LiveViews on the page\nliveSocket.connect()\n\n// expose liveSocket on window for web console debug logs and latency simulation:\n// >> liveSocket.enableDebug()\n// >> liveSocket.enableLatencySim(1000)  // enabled for duration of browser session\n// >> liveSocket.disableLatencySim()\nwindow.liveSocket = liveSocket\n\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAMA,OAAC,SAAUA,SAAQC,WAAU;AAC3B;AAGA,SAAC,WAAY;AACX,cAAI,WAAW;AACf,cAAI,UAAU,CAAC,MAAM,OAAO,UAAU,GAAG;AACzC,mBAAS,IAAI,GAAG,IAAI,QAAQ,UAAU,CAACD,QAAO,uBAAuB,EAAE,GAAG;AACxE,YAAAA,QAAO,wBACLA,QAAO,QAAQ,CAAC,IAAI,uBAAuB;AAC7C,YAAAA,QAAO,uBACLA,QAAO,QAAQ,CAAC,IAAI,sBAAsB,KAC1CA,QAAO,QAAQ,CAAC,IAAI,6BAA6B;AAAA,UACrD;AACA,cAAI,CAACA,QAAO;AACV,YAAAA,QAAO,wBAAwB,SAAU,UAAU,SAAS;AAC1D,kBAAI,YAAW,oBAAI,KAAK,GAAE,QAAQ;AAClC,kBAAI,aAAa,KAAK,IAAI,GAAG,MAAM,WAAW,SAAS;AACvD,kBAAI,KAAKA,QAAO,WAAW,WAAY;AACrC,yBAAS,WAAW,UAAU;AAAA,cAChC,GAAG,UAAU;AACb,yBAAW,WAAW;AACtB,qBAAO;AAAA,YACT;AACF,cAAI,CAACA,QAAO;AACV,YAAAA,QAAO,uBAAuB,SAAU,IAAI;AAC1C,2BAAa,EAAE;AAAA,YACjB;AAAA,QACJ,GAAG;AAEH,YAAI,QACF,iBACA,SACA,kBAAkB,MAClB,cAAc,MACd,eAAe,MACf,WAAW,SAAU,MAAM,MAAM,SAAS;AACxC,cAAI,KAAK;AAAkB,iBAAK,iBAAiB,MAAM,SAAS,KAAK;AAAA,mBAC5D,KAAK;AAAa,iBAAK,YAAY,OAAO,MAAM,OAAO;AAAA;AAC3D,iBAAK,OAAO,IAAI,IAAI;AAAA,QAC3B,GACA,UAAU;AAAA,UACR,SAAS;AAAA,UACT,cAAc;AAAA,UACd,WAAW;AAAA,YACT,GAAG;AAAA,YACH,OAAO;AAAA,YACP,OAAO;AAAA,YACP,OAAO;AAAA,YACP,OAAO;AAAA,UACT;AAAA,UACA,YAAY;AAAA,UACZ,aAAa;AAAA,UACb,WAAW;AAAA,QACb,GACA,UAAU,WAAY;AACpB,iBAAO,QAAQA,QAAO;AACtB,iBAAO,SAAS,QAAQ,eAAe;AAEvC,cAAI,MAAM,OAAO,WAAW,IAAI;AAChC,cAAI,aAAa,QAAQ;AACzB,cAAI,cAAc,QAAQ;AAE1B,cAAI,eAAe,IAAI,qBAAqB,GAAG,GAAG,OAAO,OAAO,CAAC;AACjE,mBAAS,QAAQ,QAAQ;AACvB,yBAAa,aAAa,MAAM,QAAQ,UAAU,IAAI,CAAC;AACzD,cAAI,YAAY,QAAQ;AACxB,cAAI,UAAU;AACd,cAAI,OAAO,GAAG,QAAQ,eAAe,CAAC;AACtC,cAAI;AAAA,YACF,KAAK,KAAK,kBAAkB,OAAO,KAAK;AAAA,YACxC,QAAQ,eAAe;AAAA,UACzB;AACA,cAAI,cAAc;AAClB,cAAI,OAAO;AAAA,QACb,GACA,eAAe,WAAY;AACzB,mBAASC,UAAS,cAAc,QAAQ;AACxC,cAAI,QAAQ,OAAO;AACnB,gBAAM,WAAW;AACjB,gBAAM,MAAM,MAAM,OAAO,MAAM,QAAQ,MAAM,SAAS,MAAM,UAAU;AACtE,gBAAM,SAAS;AACf,gBAAM,UAAU;AAChB,cAAI,QAAQ;AAAW,mBAAO,UAAU,IAAI,QAAQ,SAAS;AAC7D,UAAAA,UAAS,KAAK,YAAY,MAAM;AAChC,mBAASD,SAAQ,UAAU,OAAO;AAAA,QACpC,GACAE,UAAS;AAAA,UACP,QAAQ,SAAU,MAAM;AACtB,qBAAS,OAAO;AACd,kBAAI,QAAQ,eAAe,GAAG;AAAG,wBAAQ,GAAG,IAAI,KAAK,GAAG;AAAA,UAC5D;AAAA,UACA,MAAM,SAAU,OAAO;AACrB,gBAAI;AAAS;AACb,gBAAI,OAAO;AACT,kBAAI;AAAc;AAClB,6BAAe,WAAW,MAAMA,QAAO,KAAK,GAAG,KAAK;AAAA,YACtD,OAAQ;AACN,wBAAU;AACV,kBAAI,gBAAgB;AAAM,gBAAAF,QAAO,qBAAqB,WAAW;AACjE,kBAAI,CAAC;AAAQ,6BAAa;AAC1B,qBAAO,MAAM,UAAU;AACvB,qBAAO,MAAM,UAAU;AACvB,cAAAE,QAAO,SAAS,CAAC;AACjB,kBAAI,QAAQ,SAAS;AACnB,iBAAC,SAAS,OAAO;AACf,oCAAkBF,QAAO,sBAAsB,IAAI;AACnD,kBAAAE,QAAO;AAAA,oBACL,MAAM,OAAO,KAAK,IAAI,IAAI,KAAK,KAAK,eAAe,GAAG,CAAC;AAAA,kBACzD;AAAA,gBACF,GAAG;AAAA,cACL;AAAA,YACF;AAAA,UACF;AAAA,UACA,UAAU,SAAU,IAAI;AACtB,gBAAI,OAAO,OAAO;AAAa,qBAAO;AACtC,gBAAI,OAAO,OAAO,UAAU;AAC1B,oBACG,GAAG,QAAQ,GAAG,KAAK,KAAK,GAAG,QAAQ,GAAG,KAAK,IACxC,kBACA,KAAK,WAAW,EAAE;AAAA,YAC1B;AACA,8BAAkB,KAAK,IAAI,IAAI;AAC/B,oBAAQ;AACR,mBAAO;AAAA,UACT;AAAA,UACA,MAAM,WAAY;AAChB,yBAAa,YAAY;AACzB,2BAAe;AACf,gBAAI,CAAC;AAAS;AACd,sBAAU;AACV,gBAAI,mBAAmB,MAAM;AAC3B,cAAAF,QAAO,qBAAqB,eAAe;AAC3C,gCAAkB;AAAA,YACpB;AACA,aAAC,SAAS,OAAO;AACf,kBAAIE,QAAO,SAAS,KAAK,KAAK,GAAG;AAC/B,uBAAO,MAAM,WAAW;AACxB,oBAAI,OAAO,MAAM,WAAW,MAAM;AAChC,yBAAO,MAAM,UAAU;AACvB,gCAAc;AACd;AAAA,gBACF;AAAA,cACF;AACA,4BAAcF,QAAO,sBAAsB,IAAI;AAAA,YACjD,GAAG;AAAA,UACL;AAAA,QACF;AAEF,YAAI,OAAO,WAAW,YAAY,OAAO,OAAO,YAAY,UAAU;AACpE,iBAAO,UAAUE;AAAA,QACnB,WAAW,OAAO,WAAW,cAAc,OAAO,KAAK;AACrD,iBAAO,WAAY;AACjB,mBAAOA;AAAA,UACT,CAAC;AAAA,QACH,OAAO;AACL,eAAK,SAASA;AAAA,QAChB;AAAA,MACF,GAAE,KAAK,SAAM,QAAQ,QAAQ;AAAA;AAAA;;;AClK7B,GAAC,WAAW;AACV,QAAI,gBAAgB,iBAAiB;AAErC,aAAS,mBAAmB;AAC1B,UAAI,OAAO,OAAO,gBAAgB;AAAY,eAAO,OAAO;AAE5D,eAASC,aAAY,OAAO,QAAQ;AAClC,iBAAS,UAAU,EAAC,SAAS,OAAO,YAAY,OAAO,QAAQ,OAAS;AACxE,YAAI,MAAM,SAAS,YAAY,aAAa;AAC5C,YAAI,gBAAgB,OAAO,OAAO,SAAS,OAAO,YAAY,OAAO,MAAM;AAC3E,eAAO;AAAA,MACT;AACA,MAAAA,aAAY,YAAY,OAAO,MAAM;AACrC,aAAOA;AAAA,IACT;AAEA,aAAS,iBAAiB,MAAM,OAAO;AACrC,UAAI,QAAQ,SAAS,cAAc,OAAO;AAC1C,YAAM,OAAO;AACb,YAAM,OAAO;AACb,YAAM,QAAQ;AACd,aAAO;AAAA,IACT;AAEA,aAAS,YAAY,SAAS,mBAAmB;AAC/C,UAAI,KAAK,QAAQ,aAAa,SAAS,GACnC,SAAS,iBAAiB,WAAW,QAAQ,aAAa,aAAa,CAAC,GACxE,OAAO,iBAAiB,eAAe,QAAQ,aAAa,WAAW,CAAC,GACxE,OAAO,SAAS,cAAc,MAAM,GACpC,SAAS,SAAS,cAAc,OAAO,GACvC,SAAS,QAAQ,aAAa,QAAQ;AAE1C,WAAK,SAAU,QAAQ,aAAa,aAAa,MAAM,QAAS,QAAQ;AACxE,WAAK,SAAS;AACd,WAAK,MAAM,UAAU;AAErB,UAAI;AAAQ,aAAK,SAAS;AAAA,eACjB;AAAmB,aAAK,SAAS;AAE1C,WAAK,YAAY,IAAI;AACrB,WAAK,YAAY,MAAM;AACvB,eAAS,KAAK,YAAY,IAAI;AAI9B,aAAO,OAAO;AACd,WAAK,YAAY,MAAM;AACvB,aAAO,MAAM;AAAA,IACf;AAEA,WAAO,iBAAiB,SAAS,SAAS,GAAG;AAC3C,UAAI,UAAU,EAAE;AAChB,UAAI,EAAE;AAAkB;AAExB,aAAO,WAAW,QAAQ,cAAc;AACtC,YAAI,mBAAmB,IAAI,cAAc,sBAAsB;AAAA,UAC7D,WAAW;AAAA,UAAM,cAAc;AAAA,QACjC,CAAC;AAED,YAAI,CAAC,QAAQ,cAAc,gBAAgB,GAAG;AAC5C,YAAE,eAAe;AACjB,YAAE,yBAAyB;AAC3B,iBAAO;AAAA,QACT;AAEA,YAAI,QAAQ,aAAa,aAAa,KAAK,QAAQ,aAAa,SAAS,GAAG;AAC1E,sBAAY,SAAS,EAAE,WAAW,EAAE,QAAQ;AAC5C,YAAE,eAAe;AACjB,iBAAO;AAAA,QACT,OAAO;AACL,oBAAU,QAAQ;AAAA,QACpB;AAAA,MACF;AAAA,IACF,GAAG,KAAK;AAER,WAAO,iBAAiB,sBAAsB,SAAU,GAAG;AACzD,UAAI,UAAU,EAAE,OAAO,aAAa,cAAc;AAClD,UAAG,WAAW,CAAC,OAAO,QAAQ,OAAO,GAAG;AACtC,UAAE,eAAe;AAAA,MACnB;AAAA,IACF,GAAG,KAAK;AAAA,EACV,GAAG;;;AClFI,MAAI,UAAU,CAAC,UAAU;AAC9B,QAAG,OAAO,UAAU,YAAW;AAC7B,aAAO;IACT,OAAO;AACL,UAAIC,YAAU,WAAW;AAAE,eAAO;MAAM;AACxC,aAAOA;IACT;EACF;ACRO,MAAM,aAAa,OAAO,SAAS,cAAc,OAAO;AACxD,MAAM,YAAY,OAAO,WAAW,cAAc,SAAS;AAC3D,MAAM,SAAS,cAAc,aAAa;AAC1C,MAAM,cAAc;AACpB,MAAM,gBAAgB,EAAC,YAAY,GAAG,MAAM,GAAG,SAAS,GAAG,QAAQ,EAAC;AACpE,MAAM,kBAAkB;AACxB,MAAM,kBAAkB;AACxB,MAAM,iBAAiB;IAC5B,QAAQ;IACR,SAAS;IACT,QAAQ;IACR,SAAS;IACT,SAAS;EACX;AACO,MAAM,iBAAiB;IAC5B,OAAO;IACP,OAAO;IACP,MAAM;IACN,OAAO;IACP,OAAO;EACT;AAEO,MAAM,aAAa;IACxB,UAAU;IACV,WAAW;EACb;AACO,MAAM,aAAa;IACxB,UAAU;EACZ;ACrBA,MAAqB,OAArB,MAA0B;IACxB,YAAY,SAAS,OAAO,SAAS,SAAQ;AAC3C,WAAK,UAAU;AACf,WAAK,QAAQ;AACb,WAAK,UAAU,WAAW,WAAW;AAAE,eAAO,CAAC;MAAE;AACjD,WAAK,eAAe;AACpB,WAAK,UAAU;AACf,WAAK,eAAe;AACpB,WAAK,WAAW,CAAC;AACjB,WAAK,OAAO;IACd;;;;;IAMA,OAAO,SAAQ;AACb,WAAK,UAAU;AACf,WAAK,MAAM;AACX,WAAK,KAAK;IACZ;;;;IAKA,OAAM;AACJ,UAAG,KAAK,YAAY,SAAS,GAAE;AAAE;MAAO;AACxC,WAAK,aAAa;AAClB,WAAK,OAAO;AACZ,WAAK,QAAQ,OAAO,KAAK;QACvB,OAAO,KAAK,QAAQ;QACpB,OAAO,KAAK;QACZ,SAAS,KAAK,QAAQ;QACtB,KAAK,KAAK;QACV,UAAU,KAAK,QAAQ,QAAQ;MACjC,CAAC;IACH;;;;;;IAOA,QAAQ,QAAQ,UAAS;AACvB,UAAG,KAAK,YAAY,MAAM,GAAE;AAC1B,iBAAS,KAAK,aAAa,QAAQ;MACrC;AAEA,WAAK,SAAS,KAAK,EAAC,QAAQ,SAAQ,CAAC;AACrC,aAAO;IACT;;;;IAKA,QAAO;AACL,WAAK,eAAe;AACpB,WAAK,MAAM;AACX,WAAK,WAAW;AAChB,WAAK,eAAe;AACpB,WAAK,OAAO;IACd;;;;IAKA,aAAa,EAAC,QAAQ,UAAU,KAAI,GAAE;AACpC,WAAK,SAAS,OAAO,CAAA,MAAK,EAAE,WAAW,MAAM,EAC1C,QAAQ,CAAA,MAAK,EAAE,SAAS,QAAQ,CAAC;IACtC;;;;IAKA,iBAAgB;AACd,UAAG,CAAC,KAAK,UAAS;AAAE;MAAO;AAC3B,WAAK,QAAQ,IAAI,KAAK,QAAQ;IAChC;;;;IAKA,gBAAe;AACb,mBAAa,KAAK,YAAY;AAC9B,WAAK,eAAe;IACtB;;;;IAKA,eAAc;AACZ,UAAG,KAAK,cAAa;AAAE,aAAK,cAAc;MAAE;AAC5C,WAAK,MAAM,KAAK,QAAQ,OAAO,QAAQ;AACvC,WAAK,WAAW,KAAK,QAAQ,eAAe,KAAK,GAAG;AAEpD,WAAK,QAAQ,GAAG,KAAK,UAAU,CAAA,YAAW;AACxC,aAAK,eAAe;AACpB,aAAK,cAAc;AACnB,aAAK,eAAe;AACpB,aAAK,aAAa,OAAO;MAC3B,CAAC;AAED,WAAK,eAAe,WAAW,MAAM;AACnC,aAAK,QAAQ,WAAW,CAAC,CAAC;MAC5B,GAAG,KAAK,OAAO;IACjB;;;;IAKA,YAAY,QAAO;AACjB,aAAO,KAAK,gBAAgB,KAAK,aAAa,WAAW;IAC3D;;;;IAKA,QAAQ,QAAQ,UAAS;AACvB,WAAK,QAAQ,QAAQ,KAAK,UAAU,EAAC,QAAQ,SAAQ,CAAC;IACxD;EACF;AC9GA,MAAqB,QAArB,MAA2B;IACzB,YAAY,UAAU,WAAU;AAC9B,WAAK,WAAW;AAChB,WAAK,YAAY;AACjB,WAAK,QAAQ;AACb,WAAK,QAAQ;IACf;IAEA,QAAO;AACL,WAAK,QAAQ;AACb,mBAAa,KAAK,KAAK;IACzB;;;;IAKA,kBAAiB;AACf,mBAAa,KAAK,KAAK;AAEvB,WAAK,QAAQ,WAAW,MAAM;AAC5B,aAAK,QAAQ,KAAK,QAAQ;AAC1B,aAAK,SAAS;MAChB,GAAG,KAAK,UAAU,KAAK,QAAQ,CAAC,CAAC;IACnC;EACF;AC1BA,MAAqB,UAArB,MAA6B;IAC3B,YAAY,OAAO,QAAQ,QAAO;AAChC,WAAK,QAAQ,eAAe;AAC5B,WAAK,QAAQ;AACb,WAAK,SAAS,QAAQ,UAAU,CAAC,CAAC;AAClC,WAAK,SAAS;AACd,WAAK,WAAW,CAAC;AACjB,WAAK,aAAa;AAClB,WAAK,UAAU,KAAK,OAAO;AAC3B,WAAK,aAAa;AAClB,WAAK,WAAW,IAAI,KAAK,MAAM,eAAe,MAAM,KAAK,QAAQ,KAAK,OAAO;AAC7E,WAAK,aAAa,CAAC;AACnB,WAAK,kBAAkB,CAAC;AAExB,WAAK,cAAc,IAAI,MAAM,MAAM;AACjC,YAAG,KAAK,OAAO,YAAY,GAAE;AAAE,eAAK,OAAO;QAAE;MAC/C,GAAG,KAAK,OAAO,aAAa;AAC5B,WAAK,gBAAgB,KAAK,KAAK,OAAO,QAAQ,MAAM,KAAK,YAAY,MAAM,CAAC,CAAC;AAC7E,WAAK,gBAAgB;QAAK,KAAK,OAAO,OAAO,MAAM;AACjD,eAAK,YAAY,MAAM;AACvB,cAAG,KAAK,UAAU,GAAE;AAAE,iBAAK,OAAO;UAAE;QACtC,CAAC;MACD;AACA,WAAK,SAAS,QAAQ,MAAM,MAAM;AAChC,aAAK,QAAQ,eAAe;AAC5B,aAAK,YAAY,MAAM;AACvB,aAAK,WAAW,QAAQ,CAAA,cAAa,UAAU,KAAK,CAAC;AACrD,aAAK,aAAa,CAAC;MACrB,CAAC;AACD,WAAK,SAAS,QAAQ,SAAS,MAAM;AACnC,aAAK,QAAQ,eAAe;AAC5B,YAAG,KAAK,OAAO,YAAY,GAAE;AAAE,eAAK,YAAY,gBAAgB;QAAE;MACpE,CAAC;AACD,WAAK,QAAQ,MAAM;AACjB,aAAK,YAAY,MAAM;AACvB,YAAG,KAAK,OAAO,UAAU;AAAG,eAAK,OAAO,IAAI,WAAW,SAAS,KAAK,SAAS,KAAK,QAAQ,GAAG;AAC9F,aAAK,QAAQ,eAAe;AAC5B,aAAK,OAAO,OAAO,IAAI;MACzB,CAAC;AACD,WAAK,QAAQ,CAAA,WAAU;AACrB,YAAG,KAAK,OAAO,UAAU;AAAG,eAAK,OAAO,IAAI,WAAW,SAAS,KAAK,SAAS,MAAM;AACpF,YAAG,KAAK,UAAU,GAAE;AAAE,eAAK,SAAS,MAAM;QAAE;AAC5C,aAAK,QAAQ,eAAe;AAC5B,YAAG,KAAK,OAAO,YAAY,GAAE;AAAE,eAAK,YAAY,gBAAgB;QAAE;MACpE,CAAC;AACD,WAAK,SAAS,QAAQ,WAAW,MAAM;AACrC,YAAG,KAAK,OAAO,UAAU;AAAG,eAAK,OAAO,IAAI,WAAW,WAAW,KAAK,UAAU,KAAK,QAAQ,MAAM,KAAK,SAAS,OAAO;AACzH,YAAI,YAAY,IAAI,KAAK,MAAM,eAAe,OAAO,QAAQ,CAAC,CAAC,GAAG,KAAK,OAAO;AAC9E,kBAAU,KAAK;AACf,aAAK,QAAQ,eAAe;AAC5B,aAAK,SAAS,MAAM;AACpB,YAAG,KAAK,OAAO,YAAY,GAAE;AAAE,eAAK,YAAY,gBAAgB;QAAE;MACpE,CAAC;AACD,WAAK,GAAG,eAAe,OAAO,CAAC,SAAS,QAAQ;AAC9C,aAAK,QAAQ,KAAK,eAAe,GAAG,GAAG,OAAO;MAChD,CAAC;IACH;;;;;;IAOA,KAAK,UAAU,KAAK,SAAQ;AAC1B,UAAG,KAAK,YAAW;AACjB,cAAM,IAAI,MAAM,4FAA4F;MAC9G,OAAO;AACL,aAAK,UAAU;AACf,aAAK,aAAa;AAClB,aAAK,OAAO;AACZ,eAAO,KAAK;MACd;IACF;;;;;IAMA,QAAQ,UAAS;AACf,WAAK,GAAG,eAAe,OAAO,QAAQ;IACxC;;;;;IAMA,QAAQ,UAAS;AACf,aAAO,KAAK,GAAG,eAAe,OAAO,CAAA,WAAU,SAAS,MAAM,CAAC;IACjE;;;;;;;;;;;;;;;;;;IAmBA,GAAG,OAAO,UAAS;AACjB,UAAI,MAAM,KAAK;AACf,WAAK,SAAS,KAAK,EAAC,OAAO,KAAK,SAAQ,CAAC;AACzC,aAAO;IACT;;;;;;;;;;;;;;;;;;;IAoBA,IAAI,OAAO,KAAI;AACb,WAAK,WAAW,KAAK,SAAS,OAAO,CAAC,SAAS;AAC7C,eAAO,EAAE,KAAK,UAAU,UAAU,OAAO,QAAQ,eAAe,QAAQ,KAAK;MAC/E,CAAC;IACH;;;;IAKA,UAAS;AAAE,aAAO,KAAK,OAAO,YAAY,KAAK,KAAK,SAAS;IAAE;;;;;;;;;;;;;;;;;IAkB/D,KAAK,OAAO,SAAS,UAAU,KAAK,SAAQ;AAC1C,gBAAU,WAAW,CAAC;AACtB,UAAG,CAAC,KAAK,YAAW;AAClB,cAAM,IAAI,MAAM,kBAAkB,cAAc,KAAK,iEAAiE;MACxH;AACA,UAAI,YAAY,IAAI,KAAK,MAAM,OAAO,WAAW;AAAE,eAAO;MAAQ,GAAG,OAAO;AAC5E,UAAG,KAAK,QAAQ,GAAE;AAChB,kBAAU,KAAK;MACjB,OAAO;AACL,kBAAU,aAAa;AACvB,aAAK,WAAW,KAAK,SAAS;MAChC;AAEA,aAAO;IACT;;;;;;;;;;;;;;;;;IAkBA,MAAM,UAAU,KAAK,SAAQ;AAC3B,WAAK,YAAY,MAAM;AACvB,WAAK,SAAS,cAAc;AAE5B,WAAK,QAAQ,eAAe;AAC5B,UAAI,UAAU,MAAM;AAClB,YAAG,KAAK,OAAO,UAAU;AAAG,eAAK,OAAO,IAAI,WAAW,SAAS,KAAK,OAAO;AAC5E,aAAK,QAAQ,eAAe,OAAO,OAAO;MAC5C;AACA,UAAI,YAAY,IAAI,KAAK,MAAM,eAAe,OAAO,QAAQ,CAAC,CAAC,GAAG,OAAO;AACzE,gBAAU,QAAQ,MAAM,MAAM,QAAQ,CAAC,EACpC,QAAQ,WAAW,MAAM,QAAQ,CAAC;AACrC,gBAAU,KAAK;AACf,UAAG,CAAC,KAAK,QAAQ,GAAE;AAAE,kBAAU,QAAQ,MAAM,CAAC,CAAC;MAAE;AAEjD,aAAO;IACT;;;;;;;;;;;;;IAcA,UAAU,QAAQ,SAAS,MAAK;AAAE,aAAO;IAAQ;;;;IAKjD,SAAS,OAAO,OAAO,SAAS,SAAQ;AACtC,UAAG,KAAK,UAAU,OAAM;AAAE,eAAO;MAAM;AAEvC,UAAG,WAAW,YAAY,KAAK,QAAQ,GAAE;AACvC,YAAG,KAAK,OAAO,UAAU;AAAG,eAAK,OAAO,IAAI,WAAW,6BAA6B,EAAC,OAAO,OAAO,SAAS,QAAO,CAAC;AACpH,eAAO;MACT,OAAO;AACL,eAAO;MACT;IACF;;;;IAKA,UAAS;AAAE,aAAO,KAAK,SAAS;IAAI;;;;IAKpC,OAAO,UAAU,KAAK,SAAQ;AAC5B,UAAG,KAAK,UAAU,GAAE;AAAE;MAAO;AAC7B,WAAK,OAAO,eAAe,KAAK,KAAK;AACrC,WAAK,QAAQ,eAAe;AAC5B,WAAK,SAAS,OAAO,OAAO;IAC9B;;;;IAKA,QAAQ,OAAO,SAAS,KAAK,SAAQ;AACnC,UAAI,iBAAiB,KAAK,UAAU,OAAO,SAAS,KAAK,OAAO;AAChE,UAAG,WAAW,CAAC,gBAAe;AAAE,cAAM,IAAI,MAAM,6EAA6E;MAAE;AAE/H,UAAI,gBAAgB,KAAK,SAAS,OAAO,CAAA,SAAQ,KAAK,UAAU,KAAK;AAErE,eAAQ,IAAI,GAAG,IAAI,cAAc,QAAQ,KAAI;AAC3C,YAAI,OAAO,cAAc,CAAC;AAC1B,aAAK,SAAS,gBAAgB,KAAK,WAAW,KAAK,QAAQ,CAAC;MAC9D;IACF;;;;IAKA,eAAe,KAAI;AAAE,aAAO,cAAc;IAAM;;;;IAKhD,WAAU;AAAE,aAAO,KAAK,UAAU,eAAe;IAAO;;;;IAKxD,YAAW;AAAE,aAAO,KAAK,UAAU,eAAe;IAAQ;;;;IAK1D,WAAU;AAAE,aAAO,KAAK,UAAU,eAAe;IAAO;;;;IAKxD,YAAW;AAAE,aAAO,KAAK,UAAU,eAAe;IAAQ;;;;IAK1D,YAAW;AAAE,aAAO,KAAK,UAAU,eAAe;IAAQ;EAC5D;ACjTA,MAAqB,OAArB,MAA0B;IAExB,OAAO,QAAQ,QAAQ,UAAU,QAAQ,MAAM,SAAS,WAAW,UAAS;AAC1E,UAAG,OAAO,gBAAe;AACvB,YAAI,MAAM,IAAI,OAAO,eAAe;AACpC,eAAO,KAAK,eAAe,KAAK,QAAQ,UAAU,MAAM,SAAS,WAAW,QAAQ;MACtF,OAAO;AACL,YAAI,MAAM,IAAI,OAAO,eAAe;AACpC,eAAO,KAAK,WAAW,KAAK,QAAQ,UAAU,QAAQ,MAAM,SAAS,WAAW,QAAQ;MAC1F;IACF;IAEA,OAAO,eAAe,KAAK,QAAQ,UAAU,MAAM,SAAS,WAAW,UAAS;AAC9E,UAAI,UAAU;AACd,UAAI,KAAK,QAAQ,QAAQ;AACzB,UAAI,SAAS,MAAM;AACjB,YAAI,WAAW,KAAK,UAAU,IAAI,YAAY;AAC9C,oBAAY,SAAS,QAAQ;MAC/B;AACA,UAAG,WAAU;AAAE,YAAI,YAAY;MAAU;AAGzC,UAAI,aAAa,MAAM;MAAE;AAEzB,UAAI,KAAK,IAAI;AACb,aAAO;IACT;IAEA,OAAO,WAAW,KAAK,QAAQ,UAAU,QAAQ,MAAM,SAAS,WAAW,UAAS;AAClF,UAAI,KAAK,QAAQ,UAAU,IAAI;AAC/B,UAAI,UAAU;AACd,UAAI,iBAAiB,gBAAgB,MAAM;AAC3C,UAAI,UAAU,MAAM,YAAY,SAAS,IAAI;AAC7C,UAAI,qBAAqB,MAAM;AAC7B,YAAG,IAAI,eAAe,WAAW,YAAY,UAAS;AACpD,cAAI,WAAW,KAAK,UAAU,IAAI,YAAY;AAC9C,mBAAS,QAAQ;QACnB;MACF;AACA,UAAG,WAAU;AAAE,YAAI,YAAY;MAAU;AAEzC,UAAI,KAAK,IAAI;AACb,aAAO;IACT;IAEA,OAAO,UAAU,MAAK;AACpB,UAAG,CAAC,QAAQ,SAAS,IAAG;AAAE,eAAO;MAAK;AAEtC,UAAI;AACF,eAAO,KAAK,MAAM,IAAI;MACxB,SAAS,GAAT;AACE,mBAAW,QAAQ,IAAI,iCAAiC,IAAI;AAC5D,eAAO;MACT;IACF;IAEA,OAAO,UAAU,KAAK,WAAU;AAC9B,UAAI,WAAW,CAAC;AAChB,eAAQ,OAAO,KAAI;AACjB,YAAG,CAAC,OAAO,UAAU,eAAe,KAAK,KAAK,GAAG,GAAE;AAAE;QAAS;AAC9D,YAAI,WAAW,YAAY,GAAG,aAAa,SAAS;AACpD,YAAI,WAAW,IAAI,GAAG;AACtB,YAAG,OAAO,aAAa,UAAS;AAC9B,mBAAS,KAAK,KAAK,UAAU,UAAU,QAAQ,CAAC;QAClD,OAAO;AACL,mBAAS,KAAK,mBAAmB,QAAQ,IAAI,MAAM,mBAAmB,QAAQ,CAAC;QACjF;MACF;AACA,aAAO,SAAS,KAAK,GAAG;IAC1B;IAEA,OAAO,aAAa,KAAK,QAAO;AAC9B,UAAG,OAAO,KAAK,MAAM,EAAE,WAAW,GAAE;AAAE,eAAO;MAAI;AAEjD,UAAI,SAAS,IAAI,MAAM,IAAI,IAAI,MAAM;AACrC,aAAO,GAAG,MAAM,SAAS,KAAK,UAAU,MAAM;IAChD;EACF;AC3EA,MAAI,sBAAsB,CAAC,WAAW;AACpC,QAAI,SAAS;AACb,QAAI,QAAQ,IAAI,WAAW,MAAM;AACjC,QAAI,MAAM,MAAM;AAChB,aAAQ,IAAI,GAAG,IAAI,KAAK,KAAI;AAAE,gBAAU,OAAO,aAAa,MAAM,CAAC,CAAC;IAAE;AACtE,WAAO,KAAK,MAAM;EACpB;AAEA,MAAqB,WAArB,MAA8B;IAE5B,YAAY,UAAS;AACnB,WAAK,WAAW;AAChB,WAAK,QAAQ;AACb,WAAK,gBAAgB;AACrB,WAAK,OAAO,oBAAI,IAAI;AACpB,WAAK,mBAAmB;AACxB,WAAK,eAAe;AACpB,WAAK,oBAAoB;AACzB,WAAK,cAAc,CAAC;AACpB,WAAK,SAAS,WAAW;MAAE;AAC3B,WAAK,UAAU,WAAW;MAAE;AAC5B,WAAK,YAAY,WAAW;MAAE;AAC9B,WAAK,UAAU,WAAW;MAAE;AAC5B,WAAK,eAAe,KAAK,kBAAkB,QAAQ;AACnD,WAAK,aAAa,cAAc;AAEhC,iBAAW,MAAM,KAAK,KAAK,GAAG,CAAC;IACjC;IAEA,kBAAkB,UAAS;AACzB,aAAQ,SACL,QAAQ,SAAS,SAAS,EAC1B,QAAQ,UAAU,UAAU,EAC5B,QAAQ,IAAI,OAAO,UAAW,WAAW,SAAS,GAAG,QAAQ,WAAW,QAAQ;IACrF;IAEA,cAAa;AACX,aAAO,KAAK,aAAa,KAAK,cAAc,EAAC,OAAO,KAAK,MAAK,CAAC;IACjE;IAEA,cAAc,MAAM,QAAQ,UAAS;AACnC,WAAK,MAAM,MAAM,QAAQ,QAAQ;AACjC,WAAK,aAAa,cAAc;IAClC;IAEA,YAAW;AACT,WAAK,QAAQ,SAAS;AACtB,WAAK,cAAc,MAAM,WAAW,KAAK;IAC3C;IAEA,WAAU;AAAE,aAAO,KAAK,eAAe,cAAc,QAAQ,KAAK,eAAe,cAAc;IAAW;IAE1G,OAAM;AACJ,WAAK,KAAK,OAAO,oBAAoB,MAAM,MAAM,KAAK,UAAU,GAAG,CAAA,SAAQ;AACzE,YAAG,MAAK;AACN,cAAI,EAAC,QAAQ,OAAO,SAAQ,IAAI;AAChC,eAAK,QAAQ;QACf,OAAO;AACL,mBAAS;QACX;AAEA,gBAAO,QAAO;UACZ,KAAK;AACH,qBAAS,QAAQ,CAAA,QAAO;AAmBtB,yBAAW,MAAM,KAAK,UAAU,EAAC,MAAM,IAAG,CAAC,GAAG,CAAC;YACjD,CAAC;AACD,iBAAK,KAAK;AACV;UACF,KAAK;AACH,iBAAK,KAAK;AACV;UACF,KAAK;AACH,iBAAK,aAAa,cAAc;AAChC,iBAAK,OAAO,CAAC,CAAC;AACd,iBAAK,KAAK;AACV;UACF,KAAK;AACH,iBAAK,QAAQ,GAAG;AAChB,iBAAK,MAAM,MAAM,aAAa,KAAK;AACnC;UACF,KAAK;UACL,KAAK;AACH,iBAAK,QAAQ,GAAG;AAChB,iBAAK,cAAc,MAAM,yBAAyB,GAAG;AACrD;UACF;AAAS,kBAAM,IAAI,MAAM,yBAAyB,QAAQ;QAC5D;MACF,CAAC;IACH;;;;IAMA,KAAK,MAAK;AACR,UAAG,OAAO,SAAU,UAAS;AAAE,eAAO,oBAAoB,IAAI;MAAE;AAChE,UAAG,KAAK,cAAa;AACnB,aAAK,aAAa,KAAK,IAAI;MAC7B,WAAU,KAAK,kBAAiB;AAC9B,aAAK,YAAY,KAAK,IAAI;MAC5B,OAAO;AACL,aAAK,eAAe,CAAC,IAAI;AACzB,aAAK,oBAAoB,WAAW,MAAM;AACxC,eAAK,UAAU,KAAK,YAAY;AAChC,eAAK,eAAe;QACtB,GAAG,CAAC;MACN;IACF;IAEA,UAAU,UAAS;AACjB,WAAK,mBAAmB;AACxB,WAAK,KAAK,QAAQ,wBAAwB,SAAS,KAAK,IAAI,GAAG,MAAM,KAAK,QAAQ,SAAS,GAAG,CAAA,SAAQ;AACpG,aAAK,mBAAmB;AACxB,YAAG,CAAC,QAAQ,KAAK,WAAW,KAAI;AAC9B,eAAK,QAAQ,QAAQ,KAAK,MAAM;AAChC,eAAK,cAAc,MAAM,yBAAyB,KAAK;QACzD,WAAU,KAAK,YAAY,SAAS,GAAE;AACpC,eAAK,UAAU,KAAK,WAAW;AAC/B,eAAK,cAAc,CAAC;QACtB;MACF,CAAC;IACH;IAEA,MAAM,MAAM,QAAQ,UAAS;AAC3B,eAAQ,OAAO,KAAK,MAAK;AAAE,YAAI,MAAM;MAAE;AACvC,WAAK,aAAa,cAAc;AAChC,UAAI,OAAO,OAAO,OAAO,EAAC,MAAM,KAAM,QAAQ,QAAW,UAAU,KAAI,GAAG,EAAC,MAAM,QAAQ,SAAQ,CAAC;AAClG,WAAK,cAAc,CAAC;AACpB,mBAAa,KAAK,iBAAiB;AACnC,WAAK,oBAAoB;AACzB,UAAG,OAAO,eAAgB,aAAY;AACpC,aAAK,QAAQ,IAAI,WAAW,SAAS,IAAI,CAAC;MAC5C,OAAO;AACL,aAAK,QAAQ,IAAI;MACnB;IACF;IAEA,KAAK,QAAQ,aAAa,MAAM,iBAAiB,UAAS;AACxD,UAAI;AACJ,UAAI,YAAY,MAAM;AACpB,aAAK,KAAK,OAAO,GAAG;AACpB,wBAAgB;MAClB;AACA,YAAM,KAAK,QAAQ,QAAQ,KAAK,YAAY,GAAG,aAAa,MAAM,KAAK,SAAS,WAAW,CAAA,SAAQ;AACjG,aAAK,KAAK,OAAO,GAAG;AACpB,YAAG,KAAK,SAAS,GAAE;AAAE,mBAAS,IAAI;QAAE;MACtC,CAAC;AACD,WAAK,KAAK,IAAI,GAAG;IACnB;EACF;AEzKA,MAAO,qBAAQ;IACb,eAAe;IACf,aAAa;IACb,OAAO,EAAC,MAAM,GAAG,OAAO,GAAG,WAAW,EAAC;IAEvC,OAAO,KAAK,UAAS;AACnB,UAAG,IAAI,QAAQ,gBAAgB,aAAY;AACzC,eAAO,SAAS,KAAK,aAAa,GAAG,CAAC;MACxC,OAAO;AACL,YAAI,UAAU,CAAC,IAAI,UAAU,IAAI,KAAK,IAAI,OAAO,IAAI,OAAO,IAAI,OAAO;AACvE,eAAO,SAAS,KAAK,UAAU,OAAO,CAAC;MACzC;IACF;IAEA,OAAO,YAAY,UAAS;AAC1B,UAAG,WAAW,gBAAgB,aAAY;AACxC,eAAO,SAAS,KAAK,aAAa,UAAU,CAAC;MAC/C,OAAO;AACL,YAAI,CAAC,UAAU,KAAK,OAAO,OAAO,OAAO,IAAI,KAAK,MAAM,UAAU;AAClE,eAAO,SAAS,EAAC,UAAU,KAAK,OAAO,OAAO,QAAO,CAAC;MACxD;IACF;;IAIA,aAAa,SAAQ;AACnB,UAAI,EAAC,UAAU,KAAK,OAAO,OAAO,QAAO,IAAI;AAC7C,UAAI,aAAa,KAAK,cAAc,SAAS,SAAS,IAAI,SAAS,MAAM,SAAS,MAAM;AACxF,UAAI,SAAS,IAAI,YAAY,KAAK,gBAAgB,UAAU;AAC5D,UAAI,OAAO,IAAI,SAAS,MAAM;AAC9B,UAAI,SAAS;AAEb,WAAK,SAAS,UAAU,KAAK,MAAM,IAAI;AACvC,WAAK,SAAS,UAAU,SAAS,MAAM;AACvC,WAAK,SAAS,UAAU,IAAI,MAAM;AAClC,WAAK,SAAS,UAAU,MAAM,MAAM;AACpC,WAAK,SAAS,UAAU,MAAM,MAAM;AACpC,YAAM,KAAK,UAAU,CAAA,SAAQ,KAAK,SAAS,UAAU,KAAK,WAAW,CAAC,CAAC,CAAC;AACxE,YAAM,KAAK,KAAK,CAAA,SAAQ,KAAK,SAAS,UAAU,KAAK,WAAW,CAAC,CAAC,CAAC;AACnE,YAAM,KAAK,OAAO,CAAA,SAAQ,KAAK,SAAS,UAAU,KAAK,WAAW,CAAC,CAAC,CAAC;AACrE,YAAM,KAAK,OAAO,CAAA,SAAQ,KAAK,SAAS,UAAU,KAAK,WAAW,CAAC,CAAC,CAAC;AAErE,UAAI,WAAW,IAAI,WAAW,OAAO,aAAa,QAAQ,UAAU;AACpE,eAAS,IAAI,IAAI,WAAW,MAAM,GAAG,CAAC;AACtC,eAAS,IAAI,IAAI,WAAW,OAAO,GAAG,OAAO,UAAU;AAEvD,aAAO,SAAS;IAClB;IAEA,aAAa,QAAO;AAClB,UAAI,OAAO,IAAI,SAAS,MAAM;AAC9B,UAAI,OAAO,KAAK,SAAS,CAAC;AAC1B,UAAI,UAAU,IAAI,YAAY;AAC9B,cAAO,MAAK;QACV,KAAK,KAAK,MAAM;AAAM,iBAAO,KAAK,WAAW,QAAQ,MAAM,OAAO;QAClE,KAAK,KAAK,MAAM;AAAO,iBAAO,KAAK,YAAY,QAAQ,MAAM,OAAO;QACpE,KAAK,KAAK,MAAM;AAAW,iBAAO,KAAK,gBAAgB,QAAQ,MAAM,OAAO;MAC9E;IACF;IAEA,WAAW,QAAQ,MAAM,SAAQ;AAC/B,UAAI,cAAc,KAAK,SAAS,CAAC;AACjC,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,SAAS,KAAK,gBAAgB,KAAK,cAAc;AACrD,UAAI,UAAU,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,WAAW,CAAC;AACvE,eAAS,SAAS;AAClB,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,OAAO,OAAO,MAAM,QAAQ,OAAO,UAAU;AACjD,aAAO,EAAC,UAAU,SAAS,KAAK,MAAM,OAAc,OAAc,SAAS,KAAI;IACjF;IAEA,YAAY,QAAQ,MAAM,SAAQ;AAChC,UAAI,cAAc,KAAK,SAAS,CAAC;AACjC,UAAI,UAAU,KAAK,SAAS,CAAC;AAC7B,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,SAAS,KAAK,gBAAgB,KAAK;AACvC,UAAI,UAAU,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,WAAW,CAAC;AACvE,eAAS,SAAS;AAClB,UAAI,MAAM,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,OAAO,CAAC;AAC/D,eAAS,SAAS;AAClB,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,OAAO,OAAO,MAAM,QAAQ,OAAO,UAAU;AACjD,UAAI,UAAU,EAAC,QAAQ,OAAO,UAAU,KAAI;AAC5C,aAAO,EAAC,UAAU,SAAS,KAAU,OAAc,OAAO,eAAe,OAAO,QAAgB;IAClG;IAEA,gBAAgB,QAAQ,MAAM,SAAQ;AACpC,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,YAAY,KAAK,SAAS,CAAC;AAC/B,UAAI,SAAS,KAAK,gBAAgB;AAClC,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,QAAQ,QAAQ,OAAO,OAAO,MAAM,QAAQ,SAAS,SAAS,CAAC;AACnE,eAAS,SAAS;AAClB,UAAI,OAAO,OAAO,MAAM,QAAQ,OAAO,UAAU;AAEjD,aAAO,EAAC,UAAU,MAAM,KAAK,MAAM,OAAc,OAAc,SAAS,KAAI;IAC9E;EACF;ACFA,MAAqB,SAArB,MAA4B;IAC1B,YAAY,UAAU,OAAO,CAAC,GAAE;AAC9B,WAAK,uBAAuB,EAAC,MAAM,CAAC,GAAG,OAAO,CAAC,GAAG,OAAO,CAAC,GAAG,SAAS,CAAC,EAAC;AACxE,WAAK,WAAW,CAAC;AACjB,WAAK,aAAa,CAAC;AACnB,WAAK,MAAM;AACX,WAAK,UAAU,KAAK,WAAW;AAC/B,WAAK,YAAY,KAAK,aAAa,OAAO,aAAa;AACvD,WAAK,2BAA2B;AAChC,WAAK,qBAAqB,KAAK;AAC/B,WAAK,gBAAgB;AACrB,WAAK,eAAe,KAAK,kBAAmB,UAAU,OAAO;AAC7D,WAAK,yBAAyB;AAC9B,WAAK,iBAAiB,mBAAW,OAAO,KAAK,kBAAU;AACvD,WAAK,iBAAiB,mBAAW,OAAO,KAAK,kBAAU;AACvD,WAAK,gBAAgB;AACrB,WAAK,gBAAgB;AACrB,WAAK,aAAa,KAAK,cAAc;AACrC,WAAK,eAAe;AACpB,UAAG,KAAK,cAAc,UAAS;AAC7B,aAAK,SAAS,KAAK,UAAU,KAAK;AAClC,aAAK,SAAS,KAAK,UAAU,KAAK;MACpC,OAAO;AACL,aAAK,SAAS,KAAK;AACnB,aAAK,SAAS,KAAK;MACrB;AACA,UAAI,+BAA+B;AACnC,UAAG,aAAa,UAAU,kBAAiB;AACzC,kBAAU,iBAAiB,YAAY,CAAA,OAAM;AAC3C,cAAG,KAAK,MAAK;AACX,iBAAK,WAAW;AAChB,2CAA+B,KAAK;UACtC;QACF,CAAC;AACD,kBAAU,iBAAiB,YAAY,CAAA,OAAM;AAC3C,cAAG,iCAAiC,KAAK,cAAa;AACpD,2CAA+B;AAC/B,iBAAK,QAAQ;UACf;QACF,CAAC;MACH;AACA,WAAK,sBAAsB,KAAK,uBAAuB;AACvD,WAAK,gBAAgB,CAAC,UAAU;AAC9B,YAAG,KAAK,eAAc;AACpB,iBAAO,KAAK,cAAc,KAAK;QACjC,OAAO;AACL,iBAAO,CAAC,KAAM,KAAM,GAAI,EAAE,QAAQ,CAAC,KAAK;QAC1C;MACF;AACA,WAAK,mBAAmB,CAAC,UAAU;AACjC,YAAG,KAAK,kBAAiB;AACvB,iBAAO,KAAK,iBAAiB,KAAK;QACpC,OAAO;AACL,iBAAO,CAAC,IAAI,IAAI,KAAK,KAAK,KAAK,KAAK,KAAK,KAAM,GAAI,EAAE,QAAQ,CAAC,KAAK;QACrE;MACF;AACA,WAAK,SAAS,KAAK,UAAU;AAC7B,UAAG,CAAC,KAAK,UAAU,KAAK,OAAM;AAC5B,aAAK,SAAS,CAAC,MAAM,KAAK,SAAS;AAAE,kBAAQ,IAAI,GAAG,SAAS,OAAO,IAAI;QAAE;MAC5E;AACA,WAAK,oBAAoB,KAAK,qBAAqB;AACnD,WAAK,SAAS,QAAQ,KAAK,UAAU,CAAC,CAAC;AACvC,WAAK,WAAW,GAAG,YAAY,WAAW;AAC1C,WAAK,MAAM,KAAK,OAAO;AACvB,WAAK,wBAAwB;AAC7B,WAAK,iBAAiB;AACtB,WAAK,sBAAsB;AAC3B,WAAK,iBAAiB,IAAI,MAAM,MAAM;AACpC,aAAK,SAAS,MAAM,KAAK,QAAQ,CAAC;MACpC,GAAG,KAAK,gBAAgB;IAC1B;;;;IAKA,uBAAsB;AAAE,aAAO;IAAS;;;;;;;IAQxC,iBAAiB,cAAa;AAC5B,WAAK;AACL,WAAK,gBAAgB;AACrB,mBAAa,KAAK,aAAa;AAC/B,WAAK,eAAe,MAAM;AAC1B,UAAG,KAAK,MAAK;AACX,aAAK,KAAK,MAAM;AAChB,aAAK,OAAO;MACd;AACA,WAAK,YAAY;IACnB;;;;;;IAOA,WAAU;AAAE,aAAO,SAAS,SAAS,MAAM,QAAQ,IAAI,QAAQ;IAAK;;;;;;IAOpE,cAAa;AACX,UAAI,MAAM,KAAK;QACb,KAAK,aAAa,KAAK,UAAU,KAAK,OAAO,CAAC;QAAG,EAAC,KAAK,KAAK,IAAG;MAAC;AAClE,UAAG,IAAI,OAAO,CAAC,MAAM,KAAI;AAAE,eAAO;MAAI;AACtC,UAAG,IAAI,OAAO,CAAC,MAAM,KAAI;AAAE,eAAO,GAAG,KAAK,SAAS,KAAK;MAAM;AAE9D,aAAO,GAAG,KAAK,SAAS,OAAO,SAAS,OAAO;IACjD;;;;;;;;;;IAWA,WAAW,UAAU,MAAM,QAAO;AAChC,WAAK;AACL,WAAK,gBAAgB;AACrB,WAAK,gBAAgB;AACrB,mBAAa,KAAK,aAAa;AAC/B,WAAK,eAAe,MAAM;AAC1B,WAAK,SAAS,MAAM;AAClB,aAAK,gBAAgB;AACrB,oBAAY,SAAS;MACvB,GAAG,MAAM,MAAM;IACjB;;;;;;;;IASA,QAAQ,QAAO;AACb,UAAG,QAAO;AACR,mBAAW,QAAQ,IAAI,yFAAyF;AAChH,aAAK,SAAS,QAAQ,MAAM;MAC9B;AACA,UAAG,KAAK,QAAQ,CAAC,KAAK,eAAc;AAAE;MAAO;AAC7C,UAAG,KAAK,sBAAsB,KAAK,cAAc,UAAS;AACxD,aAAK,oBAAoB,UAAU,KAAK,kBAAkB;MAC5D,OAAO;AACL,aAAK,iBAAiB;MACxB;IACF;;;;;;;IAQA,IAAI,MAAM,KAAK,MAAK;AAAE,WAAK,UAAU,KAAK,OAAO,MAAM,KAAK,IAAI;IAAE;;;;IAKlE,YAAW;AAAE,aAAO,KAAK,WAAW;IAAK;;;;;;;;IASzC,OAAO,UAAS;AACd,UAAI,MAAM,KAAK,QAAQ;AACvB,WAAK,qBAAqB,KAAK,KAAK,CAAC,KAAK,QAAQ,CAAC;AACnD,aAAO;IACT;;;;;IAMA,QAAQ,UAAS;AACf,UAAI,MAAM,KAAK,QAAQ;AACvB,WAAK,qBAAqB,MAAM,KAAK,CAAC,KAAK,QAAQ,CAAC;AACpD,aAAO;IACT;;;;;;;;IASA,QAAQ,UAAS;AACf,UAAI,MAAM,KAAK,QAAQ;AACvB,WAAK,qBAAqB,MAAM,KAAK,CAAC,KAAK,QAAQ,CAAC;AACpD,aAAO;IACT;;;;;IAMA,UAAU,UAAS;AACjB,UAAI,MAAM,KAAK,QAAQ;AACvB,WAAK,qBAAqB,QAAQ,KAAK,CAAC,KAAK,QAAQ,CAAC;AACtD,aAAO;IACT;;;;;;;IAQA,KAAK,UAAS;AACZ,UAAG,CAAC,KAAK,YAAY,GAAE;AAAE,eAAO;MAAM;AACtC,UAAI,MAAM,KAAK,QAAQ;AACvB,UAAI,YAAY,KAAK,IAAI;AACzB,WAAK,KAAK,EAAC,OAAO,WAAW,OAAO,aAAa,SAAS,CAAC,GAAG,IAAQ,CAAC;AACvE,UAAI,WAAW,KAAK,UAAU,CAAA,QAAO;AACnC,YAAG,IAAI,QAAQ,KAAI;AACjB,eAAK,IAAI,CAAC,QAAQ,CAAC;AACnB,mBAAS,KAAK,IAAI,IAAI,SAAS;QACjC;MACF,CAAC;AACD,aAAO;IACT;;;;IAMA,mBAAkB;AAChB,WAAK;AACL,WAAK,gBAAgB;AACrB,WAAK,OAAO,IAAI,KAAK,UAAU,KAAK,YAAY,CAAC;AACjD,WAAK,KAAK,aAAa,KAAK;AAC5B,WAAK,KAAK,UAAU,KAAK;AACzB,WAAK,KAAK,SAAS,MAAM,KAAK,WAAW;AACzC,WAAK,KAAK,UAAU,CAAA,UAAS,KAAK,YAAY,KAAK;AACnD,WAAK,KAAK,YAAY,CAAA,UAAS,KAAK,cAAc,KAAK;AACvD,WAAK,KAAK,UAAU,CAAA,UAAS,KAAK,YAAY,KAAK;IACrD;IAEA,WAAW,KAAI;AAAE,aAAO,KAAK,gBAAgB,KAAK,aAAa,QAAQ,GAAG;IAAE;IAE5E,aAAa,KAAK,KAAI;AAAE,WAAK,gBAAgB,KAAK,aAAa,QAAQ,KAAK,GAAG;IAAE;IAEjF,oBAAoB,mBAAmB,oBAAoB,MAAK;AAC9D,mBAAa,KAAK,aAAa;AAC/B,UAAI,cAAc;AAClB,UAAI,mBAAmB;AACvB,UAAI,SAAS;AACb,UAAI,WAAW,CAAC,WAAW;AACzB,aAAK,IAAI,aAAa,mBAAmB,kBAAkB,WAAW,MAAM;AAC5E,aAAK,IAAI,CAAC,SAAS,QAAQ,CAAC;AAC5B,2BAAmB;AACnB,aAAK,iBAAiB,iBAAiB;AACvC,aAAK,iBAAiB;MACxB;AACA,UAAG,KAAK,WAAW,gBAAgB,kBAAkB,MAAM,GAAE;AAAE,eAAO,SAAS,WAAW;MAAE;AAE5F,WAAK,gBAAgB,WAAW,UAAU,iBAAiB;AAE3D,iBAAW,KAAK,QAAQ,CAAA,WAAU;AAChC,aAAK,IAAI,aAAa,SAAS,MAAM;AACrC,YAAG,oBAAoB,CAAC,aAAY;AAClC,uBAAa,KAAK,aAAa;AAC/B,mBAAS,MAAM;QACjB;MACF,CAAC;AACD,WAAK,OAAO,MAAM;AAChB,sBAAc;AACd,YAAG,CAAC,kBAAiB;AAEnB,cAAG,CAAC,KAAK,0BAAyB;AAAE,iBAAK,aAAa,gBAAgB,kBAAkB,QAAQ,MAAM;UAAE;AACxG,iBAAO,KAAK,IAAI,aAAa,eAAe,kBAAkB,eAAe;QAC/E;AAEA,qBAAa,KAAK,aAAa;AAC/B,aAAK,gBAAgB,WAAW,UAAU,iBAAiB;AAC3D,aAAK,KAAK,CAAA,QAAO;AACf,eAAK,IAAI,aAAa,8BAA8B,GAAG;AACvD,eAAK,2BAA2B;AAChC,uBAAa,KAAK,aAAa;QACjC,CAAC;MACH,CAAC;AACD,WAAK,iBAAiB;IACxB;IAEA,kBAAiB;AACf,mBAAa,KAAK,cAAc;AAChC,mBAAa,KAAK,qBAAqB;IACzC;IAEA,aAAY;AACV,UAAG,KAAK,UAAU;AAAG,aAAK,IAAI,aAAa,GAAG,KAAK,UAAU,qBAAqB,KAAK,YAAY,GAAG;AACtG,WAAK,gBAAgB;AACrB,WAAK,gBAAgB;AACrB,WAAK;AACL,WAAK,gBAAgB;AACrB,WAAK,eAAe,MAAM;AAC1B,WAAK,eAAe;AACpB,WAAK,qBAAqB,KAAK,QAAQ,CAAC,CAAC,EAAE,QAAQ,MAAM,SAAS,CAAC;IACrE;;;;IAMA,mBAAkB;AAChB,UAAG,KAAK,qBAAoB;AAC1B,aAAK,sBAAsB;AAC3B,YAAG,KAAK,UAAU,GAAE;AAAE,eAAK,IAAI,aAAa,0DAA0D;QAAE;AACxG,aAAK,iBAAiB;AACtB,aAAK,gBAAgB;AACrB,aAAK,SAAS,MAAM,KAAK,eAAe,gBAAgB,GAAG,iBAAiB,mBAAmB;MACjG;IACF;IAEA,iBAAgB;AACd,UAAG,KAAK,QAAQ,KAAK,KAAK,eAAc;AAAE;MAAO;AACjD,WAAK,sBAAsB;AAC3B,WAAK,gBAAgB;AACrB,WAAK,iBAAiB,WAAW,MAAM,KAAK,cAAc,GAAG,KAAK,mBAAmB;IACvF;IAEA,SAAS,UAAU,MAAM,QAAO;AAC9B,UAAG,CAAC,KAAK,MAAK;AACZ,eAAO,YAAY,SAAS;MAC9B;AACA,UAAI,eAAe,KAAK;AAExB,WAAK,kBAAkB,MAAM;AAC3B,YAAG,iBAAiB,KAAK,cAAa;AAAE;QAAO;AAC/C,YAAG,KAAK,MAAK;AACX,cAAG,MAAK;AAAE,iBAAK,KAAK,MAAM,MAAM,UAAU,EAAE;UAAE,OAAO;AAAE,iBAAK,KAAK,MAAM;UAAE;QAC3E;AAEA,aAAK,oBAAoB,MAAM;AAC7B,cAAG,iBAAiB,KAAK,cAAa;AAAE;UAAO;AAC/C,cAAG,KAAK,MAAK;AACX,iBAAK,KAAK,SAAS,WAAW;YAAE;AAChC,iBAAK,KAAK,UAAU,WAAW;YAAE;AACjC,iBAAK,KAAK,YAAY,WAAW;YAAE;AACnC,iBAAK,KAAK,UAAU,WAAW;YAAE;AACjC,iBAAK,OAAO;UACd;AAEA,sBAAY,SAAS;QACvB,CAAC;MACH,CAAC;IACH;IAEA,kBAAkB,UAAU,QAAQ,GAAE;AACpC,UAAG,UAAU,KAAK,CAAC,KAAK,QAAQ,CAAC,KAAK,KAAK,gBAAe;AACxD,iBAAS;AACT;MACF;AAEA,iBAAW,MAAM;AACf,aAAK,kBAAkB,UAAU,QAAQ,CAAC;MAC5C,GAAG,MAAM,KAAK;IAChB;IAEA,oBAAoB,UAAU,QAAQ,GAAE;AACtC,UAAG,UAAU,KAAK,CAAC,KAAK,QAAQ,KAAK,KAAK,eAAe,cAAc,QAAO;AAC5E,iBAAS;AACT;MACF;AAEA,iBAAW,MAAM;AACf,aAAK,oBAAoB,UAAU,QAAQ,CAAC;MAC9C,GAAG,MAAM,KAAK;IAChB;IAEA,YAAY,OAAM;AAChB,UAAI,YAAY,SAAS,MAAM;AAC/B,UAAG,KAAK,UAAU;AAAG,aAAK,IAAI,aAAa,SAAS,KAAK;AACzD,WAAK,iBAAiB;AACtB,WAAK,gBAAgB;AACrB,UAAG,CAAC,KAAK,iBAAiB,cAAc,KAAK;AAC3C,aAAK,eAAe,gBAAgB;MACtC;AACA,WAAK,qBAAqB,MAAM,QAAQ,CAAC,CAAC,EAAE,QAAQ,MAAM,SAAS,KAAK,CAAC;IAC3E;;;;IAKA,YAAY,OAAM;AAChB,UAAG,KAAK,UAAU;AAAG,aAAK,IAAI,aAAa,KAAK;AAChD,UAAI,kBAAkB,KAAK;AAC3B,UAAI,oBAAoB,KAAK;AAC7B,WAAK,qBAAqB,MAAM,QAAQ,CAAC,CAAC,EAAE,QAAQ,MAAM;AACxD,iBAAS,OAAO,iBAAiB,iBAAiB;MACpD,CAAC;AACD,UAAG,oBAAoB,KAAK,aAAa,oBAAoB,GAAE;AAC7D,aAAK,iBAAiB;MACxB;IACF;;;;IAKA,mBAAkB;AAChB,WAAK,SAAS,QAAQ,CAAA,YAAW;AAC/B,YAAG,EAAE,QAAQ,UAAU,KAAK,QAAQ,UAAU,KAAK,QAAQ,SAAS,IAAG;AACrE,kBAAQ,QAAQ,eAAe,KAAK;QACtC;MACF,CAAC;IACH;;;;IAKA,kBAAiB;AACf,cAAO,KAAK,QAAQ,KAAK,KAAK,YAAW;QACvC,KAAK,cAAc;AAAY,iBAAO;QACtC,KAAK,cAAc;AAAM,iBAAO;QAChC,KAAK,cAAc;AAAS,iBAAO;QACnC;AAAS,iBAAO;MAClB;IACF;;;;IAKA,cAAa;AAAE,aAAO,KAAK,gBAAgB,MAAM;IAAO;;;;;;IAOxD,OAAO,SAAQ;AACb,WAAK,IAAI,QAAQ,eAAe;AAChC,WAAK,WAAW,KAAK,SAAS,OAAO,CAAA,MAAK,MAAM,OAAO;IACzD;;;;;;;IAQA,IAAI,MAAK;AACP,eAAQ,OAAO,KAAK,sBAAqB;AACvC,aAAK,qBAAqB,GAAG,IAAI,KAAK,qBAAqB,GAAG,EAAE,OAAO,CAAC,CAAC,GAAG,MAAM;AAChF,iBAAO,KAAK,QAAQ,GAAG,MAAM;QAC/B,CAAC;MACH;IACF;;;;;;;;IASA,QAAQ,OAAO,aAAa,CAAC,GAAE;AAC7B,UAAI,OAAO,IAAI,QAAQ,OAAO,YAAY,IAAI;AAC9C,WAAK,SAAS,KAAK,IAAI;AACvB,aAAO;IACT;;;;IAKA,KAAK,MAAK;AACR,UAAG,KAAK,UAAU,GAAE;AAClB,YAAI,EAAC,OAAO,OAAO,SAAS,KAAK,SAAQ,IAAI;AAC7C,aAAK,IAAI,QAAQ,GAAG,SAAS,UAAU,aAAa,QAAQ,OAAO;MACrE;AAEA,UAAG,KAAK,YAAY,GAAE;AACpB,aAAK,OAAO,MAAM,CAAA,WAAU,KAAK,KAAK,KAAK,MAAM,CAAC;MACpD,OAAO;AACL,aAAK,WAAW,KAAK,MAAM,KAAK,OAAO,MAAM,CAAA,WAAU,KAAK,KAAK,KAAK,MAAM,CAAC,CAAC;MAChF;IACF;;;;;IAMA,UAAS;AACP,UAAI,SAAS,KAAK,MAAM;AACxB,UAAG,WAAW,KAAK,KAAI;AAAE,aAAK,MAAM;MAAE,OAAO;AAAE,aAAK,MAAM;MAAO;AAEjE,aAAO,KAAK,IAAI,SAAS;IAC3B;IAEA,gBAAe;AACb,UAAG,KAAK,uBAAuB,CAAC,KAAK,YAAY,GAAE;AAAE;MAAO;AAC5D,WAAK,sBAAsB,KAAK,QAAQ;AACxC,WAAK,KAAK,EAAC,OAAO,WAAW,OAAO,aAAa,SAAS,CAAC,GAAG,KAAK,KAAK,oBAAmB,CAAC;AAC5F,WAAK,wBAAwB,WAAW,MAAM,KAAK,iBAAiB,GAAG,KAAK,mBAAmB;IACjG;IAEA,kBAAiB;AACf,UAAG,KAAK,YAAY,KAAK,KAAK,WAAW,SAAS,GAAE;AAClD,aAAK,WAAW,QAAQ,CAAA,aAAY,SAAS,CAAC;AAC9C,aAAK,aAAa,CAAC;MACrB;IACF;IAEA,cAAc,YAAW;AACvB,WAAK,OAAO,WAAW,MAAM,CAAA,QAAO;AAClC,YAAI,EAAC,OAAO,OAAO,SAAS,KAAK,SAAQ,IAAI;AAC7C,YAAG,OAAO,QAAQ,KAAK,qBAAoB;AACzC,eAAK,gBAAgB;AACrB,eAAK,sBAAsB;AAC3B,eAAK,iBAAiB,WAAW,MAAM,KAAK,cAAc,GAAG,KAAK,mBAAmB;QACvF;AAEA,YAAG,KAAK,UAAU;AAAG,eAAK,IAAI,WAAW,GAAG,QAAQ,UAAU,MAAM,SAAS,SAAS,OAAO,MAAM,MAAM,OAAO,MAAM,OAAO;AAE7H,iBAAQ,IAAI,GAAG,IAAI,KAAK,SAAS,QAAQ,KAAI;AAC3C,gBAAM,UAAU,KAAK,SAAS,CAAC;AAC/B,cAAG,CAAC,QAAQ,SAAS,OAAO,OAAO,SAAS,QAAQ,GAAE;AAAE;UAAS;AACjE,kBAAQ,QAAQ,OAAO,SAAS,KAAK,QAAQ;QAC/C;AAEA,iBAAQ,IAAI,GAAG,IAAI,KAAK,qBAAqB,QAAQ,QAAQ,KAAI;AAC/D,cAAI,CAAC,EAAE,QAAQ,IAAI,KAAK,qBAAqB,QAAQ,CAAC;AACtD,mBAAS,GAAG;QACd;MACF,CAAC;IACH;IAEA,eAAe,OAAM;AACnB,UAAI,aAAa,KAAK,SAAS,KAAK,CAAA,MAAK,EAAE,UAAU,UAAU,EAAE,SAAS,KAAK,EAAE,UAAU,EAAE;AAC7F,UAAG,YAAW;AACZ,YAAG,KAAK,UAAU;AAAG,eAAK,IAAI,aAAa,4BAA4B,QAAQ;AAC/E,mBAAW,MAAM;MACnB;IACF;EACF;;;AChpBO,MAAM,sBAAsB;AAC5B,MAAM,cAAc;AACpB,MAAM,oBAAoB;AAC1B,MAAM,oBAAoB;AAC1B,MAAM,kBAAkB;AACxB,MAAM,oBAAoB;IAC/B;IAAqB;IAAsB;IAC3C;IAAuB;IAAqB;IAAoB;IAChE;EACF;AACO,MAAM,gBAAgB;AACtB,MAAM,gBAAgB;AACtB,MAAM,mBAAmB;AACzB,MAAM,iBAAiB;AACvB,MAAM,kBAAkB;AACxB,MAAM,cAAc;AACpB,MAAM,eAAe;AACrB,MAAM,mBAAmB;AACzB,MAAM,oBAAoB;AAC1B,MAAM,iBAAiB;AACvB,MAAM,uBAAuB;AAC7B,MAAM,gBAAgB;AACtB,MAAM,kBAAkB;AACxB,MAAM,wBAAwB;AAC9B,MAAM,wBAAwB;AAC9B,MAAM,WAAW;AACjB,MAAM,eAAe;AACrB,MAAM,YAAY;AAClB,MAAM,sBAAsB;AAC5B,MAAM,oBAAoB;AAC1B,MAAM,kBAAkB;AACxB,MAAM,yBAAyB;AAC/B,MAAM,yBAAyB;AAC/B,MAAM,gBAAgB;AACtB,MAAM,WAAW;AACjB,MAAM,cAAc;AACpB,MAAM,mBAAmB;AACzB,MAAM,sBAAsB;AAC5B,MAAM,qBAAqB;AAC3B,MAAM,kBAAkB;AACxB,MAAM,mBAAmB,CAAC,QAAQ,YAAY,UAAU,SAAS,YAAY,UAAU,OAAO,OAAO,QAAQ,QAAQ,kBAAkB,SAAS,OAAO;AACvJ,MAAM,mBAAmB,CAAC,YAAY,OAAO;AAC7C,MAAM,oBAAoB;AAC1B,MAAM,cAAc;AACpB,MAAM,oBAAoB,IAAI;AAC9B,MAAM,aAAa;AACnB,MAAM,aAAa;AACnB,MAAM,eAAe;AACrB,MAAM,eAAe;AACrB,MAAM,mBAAmB;AACzB,MAAM,2BAA2B;AACjC,MAAM,WAAW;AACjB,MAAM,eAAe;AACrB,MAAM,eAAe;AACrB,MAAM,aAAa;AACnB,MAAM,aAAa;AACnB,MAAM,iBAAiB;AACvB,MAAM,UAAU;AAChB,MAAM,cAAc;AACpB,MAAM,mBAAmB;AACzB,MAAM,eAAe;AACrB,MAAM,iBAAiB;AACvB,MAAM,qBAAqB;AAC3B,MAAM,0BAA0B;AAChC,MAAM,eAAe;AACrB,MAAM,cAAc;AACpB,MAAM,oBAAoB;AAC1B,MAAM,iBAAiB;AACvB,MAAM,0BAA0B;AAChC,MAAM,+BAA+B;AACrC,MAAM,uBAAuB;AAC7B,MAAM,iBAAiB;AACvB,MAAM,eAAe;AAGrB,MAAM,mBAAmB;AACzB,MAAM,YAAY;AAClB,MAAM,oBAAoB;AAC1B,MAAM,WAAW;IACtB,UAAU;IACV,UAAU;EACZ;AACO,MAAM,oBAAoB,CAAC,iBAAiB,aAAa,YAAY;AAErE,MAAM,WAAW;AACjB,MAAM,SAAS;AACf,MAAM,OAAO;AACb,MAAM,aAAa;AACnB,MAAM,SAAS;AACf,MAAM,QAAQ;AACd,MAAM,QAAQ;AACd,MAAM,YAAY;AAClB,MAAM,SAAS;ACxFtB,MAAqB,gBAArB,MAAmC;IACjC,YAAY,OAAO,QAAQC,aAAW;AACpC,UAAI,EAAC,YAAY,cAAa,IAAI;AAClC,WAAK,aAAaA;AAClB,WAAK,QAAQ;AACb,WAAK,SAAS;AACd,WAAK,YAAY;AACjB,WAAK,eAAe;AACpB,WAAK,aAAa;AAClB,WAAK,UAAU;AACf,WAAK,gBAAgBA,YAAW,QAAQ,OAAO,MAAM,OAAO,EAAC,OAAO,MAAM,SAAS,EAAC,CAAC;IACvF;IAEA,MAAM,QAAO;AACX,UAAG,KAAK,SAAQ;AAAE;MAAO;AACzB,WAAK,cAAc,MAAM;AACzB,WAAK,UAAU;AACf,mBAAa,KAAK,UAAU;AAC5B,WAAK,MAAM,MAAM,MAAM;IACzB;IAEA,SAAQ;AACN,WAAK,cAAc,QAAQ,CAAA,WAAU,KAAK,MAAM,MAAM,CAAC;AACvD,WAAK,cAAc,KAAK,EACrB,QAAQ,MAAM,CAAA,UAAS,KAAK,cAAc,CAAC,EAC3C,QAAQ,SAAS,CAAA,WAAU,KAAK,MAAM,MAAM,CAAC;IAClD;IAEA,SAAQ;AAAE,aAAO,KAAK,UAAU,KAAK,MAAM,KAAK;IAAK;IAErD,gBAAe;AACb,UAAI,SAAS,IAAI,OAAO,WAAW;AACnC,UAAI,OAAO,KAAK,MAAM,KAAK,MAAM,KAAK,QAAQ,KAAK,YAAY,KAAK,MAAM;AAC1E,aAAO,SAAS,CAAC,MAAM;AACrB,YAAG,EAAE,OAAO,UAAU,MAAK;AACzB,eAAK,UAAU,EAAE,OAAO,OAAO;AAC/B,eAAK,UAAU,EAAE,OAAO,MAAM;QAChC,OAAO;AACL,iBAAO,SAAS,iBAAiB,EAAE,OAAO,KAAK;QACjD;MACF;AACA,aAAO,kBAAkB,IAAI;IAC/B;IAEA,UAAU,OAAM;AACd,UAAG,CAAC,KAAK,cAAc,SAAS,GAAE;AAAE;MAAO;AAC3C,WAAK,cAAc,KAAK,SAAS,OAAO,KAAK,YAAY,EACtD,QAAQ,MAAM,MAAM;AACnB,aAAK,MAAM,SAAU,KAAK,SAAS,KAAK,MAAM,KAAK,OAAQ,GAAG;AAC9D,YAAG,CAAC,KAAK,OAAO,GAAE;AAChB,eAAK,aAAa,WAAW,MAAM,KAAK,cAAc,GAAG,KAAK,WAAW,cAAc,KAAK,CAAC;QAC/F;MACF,CAAC,EACA,QAAQ,SAAS,CAAC,EAAC,OAAM,MAAM,KAAK,MAAM,MAAM,CAAC;IACtD;EACF;ACrDO,MAAI,WAAW,CAAC,KAAK,QAAQ,QAAQ,SAAS,QAAQ,MAAM,KAAK,GAAG;AAEpE,MAAI,QAAQ,CAAC,QAAQ;AAC1B,QAAI,OAAO,OAAO;AAClB,WAAO,SAAS,YAAa,SAAS,YAAY,iBAAiB,KAAK,GAAG;EAC7E;AAEO,WAAS,qBAAoB;AAClC,QAAI,MAAM,oBAAI,IAAI;AAClB,QAAI,QAAQ,SAAS,iBAAiB,OAAO;AAC7C,aAAQ,IAAI,GAAG,MAAM,MAAM,QAAQ,IAAI,KAAK,KAAI;AAC9C,UAAG,IAAI,IAAI,MAAM,CAAC,EAAE,EAAE,GAAE;AACtB,gBAAQ,MAAM,0BAA0B,MAAM,CAAC,EAAE,gCAAgC;MACnF,OAAO;AACL,YAAI,IAAI,MAAM,CAAC,EAAE,EAAE;MACrB;IACF;EACF;AAEO,WAAS,2BAA2B,SAAQ;AACjD,UAAM,SAAS,oBAAI,IAAI;AACvB,WAAO,KAAK,OAAO,EAAE,QAAQ,CAAC,OAAO;AACnC,YAAM,WAAW,SAAS,eAAe,EAAE;AAC3C,UAAG,YAAY,SAAS,iBAAiB,SAAS,cAAc,aAAa,YAAY,MAAM,UAAS;AACtG,eAAO,IAAI,iCAAiC,SAAS,cAAc,kGAAkG;MACvK;IACF,CAAC;AACD,WAAO,QAAQ,CAAA,UAAS,QAAQ,MAAM,KAAK,CAAC;EAC9C;AAEO,MAAI,QAAQ,CAAC,MAAM,MAAM,KAAK,QAAQ;AAC3C,QAAG,KAAK,WAAW,eAAe,GAAE;AAClC,cAAQ,IAAI,GAAG,KAAK,MAAM,SAAS,UAAU,GAAG;IAClD;EACF;AAGO,MAAIC,WAAU,CAAC,QAAQ,OAAO,QAAQ,aAAa,MAAM,WAAW;AAAE,WAAO;EAAI;AAEjF,MAAI,QAAQ,CAAC,QAAQ;AAAE,WAAO,KAAK,MAAM,KAAK,UAAU,GAAG,CAAC;EAAE;AAE9D,MAAI,oBAAoB,CAAC,IAAI,SAAS,aAAa;AACxD,OAAG;AACD,UAAG,GAAG,QAAQ,IAAI,UAAU,KAAK,CAAC,GAAG,UAAS;AAAE,eAAO;MAAG;AAC1D,WAAK,GAAG,iBAAiB,GAAG;IAC9B,SAAQ,OAAO,QAAQ,GAAG,aAAa,KAAK,EAAG,YAAY,SAAS,WAAW,EAAE,KAAM,GAAG,QAAQ,iBAAiB;AACnH,WAAO;EACT;AAEO,MAAI,WAAW,CAAC,QAAQ;AAC7B,WAAO,QAAQ,QAAQ,OAAO,QAAQ,YAAY,EAAE,eAAe;EACrE;AAEO,MAAI,aAAa,CAAC,MAAM,SAAS,KAAK,UAAU,IAAI,MAAM,KAAK,UAAU,IAAI;AAE7E,MAAI,UAAU,CAAC,QAAQ;AAC5B,aAAQ,KAAK,KAAI;AAAE,aAAO;IAAM;AAChC,WAAO;EACT;AAEO,MAAI,QAAQ,CAAC,IAAI,aAAa,MAAM,SAAS,EAAE;AAE/C,MAAI,kBAAkB,SAAU,SAAS,SAAS,MAAMD,aAAW;AACxE,YAAQ,QAAQ,CAAA,UAAS;AACvB,UAAI,gBAAgB,IAAI,cAAc,OAAO,KAAK,QAAQA,WAAU;AACpE,oBAAc,OAAO;IACvB,CAAC;EACH;ACzEA,MAAI,UAAU;IACZ,eAAc;AAAE,aAAQ,OAAQ,QAAQ,cAAe;IAAa;IAEpE,UAAU,cAAc,WAAW,QAAO;AACxC,aAAO,aAAa,WAAW,KAAK,SAAS,WAAW,MAAM,CAAC;IACjE;IAEA,YAAY,cAAc,WAAW,QAAQ,SAAS,MAAK;AACzD,UAAI,UAAU,KAAK,SAAS,cAAc,WAAW,MAAM;AAC3D,UAAI,MAAM,KAAK,SAAS,WAAW,MAAM;AACzC,UAAI,SAAS,YAAY,OAAO,UAAU,KAAK,OAAO;AACtD,mBAAa,QAAQ,KAAK,KAAK,UAAU,MAAM,CAAC;AAChD,aAAO;IACT;IAEA,SAAS,cAAc,WAAW,QAAO;AACvC,aAAO,KAAK,MAAM,aAAa,QAAQ,KAAK,SAAS,WAAW,MAAM,CAAC,CAAC;IAC1E;IAEA,mBAAmB,UAAS;AAC1B,UAAG,CAAC,KAAK,aAAa,GAAE;AAAE;MAAO;AACjC,cAAQ,aAAa,SAAS,QAAQ,SAAS,CAAC,CAAC,GAAG,IAAI,OAAO,SAAS,IAAI;IAC9E;IAEA,UAAU,MAAM,MAAM,IAAG;AACvB,UAAG,KAAK,aAAa,GAAE;AACrB,YAAG,OAAO,OAAO,SAAS,MAAK;AAC7B,cAAG,KAAK,QAAQ,cAAc,KAAK,QAAO;AAExC,gBAAI,eAAe,QAAQ,SAAS,CAAC;AACrC,yBAAa,SAAS,KAAK;AAC3B,oBAAQ,aAAa,cAAc,IAAI,OAAO,SAAS,IAAI;UAC7D;AAEA,iBAAO,KAAK;AACZ,kBAAQ,OAAO,OAAO,EAAE,MAAM,IAAI,MAAM,IAAI;AAM5C,iBAAO,sBAAsB,MAAM;AACjC,gBAAI,SAAS,KAAK,gBAAgB,OAAO,SAAS,IAAI;AAEtD,gBAAG,QAAO;AACR,qBAAO,eAAe;YACxB,WAAU,KAAK,SAAS,YAAW;AACjC,qBAAO,OAAO,GAAG,CAAC;YACpB;UACF,CAAC;QACH;MACF,OAAO;AACL,aAAK,SAAS,EAAE;MAClB;IACF;IAEA,UAAU,MAAM,OAAO,eAAc;AACnC,UAAI,UAAU,OAAO,kBAAmB,WAAW,YAAY,mBAAmB;AAClF,eAAS,SAAS,GAAG,QAAQ,SAAS;IACxC;IAEA,UAAU,MAAK;AACb,aAAO,SAAS,OAAO,QAAQ,IAAI,OAAO,iBAAkB,2BAA8B,GAAG,IAAI;IACnG;IAEA,aAAa,MAAK;AAChB,eAAS,SAAS,GAAG;IACvB;IAEA,SAAS,OAAO,OAAM;AACpB,UAAG,OAAM;AAAE,aAAK,UAAU,qBAAqB,OAAO,EAAE;MAAE;AAC1D,aAAO,WAAW;IACpB;IAEA,SAAS,WAAW,QAAO;AAAE,aAAO,GAAG,aAAa;IAAS;IAE7D,gBAAgB,WAAU;AACxB,UAAI,OAAO,UAAU,SAAS,EAAE,UAAU,CAAC;AAC3C,UAAG,SAAS,IAAG;AAAE;MAAO;AACxB,aAAO,SAAS,eAAe,IAAI,KAAK,SAAS,cAAc,WAAW,QAAQ;IACpF;EACF;AAEA,MAAO,kBAAQ;ACtDf,MAAI,MAAM;IACR,KAAK,IAAG;AAAE,aAAO,SAAS,eAAe,EAAE,KAAK,SAAS,mBAAmB,IAAI;IAAE;IAElF,YAAY,IAAI,WAAU;AACxB,SAAG,UAAU,OAAO,SAAS;AAC7B,UAAG,GAAG,UAAU,WAAW,GAAE;AAAE,WAAG,gBAAgB,OAAO;MAAE;IAC7D;IAEA,IAAI,MAAM,OAAO,UAAS;AACxB,UAAG,CAAC,MAAK;AAAE,eAAO,CAAC;MAAE;AACrB,UAAI,QAAQ,MAAM,KAAK,KAAK,iBAAiB,KAAK,CAAC;AACnD,aAAO,WAAW,MAAM,QAAQ,QAAQ,IAAI;IAC9C;IAEA,gBAAgB,MAAK;AACnB,UAAI,WAAW,SAAS,cAAc,UAAU;AAChD,eAAS,YAAY;AACrB,aAAO,SAAS,QAAQ;IAC1B;IAEA,cAAc,IAAG;AAAE,aAAO,GAAG,SAAS,UAAU,GAAG,aAAa,cAAc,MAAM;IAAK;IAEzF,aAAa,SAAQ;AAAE,aAAO,QAAQ,aAAa,sBAAsB;IAAE;IAE3E,iBAAiB,MAAK;AACpB,YAAM,SAAS,KAAK;AACpB,YAAM,oBAAoB,KAAK,IAAI,UAAU,sBAAsB,yBAAyB,UAAU;AACtG,aAAO,KAAK,IAAI,MAAM,sBAAsB,iBAAiB,EAAE,OAAO,iBAAiB;IACzF;IAEA,sBAAsB,MAAM,KAAI;AAC9B,aAAO,KAAK,yBAAyB,KAAK,IAAI,MAAM,IAAI,kBAAkB,OAAO,GAAG,IAAI;IAC1F;IAEA,eAAe,MAAK;AAClB,aAAO,KAAK,MAAM,IAAI,QAAQ,MAAM,WAAW,IAAI,OAAO;IAC5D;IAEA,YAAY,GAAE;AACZ,UAAI,cAAc,EAAE,WAAW,EAAE,YAAY,EAAE,WAAY,EAAE,UAAU,EAAE,WAAW;AACpF,UAAI,aAAc,EAAE,kBAAkB,qBAAqB,EAAE,OAAO,aAAa,UAAU;AAC3F,UAAI,gBAAgB,EAAE,OAAO,aAAa,QAAQ,KAAK,EAAE,OAAO,aAAa,QAAQ,EAAE,YAAY,MAAM;AACzG,UAAI,mBAAmB,EAAE,OAAO,aAAa,QAAQ,KAAK,CAAC,EAAE,OAAO,aAAa,QAAQ,EAAE,WAAW,GAAG;AACzG,aAAO,eAAe,iBAAiB,cAAc;IACvD;IAEA,uBAAuB,GAAE;AAGvB,UAAI,iBAAkB,EAAE,UAAU,EAAE,OAAO,aAAa,QAAQ,MAAM,YACnE,EAAE,aAAa,EAAE,UAAU,aAAa,YAAY,MAAM;AAE7D,UAAG,gBAAe;AAChB,eAAO;MACT,OAAO;AACL,eAAO,CAAC,EAAE,oBAAoB,CAAC,KAAK,YAAY,CAAC;MACnD;IACF;IAEA,eAAe,GAAG,iBAAgB;AAChC,UAAI,OAAO,EAAE,kBAAkB,oBAAoB,EAAE,OAAO,aAAa,MAAM,IAAI;AACnF,UAAI;AAEJ,UAAG,EAAE,oBAAoB,SAAS,QAAQ,KAAK,YAAY,CAAC,GAAE;AAAE,eAAO;MAAM;AAC7E,UAAG,KAAK,WAAW,SAAS,KAAK,KAAK,WAAW,MAAM,GAAE;AAAE,eAAO;MAAM;AACxE,UAAG,EAAE,OAAO,mBAAkB;AAAE,eAAO;MAAM;AAE7C,UAAI;AACF,cAAM,IAAI,IAAI,IAAI;MACpB,SAAQE,IAAR;AACE,YAAI;AACF,gBAAM,IAAI,IAAI,MAAM,eAAe;QACrC,SAAQA,IAAR;AAEE,iBAAO;QACT;MACF;AAEA,UAAG,IAAI,SAAS,gBAAgB,QAAQ,IAAI,aAAa,gBAAgB,UAAS;AAChF,YAAG,IAAI,aAAa,gBAAgB,YAAY,IAAI,WAAW,gBAAgB,QAAO;AACpF,iBAAO,IAAI,SAAS,MAAM,CAAC,IAAI,KAAK,SAAS,GAAG;QAClD;MACF;AACA,aAAO,IAAI,SAAS,WAAW,MAAM;IACvC;IAEA,sBAAsB,IAAG;AACvB,UAAG,KAAK,WAAW,EAAE,GAAE;AAAE,WAAG,aAAa,aAAa,EAAE;MAAE;AAC1D,WAAK,WAAW,IAAI,aAAa,IAAI;IACvC;IAEA,0BAA0B,MAAM,UAAS;AACvC,UAAI,WAAW,SAAS,cAAc,UAAU;AAChD,eAAS,YAAY;AACrB,aAAO,KAAK,gBAAgB,SAAS,SAAS,QAAQ;IACxD;IAEA,UAAU,IAAI,WAAU;AACtB,cAAQ,GAAG,aAAa,SAAS,KAAK,GAAG,aAAa,iBAAiB,OAAO;IAChF;IAEA,YAAY,IAAI,WAAW,aAAY;AACrC,aAAO,GAAG,gBAAgB,YAAY,QAAQ,GAAG,aAAa,SAAS,CAAC,KAAK;IAC/E;IAEA,cAAc,IAAG;AAAE,aAAO,KAAK,IAAI,IAAI,IAAI,aAAa;IAAE;IAE1D,gBAAgB,IAAI,UAAS;AAC3B,aAAO,KAAK,IAAI,IAAI,GAAG,qBAAqB,kBAAkB,YAAY;IAC5E;IAEA,uBAAuB,MAAM,MAAK;AAMhC,UAAI,aAAa,oBAAI,IAAI;AACzB,UAAI,eAAe,oBAAI,IAAI;AAE3B,WAAK,QAAQ,CAAA,QAAO;AAClB,aAAK,yBAAyB,KAAK,IAAI,MAAM,IAAI,kBAAkB,OAAO,GAAG,IAAI,EAAE,QAAQ,CAAA,WAAU;AACnG,qBAAW,IAAI,GAAG;AAClB,eAAK,yBAAyB,KAAK,IAAI,QAAQ,IAAI,gBAAgB,GAAG,MAAM,EACzE,IAAI,CAAA,OAAM,SAAS,GAAG,aAAa,aAAa,CAAC,CAAC,EAClD,QAAQ,CAAA,aAAY,aAAa,IAAI,QAAQ,CAAC;QACnD,CAAC;MACH,CAAC;AAED,mBAAa,QAAQ,CAAA,aAAY,WAAW,OAAO,QAAQ,CAAC;AAE5D,aAAO;IACT;IAEA,yBAAyB,OAAO,QAAO;AACrC,UAAG,OAAO,cAAc,iBAAiB,GAAE;AACzC,eAAO,MAAM,OAAO,CAAA,OAAM,KAAK,mBAAmB,IAAI,MAAM,CAAC;MAC/D,OAAO;AACL,eAAO;MACT;IACF;IAEA,mBAAmB,MAAM,QAAO;AAC9B,aAAM,OAAO,KAAK,YAAW;AAC3B,YAAG,KAAK,WAAW,MAAM,GAAE;AAAE,iBAAO;QAAK;AACzC,YAAG,KAAK,aAAa,WAAW,MAAM,MAAK;AAAE,iBAAO;QAAM;MAC5D;IACF;IAEA,QAAQ,IAAI,KAAI;AAAE,aAAO,GAAG,WAAW,KAAK,GAAG,WAAW,EAAE,GAAG;IAAE;IAEjE,cAAc,IAAI,KAAI;AAAE,SAAG,WAAW,KAAK,OAAQ,GAAG,WAAW,EAAE,GAAG;IAAG;IAEzE,WAAW,IAAI,KAAK,OAAM;AACxB,UAAG,CAAC,GAAG,WAAW,GAAE;AAAE,WAAG,WAAW,IAAI,CAAC;MAAE;AAC3C,SAAG,WAAW,EAAE,GAAG,IAAI;IACzB;IAEA,cAAc,IAAI,KAAK,YAAY,YAAW;AAC5C,UAAI,WAAW,KAAK,QAAQ,IAAI,GAAG;AACnC,UAAG,aAAa,QAAU;AACxB,aAAK,WAAW,IAAI,KAAK,WAAW,UAAU,CAAC;MACjD,OAAO;AACL,aAAK,WAAW,IAAI,KAAK,WAAW,QAAQ,CAAC;MAC/C;IACF;IAEA,iBAAiB,QAAQ,MAAK;AAC5B,UAAG,CAAC,OAAO,aAAa,WAAW,GAAE;AAAE;MAAO;AAC9C,wBAAkB,QAAQ,CAAA,cAAa;AACrC,eAAO,UAAU,SAAS,SAAS,KAAK,KAAK,UAAU,IAAI,SAAS;MACtE,CAAC;AACD,wBAAkB,OAAO,CAAA,SAAQ,OAAO,aAAa,IAAI,CAAC,EAAE,QAAQ,CAAA,SAAQ;AAC1E,aAAK,aAAa,MAAM,OAAO,aAAa,IAAI,CAAC;MACnD,CAAC;IACH;IAEA,aAAa,QAAQ,QAAO;AAC1B,UAAG,OAAO,WAAW,GAAE;AACrB,eAAO,WAAW,IAAI,OAAO,WAAW;MAC1C;IACF;IAEA,SAAS,KAAI;AACX,UAAI,UAAU,SAAS,cAAc,OAAO;AAC5C,UAAG,SAAQ;AACT,YAAI,EAAC,QAAQ,QAAQ,SAAS,aAAY,IAAI,QAAQ;AACtD,YAAIC,WAAU,OAAO,QAAS,YAAY,IAAI,KAAK,MAAM;AACzD,YAAGA,YAAW,OAAO,iBAAkB,UAAS;AAAE;QAAO;AAEzD,YAAI,QAAQA,WAAU,eAAe;AACrC,iBAAS,QAAQ,GAAG,UAAU,KAAK,SAAS,KAAK,UAAU;MAC7D,OAAO;AACL,iBAAS,QAAQ;MACnB;IACF;IAEA,SAAS,IAAI,OAAO,aAAa,iBAAiB,aAAa,iBAAiB,aAAa,UAAS;AACpG,UAAI,WAAW,GAAG,aAAa,WAAW;AAC1C,UAAI,WAAW,GAAG,aAAa,WAAW;AAE1C,UAAG,aAAa,IAAG;AAAE,mBAAW;MAAgB;AAChD,UAAG,aAAa,IAAG;AAAE,mBAAW;MAAgB;AAChD,UAAI,QAAQ,YAAY;AACxB,cAAO,OAAM;QACX,KAAK;AAAM,iBAAO,SAAS;QAE3B,KAAK;AACH,eAAK,SAAS,IAAI,uBAAuB,MAAM;AAC7C,gBAAG,YAAY,GAAE;AAAE,uBAAS;YAAE;UAChC,CAAC;AACD,cAAG,KAAK,KAAK,IAAI,eAAe,GAAE;AAChC,eAAG,iBAAiB,QAAQ,MAAM,KAAK,aAAa,IAAI,qBAAqB,CAAC;UAChF;AACA;QAEF;AACE,cAAI,UAAU,SAAS,KAAK;AAC5B,cAAI,UAAU,MAAM,WAAW,KAAK,cAAc,IAAI,SAAS,IAAI,SAAS;AAC5E,cAAI,eAAe,KAAK,SAAS,IAAI,kBAAkB,OAAO;AAC9D,cAAG,MAAM,OAAO,GAAE;AAAE,mBAAO,SAAS,oCAAoC,OAAO;UAAE;AACjF,cAAG,UAAS;AACV,gBAAI,aAAa;AACjB,gBAAG,MAAM,SAAS,WAAU;AAC1B,kBAAI,UAAU,KAAK,QAAQ,IAAI,iBAAiB;AAChD,mBAAK,WAAW,IAAI,mBAAmB,MAAM,GAAG;AAChD,2BAAa,YAAY,MAAM;YACjC;AAEA,gBAAG,CAAC,cAAc,KAAK,QAAQ,IAAI,SAAS,GAAE;AAC5C,qBAAO;YACT,OAAO;AACL,uBAAS;AACT,oBAAM,IAAI,WAAW,MAAM;AACzB,oBAAG,YAAY,GAAE;AAAE,uBAAK,aAAa,IAAI,gBAAgB;gBAAE;cAC7D,GAAG,OAAO;AACV,mBAAK,WAAW,IAAI,WAAW,CAAC;YAClC;UACF,OAAO;AACL,uBAAW,MAAM;AACf,kBAAG,YAAY,GAAE;AAAE,qBAAK,aAAa,IAAI,kBAAkB,YAAY;cAAE;YAC3E,GAAG,OAAO;UACZ;AAEA,cAAI,OAAO,GAAG;AACd,cAAG,QAAQ,KAAK,KAAK,MAAM,eAAe,GAAE;AAC1C,iBAAK,iBAAiB,UAAU,MAAM;AACpC,oBAAM,KAAM,IAAI,SAAS,IAAI,EAAG,QAAQ,GAAG,CAAC,CAAC,IAAI,MAAM;AACrD,oBAAI,QAAQ,KAAK,cAAc,UAAU,QAAQ;AACjD,qBAAK,SAAS,OAAO,gBAAgB;AACrC,qBAAK,cAAc,OAAO,SAAS;cACrC,CAAC;YACH,CAAC;UACH;AACA,cAAG,KAAK,KAAK,IAAI,eAAe,GAAE;AAChC,eAAG,iBAAiB,QAAQ,MAAM;AAIhC,2BAAa,KAAK,QAAQ,IAAI,SAAS,CAAC;AACxC,mBAAK,aAAa,IAAI,gBAAgB;YACxC,CAAC;UACH;MACJ;IACF;IAEA,aAAa,IAAI,KAAK,cAAa;AACjC,UAAI,CAAC,OAAO,OAAO,IAAI,KAAK,QAAQ,IAAI,GAAG;AAC3C,UAAG,CAAC,cAAa;AAAE,uBAAe;MAAM;AACxC,UAAG,iBAAiB,OAAM;AACxB,aAAK,SAAS,IAAI,GAAG;AACrB,gBAAQ;MACV;IACF;IAEA,KAAK,IAAI,KAAI;AACX,UAAG,KAAK,QAAQ,IAAI,GAAG,MAAM,MAAK;AAAE,eAAO;MAAM;AACjD,WAAK,WAAW,IAAI,KAAK,IAAI;AAC7B,aAAO;IACT;IAEA,SAAS,IAAI,KAAK,UAAU,WAAW;IAAE,GAAE;AACzC,UAAI,CAAC,YAAY,IAAI,KAAK,QAAQ,IAAI,GAAG,KAAK,CAAC,GAAG,OAAO;AACzD;AACA,WAAK,WAAW,IAAI,KAAK,CAAC,cAAc,OAAO,CAAC;AAChD,aAAO;IACT;;;;IAKA,qBAAqB,QAAQ,MAAM,gBAAgB,mBAAkB;AAEnE,UAAG,OAAO,gBAAgB,OAAO,aAAa,eAAe,KAAK,CAAC,KAAK,aAAa,eAAe,GAAE;AACpG,aAAK,aAAa,iBAAiB,OAAO,aAAa,eAAe,CAAC;MACzE;AAEA,UAAG,KAAK,iBAAiB,KAAK,aAAa,cAAc,KAAK,KAAK,aAAa,iBAAiB,IAAG;AAClG,aAAK,aAAa,iBAAiB,wBAAwB;MAC7D;IACF;IAEA,gBAAgB,IAAI,MAAK;AACvB,UAAG,GAAG,aAAY;AAChB,WAAG,aAAa,iBAAiB,EAAE;MACrC,OAAO;AACL,gBAAQ,MAAM;;2EAEuD,GAAG;OACvE;MACH;AACA,WAAK,WAAW,IAAI,kBAAkB,IAAI;IAC5C;IAEA,gBAAgB,IAAG;AAAE,aAAO,KAAK,QAAQ,IAAI,gBAAgB;IAAE;IAE/D,YAAY,IAAG;AACb,aAAQ,GAAG,aAAa,KAAK,iBAC1B,KAAK,QAAQ,IAAI,eAAe,KAAK,KAAK,QAAQ,IAAI,iBAAiB;IAC5E;IAEA,UAAU,MAAK;AACb,YAAM,KAAK,KAAK,QAAQ,EAAE,QAAQ,CAAA,UAAS;AACzC,aAAK,cAAc,OAAO,eAAe;AACzC,aAAK,cAAc,OAAO,iBAAiB;MAC7C,CAAC;IACH;IAEA,WAAW,MAAK;AACd,aAAO,KAAK,gBAAgB,KAAK,aAAa,aAAa;IAC7D;IAEA,YAAY,MAAK;AACf,aAAO,KAAK,gBAAgB,KAAK,aAAa,UAAU,MAAM;IAChE;IAEA,aAAa,IAAI,SAAQ;AACvB,aAAO,CAAC,CAAC,QAAQ,KAAK,CAAA,WAAU,OAAO,SAAS,EAAE,CAAC;IACrD;IAEA,cAAc,IAAG;AACf,aAAO,KAAK,WAAW,EAAE,IAAI,KAAK,KAAK,IAAI,IAAI,IAAI,gBAAgB,EAAE,CAAC;IACxE;IAEA,cAAc,QAAQ,MAAM,OAAO,CAAC,GAAE;AACpC,UAAI,gBAAgB;AACpB,UAAI,iBAAiB,OAAO,aAAa,WAAW,OAAO,SAAS;AACpE,UAAG,kBAAkB,SAAS,SAAQ;AACpC,wBAAgB;MAClB;AACA,UAAI,UAAU,KAAK,YAAY,SAAY,gBAAgB,CAAC,CAAC,KAAK;AAClE,UAAI,YAAY,EAAC,SAAkB,YAAY,MAAM,QAAQ,KAAK,UAAU,CAAC,EAAC;AAC9E,UAAI,QAAQ,SAAS,UAAU,IAAI,WAAW,SAAS,SAAS,IAAI,IAAI,YAAY,MAAM,SAAS;AACnG,aAAO,cAAc,KAAK;IAC5B;IAEA,UAAU,MAAM,MAAK;AACnB,UAAG,OAAQ,SAAU,aAAY;AAC/B,eAAO,KAAK,UAAU,IAAI;MAC5B,OAAO;AACL,YAAI,SAAS,KAAK,UAAU,KAAK;AACjC,eAAO,YAAY;AACnB,eAAO;MACT;IACF;;;;IAKA,WAAW,QAAQ,QAAQ,OAAO,CAAC,GAAE;AACnC,UAAI,UAAU,IAAI,IAAI,KAAK,WAAW,CAAC,CAAC;AACxC,UAAI,YAAY,KAAK;AACrB,UAAI,cAAc,OAAO;AACzB,eAAQ,IAAI,YAAY,SAAS,GAAG,KAAK,GAAG,KAAI;AAC9C,YAAI,OAAO,YAAY,CAAC,EAAE;AAC1B,YAAG,CAAC,QAAQ,IAAI,IAAI,GAAE;AACpB,gBAAM,cAAc,OAAO,aAAa,IAAI;AAC5C,cAAG,OAAO,aAAa,IAAI,MAAM,gBAAgB,CAAC,aAAc,aAAa,KAAK,WAAW,OAAO,IAAI;AACtG,mBAAO,aAAa,MAAM,WAAW;UACvC;QACF,OAAO;AAQL,cAAG,SAAS,WAAW,OAAO,UAAU,OAAO,OAAM;AAEnD,mBAAO,aAAa,SAAS,OAAO,aAAa,IAAI,CAAC;UACxD;QACF;MACF;AAEA,UAAI,cAAc,OAAO;AACzB,eAAQ,IAAI,YAAY,SAAS,GAAG,KAAK,GAAG,KAAI;AAC9C,YAAI,OAAO,YAAY,CAAC,EAAE;AAC1B,YAAG,WAAU;AACX,cAAG,KAAK,WAAW,OAAO,KAAK,CAAC,OAAO,aAAa,IAAI,KAAK,CAAC,kBAAkB,SAAS,IAAI,GAAE;AAAE,mBAAO,gBAAgB,IAAI;UAAE;QAChI,OAAO;AACL,cAAG,CAAC,OAAO,aAAa,IAAI,GAAE;AAAE,mBAAO,gBAAgB,IAAI;UAAE;QAC/D;MACF;IACF;IAEA,kBAAkB,QAAQ,QAAO;AAE/B,UAAG,EAAE,kBAAkB,oBAAmB;AAAE,YAAI,WAAW,QAAQ,QAAQ,EAAC,SAAS,CAAC,OAAO,EAAC,CAAC;MAAE;AAEjG,UAAG,OAAO,UAAS;AACjB,eAAO,aAAa,YAAY,IAAI;MACtC,OAAO;AACL,eAAO,gBAAgB,UAAU;MACnC;IACF;IAEA,kBAAkB,IAAG;AACnB,aAAO,GAAG,sBAAsB,GAAG,SAAS,UAAU,GAAG,SAAS;IACpE;IAEA,aAAa,SAAS,gBAAgB,cAAa;AACjD,UAAG,mBAAmB,mBAAkB;AAAE,gBAAQ,MAAM;MAAE;AAC1D,UAAG,CAAC,IAAI,eAAe,OAAO,GAAE;AAAE;MAAO;AAEzC,UAAI,aAAa,QAAQ,QAAQ,QAAQ;AACzC,UAAG,CAAC,YAAW;AAAE,gBAAQ,MAAM;MAAE;AACjC,UAAG,KAAK,kBAAkB,OAAO,GAAE;AACjC,gBAAQ,kBAAkB,gBAAgB,YAAY;MACxD;IACF;IAEA,YAAY,IAAG;AAAE,aAAO,+BAA+B,KAAK,GAAG,OAAO,KAAK,GAAG,SAAS;IAAS;IAEhG,iBAAiB,IAAG;AAClB,UAAG,cAAc,oBAAoB,iBAAiB,QAAQ,GAAG,KAAK,kBAAkB,CAAC,KAAK,GAAE;AAC9F,WAAG,UAAU,GAAG,aAAa,SAAS,MAAM;MAC9C;IACF;IAEA,eAAe,IAAG;AAAE,aAAO,iBAAiB,QAAQ,GAAG,IAAI,KAAK;IAAE;IAElE,yBAAyB,IAAI,oBAAmB;AAC9C,aAAO,GAAG,gBAAgB,GAAG,aAAa,kBAAkB,MAAM,QAAQ,SAAS,KAAK,SAAS,EAAE;IACrG;IAEA,gBAAgB,WAAW,WAAU;AACnC,UAAG,IAAI,YAAY,WAAW,WAAW,CAAC,UAAU,WAAW,UAAU,CAAC,GAAE;AAC1E,YAAI,WAAW,CAAC;AAChB,kBAAU,WAAW,QAAQ,CAAA,cAAa;AACxC,cAAG,CAAC,UAAU,IAAG;AAEf,gBAAI,kBAAkB,UAAU,aAAa,KAAK,aAAa,UAAU,UAAU,KAAK,MAAM;AAC9F,gBAAG,CAAC,mBAAmB,UAAU,aAAa,KAAK,cAAa;AAC9D,uBAAS;;2BACqB,UAAU,aAAa,UAAU,WAAW,KAAK;;CAAQ;YACzF;AACA,qBAAS,KAAK,SAAS;UACzB;QACF,CAAC;AACD,iBAAS,QAAQ,CAAA,cAAa,UAAU,OAAO,CAAC;MAClD;IACF;IAEA,qBAAqB,WAAW,SAAS,OAAM;AAC7C,UAAI,gBAAgB,oBAAI,IAAI,CAAC,MAAM,aAAa,YAAY,UAAU,WAAW,CAAC;AAClF,UAAG,UAAU,QAAQ,YAAY,MAAM,QAAQ,YAAY,GAAE;AAC3D,cAAM,KAAK,UAAU,UAAU,EAC5B,OAAO,CAAA,SAAQ,CAAC,cAAc,IAAI,KAAK,KAAK,YAAY,CAAC,CAAC,EAC1D,QAAQ,CAAA,SAAQ,UAAU,gBAAgB,KAAK,IAAI,CAAC;AAEvD,eAAO,KAAK,KAAK,EACd,OAAO,CAAA,SAAQ,CAAC,cAAc,IAAI,KAAK,YAAY,CAAC,CAAC,EACrD,QAAQ,CAAA,SAAQ,UAAU,aAAa,MAAM,MAAM,IAAI,CAAC,CAAC;AAE5D,eAAO;MAET,OAAO;AACL,YAAI,eAAe,SAAS,cAAc,OAAO;AACjD,eAAO,KAAK,KAAK,EAAE,QAAQ,CAAA,SAAQ,aAAa,aAAa,MAAM,MAAM,IAAI,CAAC,CAAC;AAC/E,sBAAc,QAAQ,CAAA,SAAQ,aAAa,aAAa,MAAM,UAAU,aAAa,IAAI,CAAC,CAAC;AAC3F,qBAAa,YAAY,UAAU;AACnC,kBAAU,YAAY,YAAY;AAClC,eAAO;MACT;IACF;IAEA,UAAU,IAAI,MAAM,YAAW;AAC7B,UAAI,MAAM,IAAI,QAAQ,IAAI,QAAQ,KAAK,CAAC,GAAG,KAAK,CAAC,CAAC,YAAa,MAAM,SAAS,YAAY;AAC1F,UAAG,IAAG;AACJ,YAAI,CAAC,OAAO,KAAK,aAAa,IAAI;AAClC,eAAO;MACT,OAAO;AACL,eAAO,OAAO,eAAgB,aAAa,WAAW,IAAI;MAC5D;IACF;IAEA,aAAa,IAAI,MAAK;AACpB,WAAK,cAAc,IAAI,UAAU,CAAC,GAAG,CAAA,QAAO;AAC1C,eAAO,IAAI,OAAO,CAAC,CAAC,cAAc,CAAC,MAAM,iBAAiB,IAAI;MAChE,CAAC;IACH;IAEA,UAAU,IAAI,MAAM,IAAG;AACrB,UAAI,gBAAgB,GAAG,EAAE;AACzB,WAAK,cAAc,IAAI,UAAU,CAAC,GAAG,CAAA,QAAO;AAC1C,YAAI,gBAAgB,IAAI,UAAU,CAAC,CAAC,YAAa,MAAM,SAAS,YAAY;AAC5E,YAAG,iBAAiB,GAAE;AACpB,cAAI,aAAa,IAAI,CAAC,MAAM,IAAI,aAAa;QAC/C,OAAO;AACL,cAAI,KAAK,CAAC,MAAM,IAAI,aAAa,CAAC;QACpC;AACA,eAAO;MACT,CAAC;IACH;IAEA,sBAAsB,IAAG;AACvB,UAAI,MAAM,IAAI,QAAQ,IAAI,QAAQ;AAClC,UAAG,CAAC,KAAI;AAAE;MAAO;AAEjB,UAAI,QAAQ,CAAC,CAAC,MAAM,IAAI,QAAQ,MAAM,KAAK,UAAU,IAAI,MAAM,EAAE,CAAC;IACpE;IAEA,SAAS,IAAG;AACV,aAAO,GAAG,gBAAgB,GAAG,aAAa,YAAY;IACxD;EACF;AAEA,MAAO,cAAQ;AChiBf,MAAqB,cAArB,MAAiC;IAC/B,OAAO,SAAS,QAAQ,MAAK;AAC3B,UAAI,QAAQ,KAAK,YAAY;AAC7B,UAAI,aAAa,OAAO,aAAa,qBAAqB,EAAE,MAAM,GAAG;AACrE,UAAI,WAAW,WAAW,QAAQ,aAAa,WAAW,IAAI,CAAC,KAAK;AACpE,aAAO,KAAK,OAAO,MAAM,SAAS;IACpC;IAEA,OAAO,cAAc,QAAQ,MAAK;AAChC,UAAI,kBAAkB,OAAO,aAAa,oBAAoB,EAAE,MAAM,GAAG;AACzE,UAAI,gBAAgB,gBAAgB,QAAQ,aAAa,WAAW,IAAI,CAAC,KAAK;AAC9E,aAAO,iBAAiB,KAAK,SAAS,QAAQ,IAAI;IACpD;IAEA,OAAO,sBAAsB,MAAK;AAChC,aAAO,KAAK,yBAAyB;IACvC;IAEA,OAAO,wBAAwB,MAAK;AAClC,WAAK,uBAAuB;IAC9B;IAEA,YAAY,QAAQ,MAAM,MAAM,YAAW;AACzC,WAAK,MAAM,aAAa,WAAW,IAAI;AACvC,WAAK,SAAS;AACd,WAAK,OAAO;AACZ,WAAK,OAAO;AACZ,WAAK,OAAO;AACZ,WAAK,eAAe;AACpB,WAAK,UAAU;AACf,WAAK,YAAY;AACjB,WAAK,oBAAoB;AACzB,WAAK,UAAU,WAAU;MAAE;AAC3B,WAAK,eAAe,KAAK,YAAY,KAAK,IAAI;AAC9C,WAAK,OAAO,iBAAiB,uBAAuB,KAAK,YAAY;AACrE,WAAK,aAAa;IACpB;IAEA,WAAU;AAAE,aAAO,KAAK;IAAK;IAE7B,SAAS,UAAS;AAChB,WAAK,YAAY,KAAK,MAAM,QAAQ;AACpC,UAAG,KAAK,YAAY,KAAK,mBAAkB;AACzC,YAAG,KAAK,aAAa,KAAI;AACvB,eAAK,YAAY;AACjB,eAAK,oBAAoB;AACzB,eAAK,UAAU;AACf,eAAK,KAAK,iBAAiB,KAAK,QAAQ,KAAK,KAAK,KAAK,MAAM;AAC3D,yBAAa,YAAY,KAAK,QAAQ,KAAK,IAAI;AAC/C,iBAAK,QAAQ;UACf,CAAC;QACH,OAAO;AACL,eAAK,oBAAoB,KAAK;AAC9B,eAAK,KAAK,iBAAiB,KAAK,QAAQ,KAAK,KAAK,KAAK,SAAS;QAClE;MACF;IACF;IAEA,cAAa;AAAE,aAAO,KAAK;IAAa;IAExC,SAAQ;AACN,WAAK,KAAK,uBAAuB;AACjC,WAAK,eAAe;AACpB,WAAK,UAAU;AACf,WAAK,QAAQ;IACf;IAEA,SAAQ;AAAE,aAAO,KAAK;IAAQ;IAE9B,MAAM,SAAS,UAAS;AACtB,WAAK,OAAO,oBAAoB,uBAAuB,KAAK,YAAY;AACxE,WAAK,KAAK,iBAAiB,KAAK,QAAQ,KAAK,KAAK,EAAC,OAAO,OAAM,CAAC;AACjE,UAAG,CAAC,KAAK,aAAa,GAAE;AAAE,qBAAa,WAAW,KAAK,MAAM;MAAE;IACjE;IAEA,eAAc;AAAE,aAAO,KAAK;IAAW;;IAIvC,OAAO,UAAS;AACd,WAAK,UAAU,MAAM;AACnB,aAAK,OAAO,oBAAoB,uBAAuB,KAAK,YAAY;AACxE,iBAAS;MACX;IACF;IAEA,cAAa;AACX,UAAI,aAAa,KAAK,OAAO,aAAa,qBAAqB,EAAE,MAAM,GAAG;AAC1E,UAAG,WAAW,QAAQ,KAAK,GAAG,MAAM,IAAG;AACrC,qBAAa,YAAY,KAAK,QAAQ,KAAK,IAAI;AAC/C,aAAK,OAAO;MACd;IACF;IAEA,qBAAoB;AAClB,aAAO;QACL,eAAe,KAAK,KAAK;QACzB,MAAM,KAAK,KAAK;QAChB,eAAe,KAAK,KAAK;QACzB,MAAM,KAAK,KAAK;QAChB,MAAM,KAAK,KAAK;QAChB,KAAK,KAAK;QACV,MAAM,OAAO,KAAK,KAAK,SAAU,aAAa,KAAK,KAAK,KAAK,IAAI;MACnE;IACF;IAEA,SAAS,WAAU;AACjB,UAAG,KAAK,KAAK,UAAS;AACpB,YAAI,WAAW,UAAU,KAAK,KAAK,QAAQ,KAAK,SAAS,8BAA8B,KAAK,KAAK,UAAU;AAC3G,eAAO,EAAC,MAAM,KAAK,KAAK,UAAU,SAAkB;MACtD,OAAO;AACL,eAAO,EAAC,MAAM,WAAW,UAAU,gBAAe;MACpD;IACF;IAEA,cAAc,MAAK;AACjB,WAAK,OAAO,KAAK,QAAQ,KAAK,GAAG;AACjC,UAAG,CAAC,KAAK,MAAK;AAAE,iBAAS,kDAAkD,KAAK,OAAO,EAAC,OAAO,KAAK,QAAQ,UAAU,KAAI,CAAC;MAAE;IAC/H;EACF;ACxHA,MAAI,sBAAsB;AAE1B,MAAqB,eAArB,MAAqB,cAAa;IAChC,OAAO,WAAW,MAAK;AACrB,UAAI,MAAM,KAAK;AACf,UAAG,QAAQ,QAAU;AACnB,eAAO;MACT,OAAO;AACL,aAAK,WAAW,uBAAuB,SAAS;AAChD,eAAO,KAAK;MACd;IACF;IAEA,OAAO,gBAAgB,SAAS,KAAK,UAAS;AAC5C,UAAI,OAAO,KAAK,YAAY,OAAO,EAAE,KAAK,CAAAC,UAAQ,KAAK,WAAWA,KAAI,MAAM,GAAG;AAC/E,eAAS,IAAI,gBAAgB,IAAI,CAAC;IACpC;IAEA,OAAO,qBAAqB,QAAO;AACjC,UAAI,SAAS;AACb,kBAAI,iBAAiB,MAAM,EAAE,QAAQ,CAAA,UAAS;AAC5C,YAAG,MAAM,aAAa,oBAAoB,MAAM,MAAM,aAAa,aAAa,GAAE;AAChF;QACF;MACF,CAAC;AACD,aAAO,SAAS;IAClB;IAEA,OAAO,iBAAiB,SAAQ;AAC9B,UAAI,QAAQ,KAAK,YAAY,OAAO;AACpC,UAAI,WAAW,CAAC;AAChB,YAAM,QAAQ,CAAA,SAAQ;AACpB,YAAI,QAAQ,EAAC,MAAM,QAAQ,KAAI;AAC/B,YAAI,YAAY,QAAQ,aAAa,cAAc;AACnD,iBAAS,SAAS,IAAI,SAAS,SAAS,KAAK,CAAC;AAC9C,cAAM,MAAM,KAAK,WAAW,IAAI;AAChC,cAAM,gBAAgB,KAAK;AAC3B,cAAM,OAAO,KAAK,QAAQ,MAAM;AAChC,cAAM,gBAAgB,KAAK;AAC3B,cAAM,OAAO,KAAK;AAClB,cAAM,OAAO,KAAK;AAClB,YAAG,OAAO,KAAK,SAAU,YAAW;AAAE,gBAAM,OAAO,KAAK,KAAK;QAAE;AAC/D,iBAAS,SAAS,EAAE,KAAK,KAAK;MAChC,CAAC;AACD,aAAO;IACT;IAEA,OAAO,WAAW,SAAQ;AACxB,cAAQ,QAAQ;AAChB,cAAQ,gBAAgB,cAAc;AACtC,kBAAI,WAAW,SAAS,SAAS,CAAC,CAAC;IACrC;IAEA,OAAO,YAAY,SAAS,MAAK;AAC/B,kBAAI,WAAW,SAAS,SAAS,YAAI,QAAQ,SAAS,OAAO,EAAE,OAAO,CAAA,MAAK,CAAC,OAAO,GAAG,GAAG,IAAI,CAAC,CAAC;IACjG;IAEA,OAAO,WAAW,SAAS,OAAO,cAAa;AAC7C,UAAG,QAAQ,aAAa,UAAU,MAAM,MAAK;AAC3C,YAAI,WAAW,MAAM,OAAO,CAAA,SAAQ,CAAC,KAAK,YAAY,OAAO,EAAE,KAAK,CAAA,MAAK,OAAO,GAAG,GAAG,IAAI,CAAC,CAAC;AAC5F,oBAAI,cAAc,SAAS,SAAS,CAAC,GAAG,CAAC,aAAa,SAAS,OAAO,QAAQ,CAAC;AAC/E,gBAAQ,QAAQ;MAClB,OAAO;AAEL,YAAG,gBAAgB,aAAa,MAAM,SAAS,GAAE;AAAE,kBAAQ,QAAQ,aAAa;QAAM;AACtF,oBAAI,WAAW,SAAS,SAAS,KAAK;MACxC;IACF;IAEA,OAAO,iBAAiB,QAAO;AAC7B,UAAI,aAAa,YAAI,iBAAiB,MAAM;AAC5C,aAAO,MAAM,KAAK,UAAU,EAAE,OAAO,CAAA,OAAM,GAAG,SAAS,KAAK,YAAY,EAAE,EAAE,SAAS,CAAC;IACxF;IAEA,OAAO,YAAY,OAAM;AACvB,cAAQ,YAAI,QAAQ,OAAO,OAAO,KAAK,CAAC,GAAG,OAAO,CAAA,MAAK,YAAY,SAAS,OAAO,CAAC,CAAC;IACvF;IAEA,OAAO,wBAAwB,QAAO;AACpC,UAAI,aAAa,YAAI,iBAAiB,MAAM;AAC5C,aAAO,MAAM,KAAK,UAAU,EAAE,OAAO,CAAA,UAAS,KAAK,uBAAuB,KAAK,EAAE,SAAS,CAAC;IAC7F;IAEA,OAAO,uBAAuB,OAAM;AAClC,aAAO,KAAK,YAAY,KAAK,EAAE,OAAO,CAAA,MAAK,CAAC,YAAY,cAAc,OAAO,CAAC,KAAK,CAAC,YAAY,sBAAsB,CAAC,CAAC;IAC1H;IAEA,OAAO,wBAAwB,SAAQ;AACrC,cAAQ,QAAQ,CAAA,UAAS,YAAY,wBAAwB,MAAM,IAAI,CAAC;IAC1E;IAEA,YAAY,SAAS,MAAM,YAAW;AACpC,WAAK,aAAa,YAAI,aAAa,OAAO;AAC1C,WAAK,OAAO;AACZ,WAAK,aAAa;AAClB,WAAK,WACH,MAAM,KAAK,cAAa,uBAAuB,OAAO,KAAK,CAAC,CAAC,EAC1D,IAAI,CAAA,SAAQ,IAAI,YAAY,SAAS,MAAM,MAAM,KAAK,UAAU,CAAC;AAGtE,oBAAa,wBAAwB,KAAK,QAAQ;AAElD,WAAK,uBAAuB,KAAK,SAAS;IAC5C;IAEA,eAAc;AAAE,aAAO,KAAK;IAAW;IAEvC,UAAS;AAAE,aAAO,KAAK;IAAS;IAEhC,kBAAkB,MAAM,SAASJ,aAAW;AAC1C,WAAK,WACH,KAAK,SAAS,IAAI,CAAA,UAAS;AACzB,YAAG,MAAM,YAAY,GAAE;AACrB,eAAK;AACL,cAAG,KAAK,yBAAyB,GAAE;AAAE,iBAAK,WAAW;UAAE;QACzD,OAAO;AACL,gBAAM,cAAc,IAAI;AACxB,gBAAM,OAAO,MAAM;AACjB,iBAAK;AACL,gBAAG,KAAK,yBAAyB,GAAE;AAAE,mBAAK,WAAW;YAAE;UACzD,CAAC;QACH;AACA,eAAO;MACT,CAAC;AAEH,UAAI,iBAAiB,KAAK,SAAS,OAAO,CAAC,KAAK,UAAU;AACxD,YAAG,CAAC,MAAM,MAAK;AAAE,iBAAO;QAAI;AAC5B,YAAI,EAAC,MAAM,SAAQ,IAAI,MAAM,SAASA,YAAW,SAAS;AAC1D,YAAI,IAAI,IAAI,IAAI,IAAI,KAAK,EAAC,UAAoB,SAAS,CAAC,EAAC;AACzD,YAAI,IAAI,EAAE,QAAQ,KAAK,KAAK;AAC5B,eAAO;MACT,GAAG,CAAC,CAAC;AAEL,eAAQ,QAAQ,gBAAe;AAC7B,YAAI,EAAC,UAAU,QAAO,IAAI,eAAe,IAAI;AAC7C,iBAAS,SAAS,SAAS,MAAMA,WAAU;MAC7C;IACF;EACF;ACtJA,MAAI,OAAO;IACT,MAAM,UAAU,SAAQ;AAAE,aAAO,QAAQ,KAAK,CAAA,SAAQ,oBAAoB,IAAI;IAAE;IAEhF,YAAY,IAAI,iBAAgB;AAC9B,aACG,cAAc,qBAAqB,GAAG,QAAQ,YAC9C,cAAc,mBAAmB,GAAG,SAAS,UAC7C,CAAC,GAAG,YAAa,KAAK,MAAM,IAAI,CAAC,kBAAkB,mBAAmB,qBAAqB,iBAAiB,CAAC,KAC7G,cAAc,sBACb,GAAG,YAAY,KAAK,GAAG,aAAa,aAAa,MAAM,UAAY,CAAC,mBAAmB,GAAG,aAAa,UAAU,MAAM,QAAQ,GAAG,aAAa,aAAa,MAAM;IAExK;IAEA,aAAa,IAAI,iBAAgB;AAC/B,UAAG,KAAK,YAAY,IAAI,eAAe,GAAE;AAAE,YAAI;AAAE,aAAG,MAAM;QAAE,SAAQ,GAAR;QAAS;MAAE;AACvE,aAAO,CAAC,CAAC,SAAS,iBAAiB,SAAS,cAAc,WAAW,EAAE;IACzE;IAEA,sBAAsB,IAAG;AACvB,UAAI,QAAQ,GAAG;AACf,aAAM,OAAM;AACV,YAAG,KAAK,aAAa,OAAO,IAAI,KAAK,KAAK,sBAAsB,OAAO,IAAI,GAAE;AAC3E,iBAAO;QACT;AACA,gBAAQ,MAAM;MAChB;IACF;IAEA,WAAW,IAAG;AACZ,UAAI,QAAQ,GAAG;AACf,aAAM,OAAM;AACV,YAAG,KAAK,aAAa,KAAK,KAAK,KAAK,WAAW,KAAK,GAAE;AACpD,iBAAO;QACT;AACA,gBAAQ,MAAM;MAChB;IACF;IAEA,UAAU,IAAG;AACX,UAAI,QAAQ,GAAG;AACf,aAAM,OAAM;AACV,YAAG,KAAK,aAAa,KAAK,KAAK,KAAK,UAAU,KAAK,GAAE;AACnD,iBAAO;QACT;AACA,gBAAQ,MAAM;MAChB;IACF;EACF;AACA,MAAO,eAAQ;ACtCf,MAAI,QAAQ;IACV,gBAAgB;MACd,aAAY;AAAE,eAAO,KAAK,GAAG,aAAa,qBAAqB;MAAE;MAEjE,kBAAiB;AAAE,eAAO,KAAK,GAAG,aAAa,oBAAoB;MAAE;MAErE,UAAS;AAAE,aAAK,iBAAiB,KAAK,gBAAgB;MAAE;MAExD,UAAS;AACP,YAAI,gBAAgB,KAAK,gBAAgB;AACzC,YAAG,KAAK,mBAAmB,eAAc;AACvC,eAAK,iBAAiB;AACtB,cAAG,kBAAkB,IAAG;AACtB,iBAAK,OAAO,EAAE,aAAa,KAAK,GAAG,IAAI;UACzC;QACF;AAEA,YAAG,KAAK,WAAW,MAAM,IAAG;AAAE,eAAK,GAAG,QAAQ;QAAK;AACnD,aAAK,GAAG,cAAc,IAAI,YAAY,qBAAqB,CAAC;MAC9D;IACF;IAEA,gBAAgB;MACd,UAAS;AACP,aAAK,MAAM,KAAK,GAAG,aAAa,oBAAoB;AACpD,aAAK,UAAU,SAAS,eAAe,KAAK,GAAG,aAAa,cAAc,CAAC;AAC3E,qBAAa,gBAAgB,KAAK,SAAS,KAAK,KAAK,CAAA,QAAO;AAC1D,eAAK,MAAM;AACX,eAAK,GAAG,MAAM;QAChB,CAAC;MACH;MACA,YAAW;AACT,YAAI,gBAAgB,KAAK,GAAG;MAC9B;IACF;IACA,WAAW;MACT,UAAS;AACP,aAAK,aAAa,KAAK,GAAG;AAC1B,aAAK,WAAW,KAAK,GAAG;AACxB,aAAK,WAAW,iBAAiB,SAAS,CAAC,MAAM;AAC/C,cAAG,CAAC,EAAE,iBAAiB,CAAC,KAAK,GAAG,SAAS,EAAE,aAAa,GAAE;AAGxD,kBAAM,YAAY,EAAE,OAAO;AAC3B,yBAAK,aAAa,SAAS,KAAK,aAAK,WAAW,SAAS;UAC3D,OAAO;AACL,yBAAK,UAAU,KAAK,EAAE;UACxB;QACF,CAAC;AACD,aAAK,SAAS,iBAAiB,SAAS,CAAC,MAAM;AAC7C,cAAG,CAAC,EAAE,iBAAiB,CAAC,KAAK,GAAG,SAAS,EAAE,aAAa,GAAE;AAGxD,kBAAM,YAAY,EAAE,OAAO;AAC3B,yBAAK,aAAa,SAAS,KAAK,aAAK,UAAU,SAAS;UAC1D,OAAO;AACL,yBAAK,WAAW,KAAK,EAAE;UACzB;QACF,CAAC;AACD,aAAK,GAAG,iBAAiB,gBAAgB,MAAM,KAAK,GAAG,MAAM,CAAC;AAC9D,YAAG,OAAO,iBAAiB,KAAK,EAAE,EAAE,YAAY,QAAO;AACrD,uBAAK,WAAW,KAAK,EAAE;QACzB;MACF;IACF;EACF;AAEA,MAAI,sBAAsB,CAAC,OAAO;AAGhC,QAAG,CAAC,QAAQ,MAAM,EAAE,QAAQ,GAAG,SAAS,YAAY,CAAC,KAAK;AAAG,aAAO;AACpE,QAAG,CAAC,UAAU,MAAM,EAAE,QAAQ,iBAAiB,EAAE,EAAE,SAAS,KAAK;AAAG,aAAO;AAC3E,WAAO,oBAAoB,GAAG,aAAa;EAC7C;AAEA,MAAI,YAAY,CAAC,oBAAoB;AACnC,QAAG,iBAAgB;AACjB,aAAO,gBAAgB;IACzB,OAAO;AACL,aAAO,SAAS,gBAAgB,aAAa,SAAS,KAAK;IAC7D;EACF;AAEA,MAAI,SAAS,CAAC,oBAAoB;AAChC,QAAG,iBAAgB;AACjB,aAAO,gBAAgB,sBAAsB,EAAE;IACjD,OAAO;AAGL,aAAO,OAAO,eAAe,SAAS,gBAAgB;IACxD;EACF;AAEA,MAAI,MAAM,CAAC,oBAAoB;AAC7B,QAAG,iBAAgB;AACjB,aAAO,gBAAgB,sBAAsB,EAAE;IACjD,OAAO;AAGL,aAAO;IACT;EACF;AAEA,MAAI,kBAAkB,CAAC,IAAI,oBAAoB;AAC7C,QAAI,OAAO,GAAG,sBAAsB;AACpC,WAAO,KAAK,KAAK,KAAK,GAAG,KAAK,IAAI,eAAe,KAAK,KAAK,KAAK,KAAK,IAAI,KAAK,KAAK,KAAK,MAAM,KAAK,GAAG,KAAK,OAAO,eAAe;EACnI;AAEA,MAAI,qBAAqB,CAAC,IAAI,oBAAoB;AAChD,QAAI,OAAO,GAAG,sBAAsB;AACpC,WAAO,KAAK,KAAK,KAAK,MAAM,KAAK,IAAI,eAAe,KAAK,KAAK,KAAK,KAAK,IAAI,KAAK,KAAK,KAAK,MAAM,KAAK,MAAM,KAAK,OAAO,eAAe;EACzI;AAEA,MAAI,mBAAmB,CAAC,IAAI,oBAAoB;AAC9C,QAAI,OAAO,GAAG,sBAAsB;AACpC,WAAO,KAAK,KAAK,KAAK,GAAG,KAAK,IAAI,eAAe,KAAK,KAAK,KAAK,KAAK,IAAI,KAAK,KAAK,KAAK,MAAM,KAAK,GAAG,KAAK,OAAO,eAAe;EACnI;AAEA,QAAM,iBAAiB;IACrB,UAAS;AACP,WAAK,kBAAkB,oBAAoB,KAAK,EAAE;AAClD,UAAI,eAAe,UAAU,KAAK,eAAe;AACjD,UAAI,aAAa;AACjB,UAAI,mBAAmB;AACvB,UAAI,YAAY;AAEhB,UAAI,eAAe,KAAK,SAAS,kBAAkB,CAAC,UAAU,eAAe;AAC3E,oBAAY,MAAM;AAClB,aAAK,WAAW,eAAe,KAAK,IAAI,UAAU,EAAC,IAAI,WAAW,IAAI,UAAU,KAAI,GAAG,MAAM;AAC3F,sBAAY;QACd,CAAC;MACH,CAAC;AAED,UAAI,oBAAoB,KAAK,SAAS,kBAAkB,CAAC,UAAU,eAAe;AAChF,oBAAY,MAAM,WAAW,eAAe,EAAC,OAAO,QAAO,CAAC;AAC5D,aAAK,WAAW,eAAe,KAAK,IAAI,UAAU,EAAC,IAAI,WAAW,GAAE,GAAG,MAAM;AAC3E,sBAAY;AAEZ,iBAAO,sBAAsB,MAAM;AACjC,gBAAG,CAAC,iBAAiB,YAAY,KAAK,eAAe,GAAE;AACrD,yBAAW,eAAe,EAAC,OAAO,QAAO,CAAC;YAC5C;UACF,CAAC;QACH,CAAC;MACH,CAAC;AAED,UAAI,sBAAsB,KAAK,SAAS,kBAAkB,CAAC,aAAa,cAAc;AACpF,oBAAY,MAAM,UAAU,eAAe,EAAC,OAAO,MAAK,CAAC;AACzD,aAAK,WAAW,eAAe,KAAK,IAAI,aAAa,EAAC,IAAI,UAAU,GAAE,GAAG,MAAM;AAC7E,sBAAY;AAEZ,iBAAO,sBAAsB,MAAM;AACjC,gBAAG,CAAC,iBAAiB,WAAW,KAAK,eAAe,GAAE;AACpD,wBAAU,eAAe,EAAC,OAAO,MAAK,CAAC;YACzC;UACF,CAAC;QACH,CAAC;MACH,CAAC;AAED,WAAK,WAAW,CAAC,OAAO;AACtB,YAAI,YAAY,UAAU,KAAK,eAAe;AAE9C,YAAG,WAAU;AACX,yBAAe;AACf,iBAAO,UAAU;QACnB;AACA,YAAI,OAAO,KAAK,GAAG,sBAAsB;AACzC,YAAI,WAAW,KAAK,GAAG,aAAa,KAAK,WAAW,QAAQ,cAAc,CAAC;AAC3E,YAAI,cAAc,KAAK,GAAG,aAAa,KAAK,WAAW,QAAQ,iBAAiB,CAAC;AACjF,YAAI,YAAY,KAAK,GAAG;AACxB,YAAI,aAAa,KAAK,GAAG;AACzB,YAAI,gBAAgB,YAAY;AAChC,YAAI,kBAAkB,YAAY;AAGlC,YAAG,iBAAiB,YAAY,CAAC,cAAc,KAAK,OAAO,GAAE;AAC3D,uBAAa;AACb,uBAAa,UAAU,UAAU;QACnC,WAAU,mBAAmB,cAAc,KAAK,OAAO,GAAE;AACvD,uBAAa;QACf;AAEA,YAAG,YAAY,iBAAiB,gBAAgB,YAAY,KAAK,eAAe,GAAE;AAChF,4BAAkB,UAAU,UAAU;QACxC,WAAU,eAAe,mBAAmB,mBAAmB,WAAW,KAAK,eAAe,GAAE;AAC9F,8BAAoB,aAAa,SAAS;QAC5C;AACA,uBAAe;MACjB;AAEA,UAAG,KAAK,iBAAgB;AACtB,aAAK,gBAAgB,iBAAiB,UAAU,KAAK,QAAQ;MAC/D,OAAO;AACL,eAAO,iBAAiB,UAAU,KAAK,QAAQ;MACjD;IACF;IAEA,YAAW;AACT,UAAG,KAAK,iBAAgB;AACtB,aAAK,gBAAgB,oBAAoB,UAAU,KAAK,QAAQ;MAClE,OAAO;AACL,eAAO,oBAAoB,UAAU,KAAK,QAAQ;MACpD;IACF;IAEA,SAAS,UAAU,UAAS;AAC1B,UAAI,aAAa;AACjB,UAAI;AAEJ,aAAO,IAAI,SAAS;AAClB,YAAI,MAAM,KAAK,IAAI;AACnB,YAAI,gBAAgB,YAAY,MAAM;AAEtC,YAAG,iBAAiB,KAAK,gBAAgB,UAAS;AAChD,cAAG,OAAM;AACP,yBAAa,KAAK;AAClB,oBAAQ;UACV;AACA,uBAAa;AACb,mBAAS,GAAG,IAAI;QAClB,WAAU,CAAC,OAAM;AACf,kBAAQ,WAAW,MAAM;AACvB,yBAAa,KAAK,IAAI;AACtB,oBAAQ;AACR,qBAAS,GAAG,IAAI;UAClB,GAAG,aAAa;QAClB;MACF;IACF;EACF;AACA,MAAO,gBAAQ;ACnOf,MAAqB,aAArB,MAAgC;IAC9B,OAAO,SAAS,IAAI,UAAS;AAC3B,UAAG,CAAC,YAAI,SAAS,EAAE,KAAK,CAAC,GAAG,QAAQ,IAAI,eAAe,GAAE;AAAE,eAAO,SAAS;MAAE;AAC7E,YAAM,cAAc,GAAG,QAAQ,IAAI,eAAe;AAClD,YAAM,MAAM,YAAY,QAAQ,IAAI,eAAe,EAAE,aAAa,YAAY;AAC9E,kBAAY,iBAAiB,iBAAiB,OAAO,MAAM;AACzD,iBAAS;MACX,GAAG,EAAC,MAAM,KAAI,CAAC;IACjB;IAEA,YAAY,IAAG;AACb,WAAK,KAAK;AACV,WAAK,aAAa,GAAG,aAAa,eAAe,IAAI,SAAS,GAAG,aAAa,eAAe,GAAG,EAAE,IAAI;AACtG,WAAK,UAAU,GAAG,aAAa,YAAY,IAAI,SAAS,GAAG,aAAa,YAAY,GAAG,EAAE,IAAI;IAC/F;;IAIA,UAAU,KAAK,UAAU,mBAAkB;AACzC,UAAG,CAAC,KAAK,SAAS,GAAG,GAAE;AAGrB,oBAAI,cAAc,KAAK,IAAI,kBAAkB,CAAC,GAAG,CAAC,gBAAgB;AAChE,sBAAY,KAAK,GAAG;AACpB,iBAAO;QACT,CAAC;AACD;MACF;AAGA,WAAK,UAAU,KAAK,UAAU,iBAAiB;AAG/C,WAAK,YAAY,KAAK,QAAQ;AAI9B,kBAAI,cAAc,KAAK,IAAI,kBAAkB,CAAC,GAAG,CAAC,gBAAgB;AAChE,eAAO,YAAY,OAAO,CAAC,eAAe;AACxC,cAAI,OAAO;YACT,QAAQ,EAAC,KAAK,YAAY,OAAO,SAAQ;YACzC,SAAS;YACT,YAAY;UACd;AACA,cAAG,KAAK,cAAc,KAAK,aAAa,YAAW;AACjD,iBAAK,GAAG;cACN,IAAI,YAAY,oBAAoB,cAAc,IAAI;YACxD;UACF;AACA,cAAG,KAAK,WAAW,KAAK,UAAU,YAAW;AAC3C,iBAAK,GAAG;cACN,IAAI,YAAY,iBAAiB,cAAc,IAAI;YACrD;UACF;AACA,iBAAO,aAAa;QACtB,CAAC;MACH,CAAC;AAGD,UAAG,KAAK,kBAAkB,GAAG,GAAE;AAAE,aAAK,GAAG,gBAAgB,WAAW;MAAE;IACxE;;IAIA,SAAS,KAAI;AACX,aAAO,EAAG,KAAK,eAAe,QAAQ,KAAK,aAAa,QAAS,KAAK,YAAY,QAAQ,KAAK,UAAU;IAC3G;;;;;;;IAQA,UAAU,KAAK,UAAU,mBAAkB;AACzC,UAAG,CAAC,KAAK,eAAe,GAAG,GAAE;AAAE;MAAO;AAEtC,UAAI,aAAa,YAAI,QAAQ,KAAK,IAAI,YAAY;AAClD,UAAG,YAAW;AACZ,0BAAkB,UAAU;AAC5B,oBAAI,cAAc,KAAK,IAAI,YAAY;MACzC;AACA,WAAK,GAAG,gBAAgB,YAAY;AAEpC,UAAI,OAAO,EAAC,QAAQ,EAAC,KAAU,OAAO,SAAQ,GAAG,SAAS,MAAM,YAAY,MAAK;AACjF,WAAK,GAAG,cAAc,IAAI,YAAY,iBAAiB,KAAK,WAAW,IAAI,CAAC;IAC9E;IAEA,YAAY,KAAK,UAAS;AACxB,UAAG,CAAC,KAAK,kBAAkB,GAAG,GAAE;AAC9B,YAAG,KAAK,eAAe,GAAG,KAAK,KAAK,GAAG,UAAU,SAAS,oBAAoB,GAAE;AAC9E,eAAK,GAAG,UAAU,OAAO,oBAAoB;QAC/C;AACA;MACF;AAEA,UAAG,KAAK,eAAe,GAAG,GAAE;AAC1B,aAAK,GAAG,gBAAgB,eAAe;AACvC,YAAI,cAAc,KAAK,GAAG,aAAa,YAAY;AACnD,YAAI,cAAc,KAAK,GAAG,aAAa,YAAY;AAEnD,YAAG,gBAAgB,MAAK;AACtB,eAAK,GAAG,WAAW,gBAAgB,SAAS,OAAO;AACnD,eAAK,GAAG,gBAAgB,YAAY;QACtC;AACA,YAAG,gBAAgB,MAAK;AACtB,eAAK,GAAG,WAAW,gBAAgB,SAAS,OAAO;AACnD,eAAK,GAAG,gBAAgB,YAAY;QACtC;AAEA,YAAI,iBAAiB,KAAK,GAAG,aAAa,wBAAwB;AAClE,YAAG,mBAAmB,MAAK;AACzB,eAAK,GAAG,YAAY;AACpB,eAAK,GAAG,gBAAgB,wBAAwB;QAClD;AAEA,YAAI,OAAO,EAAC,QAAQ,EAAC,KAAU,OAAO,SAAQ,GAAG,SAAS,MAAM,YAAY,MAAK;AACjF,aAAK,GAAG,cAAc,IAAI,YAAY,oBAAoB,KAAK,cAAc,IAAI,CAAC;MACpF;AAGA,wBAAkB,QAAQ,CAAA,SAAQ;AAChC,YAAG,SAAS,wBAAwB,KAAK,eAAe,GAAG,GAAE;AAC3D,sBAAI,YAAY,KAAK,IAAI,IAAI;QAC/B;MACF,CAAC;IACH;IAEA,kBAAkB,KAAI;AAAE,aAAO,KAAK,eAAe,OAAO,QAAQ,KAAK,cAAc;IAAI;IACzF,eAAe,KAAI;AAAE,aAAO,KAAK,YAAY,OAAO,QAAQ,KAAK,WAAW;IAAI;IAEhF,kBAAkB,KAAI;AACpB,cAAQ,KAAK,eAAe,QAAQ,KAAK,cAAc,SAAS,KAAK,YAAY,QAAQ,KAAK,WAAW;IAC3G;;IAGA,eAAe,KAAI;AAAE,aAAO,KAAK,YAAY,QAAQ,KAAK,WAAW;IAAI;EAC3E;AChJA,MAAqB,uBAArB,MAA0C;IACxC,YAAY,iBAAiB,gBAAgB,YAAW;AACtD,UAAI,YAAY,oBAAI,IAAI;AACxB,UAAI,WAAW,IAAI,IAAI,CAAC,GAAG,eAAe,QAAQ,EAAE,IAAI,CAAA,UAAS,MAAM,EAAE,CAAC;AAE1E,UAAI,mBAAmB,CAAC;AAExB,YAAM,KAAK,gBAAgB,QAAQ,EAAE,QAAQ,CAAA,UAAS;AACpD,YAAG,MAAM,IAAG;AACV,oBAAU,IAAI,MAAM,EAAE;AACtB,cAAG,SAAS,IAAI,MAAM,EAAE,GAAE;AACxB,gBAAI,oBAAoB,MAAM,0BAA0B,MAAM,uBAAuB;AACrF,6BAAiB,KAAK,EAAC,WAAW,MAAM,IAAI,kBAAoC,CAAC;UACnF;QACF;MACF,CAAC;AAED,WAAK,cAAc,eAAe;AAClC,WAAK,aAAa;AAClB,WAAK,mBAAmB;AACxB,WAAK,kBAAkB,CAAC,GAAG,QAAQ,EAAE,OAAO,CAAA,OAAM,CAAC,UAAU,IAAI,EAAE,CAAC;IACtE;;;;;;;IAQA,UAAS;AACP,UAAI,YAAY,YAAI,KAAK,KAAK,WAAW;AACzC,WAAK,iBAAiB,QAAQ,CAAA,oBAAmB;AAC/C,YAAG,gBAAgB,mBAAkB;AACnC,gBAAM,SAAS,eAAe,gBAAgB,iBAAiB,GAAG,CAAA,iBAAgB;AAChF,kBAAM,SAAS,eAAe,gBAAgB,SAAS,GAAG,CAAA,SAAQ;AAChE,kBAAI,iBAAiB,KAAK,0BAA0B,KAAK,uBAAuB,MAAM,aAAa;AACnG,kBAAG,CAAC,gBAAe;AACjB,6BAAa,sBAAsB,YAAY,IAAI;cACrD;YACF,CAAC;UACH,CAAC;QACH,OAAO;AAEL,gBAAM,SAAS,eAAe,gBAAgB,SAAS,GAAG,CAAA,SAAQ;AAChE,gBAAI,iBAAiB,KAAK,0BAA0B;AACpD,gBAAG,CAAC,gBAAe;AACjB,wBAAU,sBAAsB,cAAc,IAAI;YACpD;UACF,CAAC;QACH;MACF,CAAC;AAED,UAAG,KAAK,cAAc,WAAU;AAC9B,aAAK,gBAAgB,QAAQ,EAAE,QAAQ,CAAA,WAAU;AAC/C,gBAAM,SAAS,eAAe,MAAM,GAAG,CAAA,SAAQ,UAAU,sBAAsB,cAAc,IAAI,CAAC;QACpG,CAAC;MACH;IACF;EACF;AChEA,MAAI,yBAAyB;AAE7B,WAAS,WAAW,UAAU,QAAQ;AAClC,QAAI,cAAc,OAAO;AACzB,QAAI;AACJ,QAAI;AACJ,QAAI;AACJ,QAAI;AACJ,QAAI;AAGJ,QAAI,OAAO,aAAa,0BAA0B,SAAS,aAAa,wBAAwB;AAC9F;IACF;AAGA,aAAS,IAAI,YAAY,SAAS,GAAG,KAAK,GAAG,KAAK;AAC9C,aAAO,YAAY,CAAC;AACpB,iBAAW,KAAK;AAChB,yBAAmB,KAAK;AACxB,kBAAY,KAAK;AAEjB,UAAI,kBAAkB;AAClB,mBAAW,KAAK,aAAa;AAC7B,oBAAY,SAAS,eAAe,kBAAkB,QAAQ;AAE9D,YAAI,cAAc,WAAW;AACzB,cAAI,KAAK,WAAW,SAAQ;AACxB,uBAAW,KAAK;UACpB;AACA,mBAAS,eAAe,kBAAkB,UAAU,SAAS;QACjE;MACJ,OAAO;AACH,oBAAY,SAAS,aAAa,QAAQ;AAE1C,YAAI,cAAc,WAAW;AACzB,mBAAS,aAAa,UAAU,SAAS;QAC7C;MACJ;IACJ;AAIA,QAAI,gBAAgB,SAAS;AAE7B,aAAS,IAAI,cAAc,SAAS,GAAG,KAAK,GAAG,KAAK;AAChD,aAAO,cAAc,CAAC;AACtB,iBAAW,KAAK;AAChB,yBAAmB,KAAK;AAExB,UAAI,kBAAkB;AAClB,mBAAW,KAAK,aAAa;AAE7B,YAAI,CAAC,OAAO,eAAe,kBAAkB,QAAQ,GAAG;AACpD,mBAAS,kBAAkB,kBAAkB,QAAQ;QACzD;MACJ,OAAO;AACH,YAAI,CAAC,OAAO,aAAa,QAAQ,GAAG;AAChC,mBAAS,gBAAgB,QAAQ;QACrC;MACJ;IACJ;EACJ;AAEA,MAAI;AACJ,MAAI,WAAW;AAEf,MAAI,MAAM,OAAO,aAAa,cAAc,SAAY;AACxD,MAAI,uBAAuB,CAAC,CAAC,OAAO,aAAa,IAAI,cAAc,UAAU;AAC7E,MAAI,oBAAoB,CAAC,CAAC,OAAO,IAAI,eAAe,8BAA8B,IAAI,YAAY;AAElG,WAAS,2BAA2B,KAAK;AACrC,QAAI,WAAW,IAAI,cAAc,UAAU;AAC3C,aAAS,YAAY;AACrB,WAAO,SAAS,QAAQ,WAAW,CAAC;EACxC;AAEA,WAAS,wBAAwB,KAAK;AAClC,QAAI,CAAC,OAAO;AACR,cAAQ,IAAI,YAAY;AACxB,YAAM,WAAW,IAAI,IAAI;IAC7B;AAEA,QAAI,WAAW,MAAM,yBAAyB,GAAG;AACjD,WAAO,SAAS,WAAW,CAAC;EAChC;AAEA,WAAS,uBAAuB,KAAK;AACjC,QAAI,WAAW,IAAI,cAAc,MAAM;AACvC,aAAS,YAAY;AACrB,WAAO,SAAS,WAAW,CAAC;EAChC;AAUA,WAAS,UAAU,KAAK;AACpB,UAAM,IAAI,KAAK;AACf,QAAI,sBAAsB;AAIxB,aAAO,2BAA2B,GAAG;IACvC,WAAW,mBAAmB;AAC5B,aAAO,wBAAwB,GAAG;IACpC;AAEA,WAAO,uBAAuB,GAAG;EACrC;AAYA,WAAS,iBAAiB,QAAQ,MAAM;AACpC,QAAI,eAAe,OAAO;AAC1B,QAAI,aAAa,KAAK;AACtB,QAAI,eAAe;AAEnB,QAAI,iBAAiB,YAAY;AAC7B,aAAO;IACX;AAEA,oBAAgB,aAAa,WAAW,CAAC;AACzC,kBAAc,WAAW,WAAW,CAAC;AAMrC,QAAI,iBAAiB,MAAM,eAAe,IAAI;AAC1C,aAAO,iBAAiB,WAAW,YAAY;IACnD,WAAW,eAAe,MAAM,iBAAiB,IAAI;AACjD,aAAO,eAAe,aAAa,YAAY;IACnD,OAAO;AACH,aAAO;IACX;EACJ;AAWA,WAAS,gBAAgB,MAAM,cAAc;AACzC,WAAO,CAAC,gBAAgB,iBAAiB,WACrC,IAAI,cAAc,IAAI,IACtB,IAAI,gBAAgB,cAAc,IAAI;EAC9C;AAKA,WAAS,aAAa,QAAQ,MAAM;AAChC,QAAI,WAAW,OAAO;AACtB,WAAO,UAAU;AACb,UAAI,YAAY,SAAS;AACzB,WAAK,YAAY,QAAQ;AACzB,iBAAW;IACf;AACA,WAAO;EACX;AAEA,WAAS,oBAAoB,QAAQ,MAAM,MAAM;AAC7C,QAAI,OAAO,IAAI,MAAM,KAAK,IAAI,GAAG;AAC7B,aAAO,IAAI,IAAI,KAAK,IAAI;AACxB,UAAI,OAAO,IAAI,GAAG;AACd,eAAO,aAAa,MAAM,EAAE;MAChC,OAAO;AACH,eAAO,gBAAgB,IAAI;MAC/B;IACJ;EACJ;AAEA,MAAI,oBAAoB;IACpB,QAAQ,SAAS,QAAQ,MAAM;AAC3B,UAAI,aAAa,OAAO;AACxB,UAAI,YAAY;AACZ,YAAI,aAAa,WAAW,SAAS,YAAY;AACjD,YAAI,eAAe,YAAY;AAC3B,uBAAa,WAAW;AACxB,uBAAa,cAAc,WAAW,SAAS,YAAY;QAC/D;AACA,YAAI,eAAe,YAAY,CAAC,WAAW,aAAa,UAAU,GAAG;AACjE,cAAI,OAAO,aAAa,UAAU,KAAK,CAAC,KAAK,UAAU;AAInD,mBAAO,aAAa,YAAY,UAAU;AAC1C,mBAAO,gBAAgB,UAAU;UACrC;AAIA,qBAAW,gBAAgB;QAC/B;MACJ;AACA,0BAAoB,QAAQ,MAAM,UAAU;IAChD;;;;;;;IAOA,OAAO,SAAS,QAAQ,MAAM;AAC1B,0BAAoB,QAAQ,MAAM,SAAS;AAC3C,0BAAoB,QAAQ,MAAM,UAAU;AAE5C,UAAI,OAAO,UAAU,KAAK,OAAO;AAC7B,eAAO,QAAQ,KAAK;MACxB;AAEA,UAAI,CAAC,KAAK,aAAa,OAAO,GAAG;AAC7B,eAAO,gBAAgB,OAAO;MAClC;IACJ;IAEA,UAAU,SAAS,QAAQ,MAAM;AAC7B,UAAI,WAAW,KAAK;AACpB,UAAI,OAAO,UAAU,UAAU;AAC3B,eAAO,QAAQ;MACnB;AAEA,UAAI,aAAa,OAAO;AACxB,UAAI,YAAY;AAGZ,YAAI,WAAW,WAAW;AAE1B,YAAI,YAAY,YAAa,CAAC,YAAY,YAAY,OAAO,aAAc;AACvE;QACJ;AAEA,mBAAW,YAAY;MAC3B;IACJ;IACA,QAAQ,SAAS,QAAQ,MAAM;AAC3B,UAAI,CAAC,KAAK,aAAa,UAAU,GAAG;AAChC,YAAI,gBAAgB;AACpB,YAAI,IAAI;AAKR,YAAI,WAAW,OAAO;AACtB,YAAI;AACJ,YAAI;AACJ,eAAM,UAAU;AACZ,qBAAW,SAAS,YAAY,SAAS,SAAS,YAAY;AAC9D,cAAI,aAAa,YAAY;AACzB,uBAAW;AACX,uBAAW,SAAS;UACxB,OAAO;AACH,gBAAI,aAAa,UAAU;AACvB,kBAAI,SAAS,aAAa,UAAU,GAAG;AACnC,gCAAgB;AAChB;cACJ;AACA;YACJ;AACA,uBAAW,SAAS;AACpB,gBAAI,CAAC,YAAY,UAAU;AACvB,yBAAW,SAAS;AACpB,yBAAW;YACf;UACJ;QACJ;AAEA,eAAO,gBAAgB;MAC3B;IACJ;EACJ;AAEA,MAAI,eAAe;AACnB,MAAI,2BAA2B;AAC/B,MAAI,YAAY;AAChB,MAAI,eAAe;AAEnB,WAAS,OAAO;EAAC;AAEjB,WAAS,kBAAkB,MAAM;AAC/B,QAAI,MAAM;AACR,aAAQ,KAAK,gBAAgB,KAAK,aAAa,IAAI,KAAM,KAAK;IAChE;EACF;AAEA,WAAS,gBAAgBK,aAAY;AAEnC,WAAO,SAASC,UAAS,UAAU,QAAQ,SAAS;AAClD,UAAI,CAAC,SAAS;AACZ,kBAAU,CAAC;MACb;AAEA,UAAI,OAAO,WAAW,UAAU;AAC9B,YAAI,SAAS,aAAa,eAAe,SAAS,aAAa,UAAU,SAAS,aAAa,QAAQ;AACrG,cAAI,aAAa;AACjB,mBAAS,IAAI,cAAc,MAAM;AACjC,iBAAO,YAAY;QACrB,OAAO;AACL,mBAAS,UAAU,MAAM;QAC3B;MACF,WAAW,OAAO,aAAa,0BAA0B;AACvD,iBAAS,OAAO;MAClB;AAEA,UAAI,aAAa,QAAQ,cAAc;AACvC,UAAI,oBAAoB,QAAQ,qBAAqB;AACrD,UAAI,cAAc,QAAQ,eAAe;AACzC,UAAI,oBAAoB,QAAQ,qBAAqB;AACrD,UAAI,cAAc,QAAQ,eAAe;AACzC,UAAI,wBAAwB,QAAQ,yBAAyB;AAC7D,UAAI,kBAAkB,QAAQ,mBAAmB;AACjD,UAAI,4BAA4B,QAAQ,6BAA6B;AACrE,UAAI,mBAAmB,QAAQ,oBAAoB;AACnD,UAAI,WAAW,QAAQ,YAAY,SAAS,QAAQ,OAAM;AAAE,eAAO,OAAO,YAAY,KAAK;MAAG;AAC9F,UAAI,eAAe,QAAQ,iBAAiB;AAG5C,UAAI,kBAAkB,uBAAO,OAAO,IAAI;AACxC,UAAI,mBAAmB,CAAC;AAExB,eAAS,gBAAgB,KAAK;AAC5B,yBAAiB,KAAK,GAAG;MAC3B;AAEA,eAAS,wBAAwB,MAAM,gBAAgB;AACrD,YAAI,KAAK,aAAa,cAAc;AAClC,cAAI,WAAW,KAAK;AACpB,iBAAO,UAAU;AAEf,gBAAI,MAAM;AAEV,gBAAI,mBAAmB,MAAM,WAAW,QAAQ,IAAI;AAGlD,8BAAgB,GAAG;YACrB,OAAO;AAIL,8BAAgB,QAAQ;AACxB,kBAAI,SAAS,YAAY;AACvB,wCAAwB,UAAU,cAAc;cAClD;YACF;AAEA,uBAAW,SAAS;UACtB;QACF;MACF;AAUA,eAAS,WAAW,MAAM,YAAY,gBAAgB;AACpD,YAAI,sBAAsB,IAAI,MAAM,OAAO;AACzC;QACF;AAEA,YAAI,YAAY;AACd,qBAAW,YAAY,IAAI;QAC7B;AAEA,wBAAgB,IAAI;AACpB,gCAAwB,MAAM,cAAc;MAC9C;AA8BA,eAAS,UAAU,MAAM;AACvB,YAAI,KAAK,aAAa,gBAAgB,KAAK,aAAa,0BAA0B;AAChF,cAAI,WAAW,KAAK;AACpB,iBAAO,UAAU;AACf,gBAAI,MAAM,WAAW,QAAQ;AAC7B,gBAAI,KAAK;AACP,8BAAgB,GAAG,IAAI;YACzB;AAGA,sBAAU,QAAQ;AAElB,uBAAW,SAAS;UACtB;QACF;MACF;AAEA,gBAAU,QAAQ;AAElB,eAAS,gBAAgB,IAAI;AAC3B,oBAAY,EAAE;AAEd,YAAI,WAAW,GAAG;AAClB,eAAO,UAAU;AACf,cAAI,cAAc,SAAS;AAE3B,cAAI,MAAM,WAAW,QAAQ;AAC7B,cAAI,KAAK;AACP,gBAAI,kBAAkB,gBAAgB,GAAG;AAGzC,gBAAI,mBAAmB,iBAAiB,UAAU,eAAe,GAAG;AAClE,uBAAS,WAAW,aAAa,iBAAiB,QAAQ;AAC1D,sBAAQ,iBAAiB,QAAQ;YACnC,OAAO;AACL,8BAAgB,QAAQ;YAC1B;UACF,OAAO;AAGL,4BAAgB,QAAQ;UAC1B;AAEA,qBAAW;QACb;MACF;AAEA,eAAS,cAAc,QAAQ,kBAAkB,gBAAgB;AAI/D,eAAO,kBAAkB;AACvB,cAAI,kBAAkB,iBAAiB;AACvC,cAAK,iBAAiB,WAAW,gBAAgB,GAAI;AAGnD,4BAAgB,cAAc;UAChC,OAAO;AAGL;cAAW;cAAkB;cAAQ;;YAA2B;UAClE;AACA,6BAAmB;QACrB;MACF;AAEA,eAAS,QAAQ,QAAQ,MAAMC,eAAc;AAC3C,YAAI,UAAU,WAAW,IAAI;AAE7B,YAAI,SAAS;AAGX,iBAAO,gBAAgB,OAAO;QAChC;AAEA,YAAI,CAACA,eAAc;AAEjB,cAAI,qBAAqB,kBAAkB,QAAQ,IAAI;AACvD,cAAI,uBAAuB,OAAO;AAChC;UACF,WAAW,8BAA8B,aAAa;AACpD,qBAAS;AAKT,sBAAU,MAAM;UAClB;AAGAF,sBAAW,QAAQ,IAAI;AAEvB,sBAAY,MAAM;AAElB,cAAI,0BAA0B,QAAQ,IAAI,MAAM,OAAO;AACrD;UACF;QACF;AAEA,YAAI,OAAO,aAAa,YAAY;AAClC,wBAAc,QAAQ,IAAI;QAC5B,OAAO;AACL,4BAAkB,SAAS,QAAQ,IAAI;QACzC;MACF;AAEA,eAAS,cAAc,QAAQ,MAAM;AACnC,YAAI,WAAW,iBAAiB,QAAQ,IAAI;AAC5C,YAAI,iBAAiB,KAAK;AAC1B,YAAI,mBAAmB,OAAO;AAC9B,YAAI;AACJ,YAAI;AAEJ,YAAI;AACJ,YAAI;AACJ,YAAI;AAGJ;AAAO,iBAAO,gBAAgB;AAC5B,4BAAgB,eAAe;AAC/B,2BAAe,WAAW,cAAc;AAGxC,mBAAO,CAAC,YAAY,kBAAkB;AACpC,gCAAkB,iBAAiB;AAEnC,kBAAI,eAAe,cAAc,eAAe,WAAW,gBAAgB,GAAG;AAC5E,iCAAiB;AACjB,mCAAmB;AACnB,yBAAS;cACX;AAEA,+BAAiB,WAAW,gBAAgB;AAE5C,kBAAI,kBAAkB,iBAAiB;AAGvC,kBAAI,eAAe;AAEnB,kBAAI,oBAAoB,eAAe,UAAU;AAC/C,oBAAI,oBAAoB,cAAc;AAGpC,sBAAI,cAAc;AAGhB,wBAAI,iBAAiB,gBAAgB;AAInC,0BAAK,iBAAiB,gBAAgB,YAAY,GAAI;AACpD,4BAAI,oBAAoB,gBAAgB;AAMtC,yCAAe;wBACjB,OAAO;AAQL,iCAAO,aAAa,gBAAgB,gBAAgB;AAIpD,8BAAI,gBAAgB;AAGlB,4CAAgB,cAAc;0BAChC,OAAO;AAGL;8BAAW;8BAAkB;8BAAQ;;4BAA2B;0BAClE;AAEA,6CAAmB;AACnB,2CAAiB,WAAW,gBAAgB;wBAC9C;sBACF,OAAO;AAGL,uCAAe;sBACjB;oBACF;kBACF,WAAW,gBAAgB;AAEzB,mCAAe;kBACjB;AAEA,iCAAe,iBAAiB,SAAS,iBAAiB,kBAAkB,cAAc;AAC1F,sBAAI,cAAc;AAKhB,4BAAQ,kBAAkB,cAAc;kBAC1C;gBAEF,WAAW,oBAAoB,aAAa,mBAAmB,cAAc;AAE3E,iCAAe;AAGf,sBAAI,iBAAiB,cAAc,eAAe,WAAW;AAC3D,qCAAiB,YAAY,eAAe;kBAC9C;gBAEF;cACF;AAEA,kBAAI,cAAc;AAGhB,iCAAiB;AACjB,mCAAmB;AACnB,yBAAS;cACX;AAQA,kBAAI,gBAAgB;AAGlB,gCAAgB,cAAc;cAChC,OAAO;AAGL;kBAAW;kBAAkB;kBAAQ;;gBAA2B;cAClE;AAEA,iCAAmB;YACrB;AAMA,gBAAI,iBAAiB,iBAAiB,gBAAgB,YAAY,MAAM,iBAAiB,gBAAgB,cAAc,GAAG;AAExH,kBAAG,CAAC,UAAS;AAAE,yBAAS,QAAQ,cAAc;cAAG;AACjD,sBAAQ,gBAAgB,cAAc;YACxC,OAAO;AACL,kBAAI,0BAA0B,kBAAkB,cAAc;AAC9D,kBAAI,4BAA4B,OAAO;AACrC,oBAAI,yBAAyB;AAC3B,mCAAiB;gBACnB;AAEA,oBAAI,eAAe,WAAW;AAC5B,mCAAiB,eAAe,UAAU,OAAO,iBAAiB,GAAG;gBACvE;AACA,yBAAS,QAAQ,cAAc;AAC/B,gCAAgB,cAAc;cAChC;YACF;AAEA,6BAAiB;AACjB,+BAAmB;UACrB;AAEA,sBAAc,QAAQ,kBAAkB,cAAc;AAEtD,YAAI,mBAAmB,kBAAkB,OAAO,QAAQ;AACxD,YAAI,kBAAkB;AACpB,2BAAiB,QAAQ,IAAI;QAC/B;MACF;AAEA,UAAI,cAAc;AAClB,UAAI,kBAAkB,YAAY;AAClC,UAAI,aAAa,OAAO;AAExB,UAAI,CAAC,cAAc;AAGjB,YAAI,oBAAoB,cAAc;AACpC,cAAI,eAAe,cAAc;AAC/B,gBAAI,CAAC,iBAAiB,UAAU,MAAM,GAAG;AACvC,8BAAgB,QAAQ;AACxB,4BAAc,aAAa,UAAU,gBAAgB,OAAO,UAAU,OAAO,YAAY,CAAC;YAC5F;UACF,OAAO;AAEL,0BAAc;UAChB;QACF,WAAW,oBAAoB,aAAa,oBAAoB,cAAc;AAC5E,cAAI,eAAe,iBAAiB;AAClC,gBAAI,YAAY,cAAc,OAAO,WAAW;AAC9C,0BAAY,YAAY,OAAO;YACjC;AAEA,mBAAO;UACT,OAAO;AAEL,0BAAc;UAChB;QACF;MACF;AAEA,UAAI,gBAAgB,QAAQ;AAG1B,wBAAgB,QAAQ;MAC1B,OAAO;AACL,YAAI,OAAO,cAAc,OAAO,WAAW,WAAW,GAAG;AACvD;QACF;AAEA,gBAAQ,aAAa,QAAQ,YAAY;AAOzC,YAAI,kBAAkB;AACpB,mBAAS,IAAE,GAAG,MAAI,iBAAiB,QAAQ,IAAE,KAAK,KAAK;AACrD,gBAAI,aAAa,gBAAgB,iBAAiB,CAAC,CAAC;AACpD,gBAAI,YAAY;AACd,yBAAW,YAAY,WAAW,YAAY,KAAK;YACrD;UACF;QACF;MACF;AAEA,UAAI,CAAC,gBAAgB,gBAAgB,YAAY,SAAS,YAAY;AACpE,YAAI,YAAY,WAAW;AACzB,wBAAc,YAAY,UAAU,SAAS,iBAAiB,GAAG;QACnE;AAMA,iBAAS,WAAW,aAAa,aAAa,QAAQ;MACxD;AAEA,aAAO;IACT;EACF;AAEA,MAAI,WAAW,gBAAgB,UAAU;AAEzC,MAAO,uBAAQ;ACpuBf,MAAqB,WAArB,MAA8B;IAC5B,YAAY,MAAM,WAAW,IAAI,MAAM,SAAS,WAAW,OAAK,CAAC,GAAE;AACjE,WAAK,OAAO;AACZ,WAAK,aAAa,KAAK;AACvB,WAAK,YAAY;AACjB,WAAK,KAAK;AACV,WAAK,SAAS,KAAK,KAAK;AACxB,WAAK,OAAO;AACZ,WAAK,UAAU;AACf,WAAK,gBAAgB,CAAC;AACtB,WAAK,yBAAyB,CAAC;AAC/B,WAAK,YAAY;AACjB,WAAK,WAAW,MAAM,KAAK,SAAS;AACpC,WAAK,iBAAiB,CAAC;AACvB,WAAK,YAAY,KAAK,WAAW,QAAQ,QAAQ;AACjD,WAAK,kBAAkB,KAAK,WAAW,IAAI,KAAK,mBAAmB,IAAI,IAAI;AAC3E,WAAK,YAAY;QACf,aAAa,CAAC;QAAG,eAAe,CAAC;QAAG,qBAAqB,CAAC;QAC1D,YAAY,CAAC;QAAG,cAAc,CAAC;QAAG,gBAAgB,CAAC;QAAG,oBAAoB,CAAC;QAC3E,2BAA2B,CAAC;MAC9B;AACA,WAAK,eAAe,KAAK,gBAAgB,KAAK,WAAW;AACzD,WAAK,UAAU,KAAK;IACtB;IAEA,OAAO,MAAM,UAAS;AAAE,WAAK,UAAU,SAAS,MAAM,EAAE,KAAK,QAAQ;IAAE;IACvE,MAAM,MAAM,UAAS;AAAE,WAAK,UAAU,QAAQ,MAAM,EAAE,KAAK,QAAQ;IAAE;IAErE,YAAY,SAAS,MAAK;AACxB,WAAK,UAAU,SAAS,MAAM,EAAE,QAAQ,CAAA,aAAY,SAAS,GAAG,IAAI,CAAC;IACvE;IAEA,WAAW,SAAS,MAAK;AACvB,WAAK,UAAU,QAAQ,MAAM,EAAE,QAAQ,CAAA,aAAY,SAAS,GAAG,IAAI,CAAC;IACtE;IAEA,gCAA+B;AAC7B,UAAI,YAAY,KAAK,WAAW,QAAQ,UAAU;AAClD,kBAAI,IAAI,KAAK,WAAW,IAAI,2BAA2B,0BAA0B,CAAA,OAAM;AACrF,WAAG,aAAa,WAAW,EAAE;MAC/B,CAAC;IACH;IAEA,QAAQ,aAAY;AAClB,UAAI,EAAC,MAAM,YAAAL,aAAY,MAAM,WAAW,gBAAe,IAAI;AAC3D,UAAG,KAAK,WAAW,KAAK,CAAC,iBAAgB;AAAE;MAAO;AAElD,UAAI,UAAUA,YAAW,iBAAiB;AAC1C,UAAI,EAAC,gBAAgB,aAAY,IAAI,WAAW,YAAI,kBAAkB,OAAO,IAAI,UAAU,CAAC;AAC5F,UAAI,YAAYA,YAAW,QAAQ,UAAU;AAC7C,UAAI,iBAAiBA,YAAW,QAAQ,gBAAgB;AACxD,UAAI,oBAAoBA,YAAW,QAAQ,mBAAmB;AAC9D,UAAI,qBAAqBA,YAAW,QAAQ,kBAAkB;AAC9D,UAAI,QAAQ,CAAC;AACb,UAAI,UAAU,CAAC;AACf,UAAI,uBAAuB,CAAC;AAE5B,UAAI,wBAAwB;AAE5B,eAAS,MAAMQ,kBAAiB,QAAQ,eAAa,KAAK,cAAa;AACrE,YAAI,iBAAiB;;;;;UAKnB,cAAcA,iBAAgB,aAAa,aAAa,MAAM,QAAQ,CAAC;UACvE,YAAY,CAAC,SAAS;AACpB,gBAAG,YAAI,eAAe,IAAI,GAAE;AAAE,qBAAO;YAAK;AAG1C,gBAAG,aAAY;AAAE,qBAAO,KAAK;YAAG;AAChC,mBAAO,KAAK,MAAO,KAAK,gBAAgB,KAAK,aAAa,YAAY;UACxE;;UAEA,kBAAkB,CAAC,SAAS;AAAE,mBAAO,KAAK,aAAa,SAAS,MAAM;UAAW;;UAEjF,UAAU,CAAC,QAAQ,UAAU;AAC3B,gBAAI,EAAC,KAAK,SAAQ,IAAI,KAAK,gBAAgB,KAAK;AAChD,gBAAG,QAAQ,QAAU;AAAE,qBAAO,OAAO,YAAY,KAAK;YAAE;AAExD,iBAAK,aAAa,OAAO,GAAG;AAG5B,gBAAG,aAAa,GAAE;AAChB,qBAAO,sBAAsB,cAAc,KAAK;YAClD,WAAU,aAAa,IAAG;AACxB,kBAAI,YAAY,OAAO;AACvB,kBAAG,aAAa,CAAC,UAAU,aAAa,cAAc,GAAE;AACtD,oBAAI,iBAAiB,MAAM,KAAK,OAAO,QAAQ,EAAE,KAAK,CAAA,MAAK,CAAC,EAAE,aAAa,cAAc,CAAC;AAC1F,uBAAO,aAAa,OAAO,cAAc;cAC3C,OAAO;AACL,uBAAO,YAAY,KAAK;cAC1B;YACF,WAAU,WAAW,GAAE;AACrB,kBAAI,UAAU,MAAM,KAAK,OAAO,QAAQ,EAAE,QAAQ;AAClD,qBAAO,aAAa,OAAO,OAAO;YACpC;UACF;UACA,mBAAmB,CAAC,OAAO;AACzB,wBAAI,qBAAqB,IAAI,IAAI,gBAAgB,iBAAiB;AAClE,iBAAK,YAAY,SAAS,EAAE;AAE5B,gBAAI,YAAY;AAEhB,gBAAG,KAAK,uBAAuB,GAAG,EAAE,GAAE;AACpC,0BAAY,KAAK,uBAAuB,GAAG,EAAE;AAC7C,qBAAO,KAAK,uBAAuB,GAAG,EAAE;AACxC,oBAAM,KAAK,MAAM,WAAW,IAAI,IAAI;YACtC;AAEA,mBAAO;UACT;UACA,aAAa,CAAC,OAAO;AACnB,gBAAG,GAAG,cAAa;AAAE,mBAAK,mBAAmB,IAAI,IAAI;YAAE;AAGvD,gBAAG,cAAc,oBAAoB,GAAG,QAAO;AAC7C,iBAAG,SAAS,GAAG;YACjB,WAAU,cAAc,oBAAoB,GAAG,UAAS;AACtD,iBAAG,KAAK;YACV;AACA,gBAAG,YAAI,yBAAyB,IAAI,kBAAkB,GAAE;AACtD,sCAAwB;YAC1B;AAGA,gBAAI,YAAI,WAAW,EAAE,KAAK,KAAK,YAAY,EAAE,KAAM,YAAI,YAAY,EAAE,KAAK,KAAK,YAAY,GAAG,UAAU,GAAE;AACxG,mBAAK,WAAW,iBAAiB,EAAE;YACrC;AACA,kBAAM,KAAK,EAAE;UACf;UACA,iBAAiB,CAAC,OAAO,KAAK,gBAAgB,EAAE;UAChD,uBAAuB,CAAC,OAAO;AAC7B,gBAAG,GAAG,gBAAgB,GAAG,aAAa,SAAS,MAAM,MAAK;AAAE,qBAAO;YAAK;AACxE,gBAAG,GAAG,kBAAkB,QAAQ,GAAG,MACjC,YAAI,YAAY,GAAG,eAAe,WAAW,CAAC,YAAY,UAAU,SAAS,CAAC,GAAE;AAChF,qBAAO;YACT;AACA,gBAAG,KAAK,mBAAmB,EAAE,GAAE;AAAE,qBAAO;YAAM;AAC9C,gBAAG,KAAK,eAAe,EAAE,GAAE;AAAE,qBAAO;YAAM;AAE1C,mBAAO;UACT;UACA,aAAa,CAAC,OAAO;AACnB,gBAAG,YAAI,yBAAyB,IAAI,kBAAkB,GAAE;AACtD,sCAAwB;YAC1B;AACA,oBAAQ,KAAK,EAAE;AACf,iBAAK,mBAAmB,IAAI,KAAK;UACnC;UACA,mBAAmB,CAAC,QAAQ,SAAS;AAGnC,gBAAG,OAAO,MAAM,OAAO,WAAWA,gBAAe,KAAK,OAAO,OAAO,KAAK,IAAG;AAC1E,6BAAe,gBAAgB,MAAM;AACrC,qBAAO,YAAY,IAAI;AACvB,qBAAO,eAAe,YAAY,IAAI;YACxC;AACA,wBAAI,iBAAiB,QAAQ,IAAI;AACjC,wBAAI,qBAAqB,QAAQ,MAAM,gBAAgB,iBAAiB;AACxE,wBAAI,gBAAgB,MAAM,SAAS;AACnC,gBAAG,KAAK,eAAe,IAAI,GAAE;AAE3B,mBAAK,mBAAmB,MAAM;AAC9B,qBAAO;YACT;AACA,gBAAG,YAAI,YAAY,MAAM,GAAE;AACzB,eAAC,aAAa,YAAY,WAAW,EAClC,IAAI,CAAA,SAAQ,CAAC,MAAM,OAAO,aAAa,IAAI,GAAG,KAAK,aAAa,IAAI,CAAC,CAAC,EACtE,QAAQ,CAAC,CAAC,MAAM,SAAS,KAAK,MAAM;AACnC,oBAAG,SAAS,YAAY,OAAM;AAAE,yBAAO,aAAa,MAAM,KAAK;gBAAE;cACnE,CAAC;AAEH,qBAAO;YACT;AACA,gBAAG,YAAI,UAAU,QAAQ,SAAS,KAAM,OAAO,QAAQ,OAAO,KAAK,WAAW,qBAAqB,GAAG;AACpG,mBAAK,YAAY,WAAW,QAAQ,IAAI;AACxC,0BAAI,WAAW,QAAQ,MAAM,EAAC,WAAW,YAAI,UAAU,QAAQ,SAAS,EAAC,CAAC;AAC1E,sBAAQ,KAAK,MAAM;AACnB,0BAAI,sBAAsB,MAAM;AAChC,qBAAO;YACT;AACA,gBAAG,OAAO,SAAS,aAAa,OAAO,YAAY,OAAO,SAAS,WAAU;AAAE,qBAAO;YAAM;AAO5F,gBAAI,kBAAkB,WAAW,OAAO,WAAW,OAAO,KAAK,YAAI,YAAY,MAAM;AACrF,gBAAI,uBAAuB,mBAAmB,KAAK,gBAAgB,QAAQ,IAAI;AAC/E,gBAAG,OAAO,aAAa,WAAW,GAAE;AAClC,oBAAM,MAAM,IAAI,WAAW,MAAM;AAEjC,kBAAG,IAAI,YAAY,CAAC,KAAK,WAAW,CAAC,IAAI,eAAe,KAAK,OAAO,IAAG;AACrE,oBAAG,YAAI,cAAc,MAAM,GAAE;AAC3B,8BAAI,WAAW,QAAQ,MAAM,EAAC,WAAW,KAAI,CAAC;AAC9C,uBAAK,YAAY,WAAW,QAAQ,IAAI;AACxC,0BAAQ,KAAK,MAAM;gBACrB;AACA,4BAAI,sBAAsB,MAAM;AAChC,oBAAI,WAAW,OAAO,aAAa,YAAY;AAC/C,oBAAIC,SAAQ,WAAW,YAAI,QAAQ,QAAQ,YAAY,KAAK,OAAO,UAAU,IAAI,IAAI;AACrF,oBAAGA,QAAM;AACP,8BAAI,WAAW,QAAQ,cAAcA,MAAK;AAC1C,sBAAG,CAAC,iBAAgB;AAClB,6BAASA;kBACX;gBACF;cACF;YACF;AAGA,gBAAG,YAAI,WAAW,IAAI,GAAE;AACtB,kBAAI,cAAc,OAAO,aAAa,WAAW;AACjD,0BAAI,WAAW,QAAQ,MAAM,EAAC,SAAS,CAAC,UAAU,EAAC,CAAC;AACpD,kBAAG,gBAAgB,IAAG;AAAE,uBAAO,aAAa,aAAa,WAAW;cAAE;AACtE,qBAAO,aAAa,aAAa,KAAK,MAAM;AAC5C,0BAAI,sBAAsB,MAAM;AAChC,qBAAO;YACT;AAGA,gBAAG,KAAK,WAAW,YAAI,QAAQ,MAAM,YAAY,GAAE;AACjD,0BAAI,WAAW,QAAQ,cAAc,YAAI,QAAQ,MAAM,YAAY,CAAC;YACtE;AAEA,wBAAI,aAAa,MAAM,MAAM;AAG7B,gBAAG,mBAAmB,OAAO,SAAS,YAAY,CAAC,sBAAqB;AACtE,mBAAK,YAAY,WAAW,QAAQ,IAAI;AACxC,0BAAI,kBAAkB,QAAQ,IAAI;AAClC,0BAAI,iBAAiB,MAAM;AAC3B,sBAAQ,KAAK,MAAM;AACnB,0BAAI,sBAAsB,MAAM;AAChC,qBAAO;YACT,OAAO;AAEL,kBAAG,sBAAqB;AAAE,uBAAO,KAAK;cAAE;AACxC,kBAAG,YAAI,YAAY,MAAM,WAAW,CAAC,UAAU,SAAS,CAAC,GAAE;AACzD,qCAAqB,KAAK,IAAI,qBAAqB,QAAQ,MAAM,KAAK,aAAa,SAAS,CAAC,CAAC;cAChG;AAEA,0BAAI,iBAAiB,IAAI;AACzB,0BAAI,sBAAsB,IAAI;AAC9B,mBAAK,YAAY,WAAW,QAAQ,IAAI;AACxC,qBAAO;YACT;UACF;QACF;AACA,6BAASD,kBAAiB,QAAQ,cAAc;MAClD;AAEA,WAAK,YAAY,SAAS,SAAS;AACnC,WAAK,YAAY,WAAW,WAAW,SAAS;AAEhD,MAAAR,YAAW,KAAK,YAAY,MAAM;AAChC,aAAK,QAAQ,QAAQ,CAAC,CAAC,KAAK,SAAS,WAAW,KAAK,MAAM;AACzD,kBAAQ,QAAQ,CAAC,CAAC,KAAK,UAAU,KAAK,MAAM;AAC1C,iBAAK,cAAc,GAAG,IAAI,EAAC,KAAK,UAAU,OAAO,MAAK;UACxD,CAAC;AACD,cAAG,UAAU,QAAU;AACrB,wBAAI,IAAI,WAAW,IAAI,mBAAmB,SAAS,CAAA,UAAS;AAC1D,mBAAK,yBAAyB,KAAK;YACrC,CAAC;UACH;AACA,oBAAU,QAAQ,CAAA,OAAM;AACtB,gBAAI,QAAQ,UAAU,cAAc,QAAQ,MAAM;AAClD,gBAAG,OAAM;AAAE,mBAAK,yBAAyB,KAAK;YAAE;UAClD,CAAC;QACH,CAAC;AAGD,YAAG,aAAY;AACb,sBAAI,IAAI,KAAK,WAAW,IAAI,aAAa,aAAa,EAInD,OAAO,CAAA,OAAM,KAAK,KAAK,YAAY,EAAE,CAAC,EACtC,QAAQ,CAAA,OAAM;AACb,kBAAM,KAAK,GAAG,QAAQ,EAAE,QAAQ,CAAA,UAAS;AAIvC,mBAAK,yBAAyB,OAAO,IAAI;YAC3C,CAAC;UACH,CAAC;QACL;AAEA,cAAM,KAAK,MAAM,iBAAiB,IAAI;MACxC,CAAC;AAED,UAAGA,YAAW,eAAe,GAAE;AAC7B,2BAAmB;AACnB,mCAA2B,KAAK,aAAa;AAE7C,cAAM,KAAK,SAAS,iBAAiB,gBAAgB,CAAC,EAAE,QAAQ,CAAA,SAAQ;AACtE,cAAG,KAAK,MAAK;AACX,oBAAQ,MAAM,qGAAuG,IAAI;UAC3H;QACF,CAAC;MACH;AAEA,UAAG,qBAAqB,SAAS,GAAE;AACjC,QAAAA,YAAW,KAAK,yCAAyC,MAAM;AAC7D,+BAAqB,QAAQ,CAAA,WAAU,OAAO,QAAQ,CAAC;QACzD,CAAC;MACH;AAEA,MAAAA,YAAW,cAAc,MAAM,YAAI,aAAa,SAAS,gBAAgB,YAAY,CAAC;AACtF,kBAAI,cAAc,UAAU,YAAY;AACxC,YAAM,QAAQ,CAAA,OAAM,KAAK,WAAW,SAAS,EAAE,CAAC;AAChD,cAAQ,QAAQ,CAAA,OAAM,KAAK,WAAW,WAAW,EAAE,CAAC;AAEpD,WAAK,yBAAyB;AAE9B,UAAG,uBAAsB;AACvB,QAAAA,YAAW,OAAO;AAIlB,cAAM,YAAY,YAAI,QAAQ,uBAAuB,WAAW;AAChE,YAAG,aAAa,UAAU,QAAQ,gBAAgB,SAAS,SAAS,GAAE;AACpE,gBAAM,QAAQ,SAAS,cAAc,OAAO;AAC5C,gBAAM,OAAO;AACb,gBAAM,SAAS,UAAU,aAAa,MAAM;AAC5C,cAAG,QAAO;AACR,kBAAM,aAAa,QAAQ,MAAM;UACnC;AACA,gBAAM,OAAO,UAAU;AACvB,gBAAM,QAAQ,UAAU;AACxB,oBAAU,cAAc,aAAa,OAAO,SAAS;QACvD;AAGA,eAAO,eAAe,qBAAqB,EAAE,OAAO,KAAK,qBAAqB;MAChF;AACA,aAAO;IACT;IAEA,gBAAgB,IAAG;AAEjB,UAAG,YAAI,WAAW,EAAE,KAAK,YAAI,YAAY,EAAE,GAAE;AAAE,aAAK,WAAW,gBAAgB,EAAE;MAAE;AACnF,WAAK,WAAW,aAAa,EAAE;IACjC;IAEA,mBAAmB,MAAK;AACtB,UAAG,KAAK,gBAAgB,KAAK,aAAa,KAAK,SAAS,MAAM,MAAK;AACjE,aAAK,eAAe,KAAK,IAAI;AAC7B,eAAO;MACT,OAAO;AACL,eAAO;MACT;IACF;IAEA,yBAAyB,OAAO,QAAM,OAAM;AAI1C,UAAG,CAAC,SAAS,CAAC,KAAK,KAAK,YAAY,KAAK,GAAE;AAAE;MAAO;AAIpD,UAAG,KAAK,cAAc,MAAM,EAAE,GAAE;AAC9B,aAAK,uBAAuB,MAAM,EAAE,IAAI;AACxC,cAAM,OAAO;MACf,OAAO;AAEL,YAAG,CAAC,KAAK,mBAAmB,KAAK,GAAE;AACjC,gBAAM,OAAO;AACb,eAAK,gBAAgB,KAAK;QAC5B;MACF;IACF;IAEA,gBAAgB,IAAG;AACjB,UAAI,SAAS,GAAG,KAAK,KAAK,cAAc,GAAG,EAAE,IAAI,CAAC;AAClD,aAAO,UAAU,CAAC;IACpB;IAEA,aAAa,IAAI,KAAI;AACnB,kBAAI,UAAU,IAAI,gBAAgB,CAAAU,QAAMA,IAAG,aAAa,gBAAgB,GAAG,CAAC;IAC9E;IAEA,mBAAmB,IAAI,OAAM;AAC3B,UAAI,EAAC,KAAK,UAAU,MAAK,IAAI,KAAK,gBAAgB,EAAE;AACpD,UAAG,aAAa,QAAU;AAAE;MAAO;AAGnC,WAAK,aAAa,IAAI,GAAG;AAEzB,UAAG,CAAC,SAAS,CAAC,OAAM;AAElB;MACF;AAMA,UAAG,CAAC,GAAG,eAAc;AAAE;MAAO;AAE9B,UAAG,aAAa,GAAE;AAChB,WAAG,cAAc,aAAa,IAAI,GAAG,cAAc,iBAAiB;MACtE,WAAU,WAAW,GAAE;AACrB,YAAI,WAAW,MAAM,KAAK,GAAG,cAAc,QAAQ;AACnD,YAAI,WAAW,SAAS,QAAQ,EAAE;AAClC,YAAG,YAAY,SAAS,SAAS,GAAE;AACjC,aAAG,cAAc,YAAY,EAAE;QACjC,OAAO;AACL,cAAI,UAAU,SAAS,QAAQ;AAC/B,cAAG,WAAW,UAAS;AACrB,eAAG,cAAc,aAAa,IAAI,OAAO;UAC3C,OAAO;AACL,eAAG,cAAc,aAAa,IAAI,QAAQ,kBAAkB;UAC9D;QACF;MACF;AAEA,WAAK,iBAAiB,EAAE;IAC1B;IAEA,iBAAiB,IAAG;AAClB,UAAI,EAAC,MAAK,IAAI,KAAK,gBAAgB,EAAE;AACrC,UAAI,WAAW,UAAU,QAAQ,MAAM,KAAK,GAAG,cAAc,QAAQ;AACrE,UAAG,SAAS,QAAQ,KAAK,SAAS,SAAS,QAAQ,IAAG;AACpD,iBAAS,MAAM,GAAG,SAAS,SAAS,KAAK,EAAE,QAAQ,CAAA,UAAS,KAAK,yBAAyB,KAAK,CAAC;MAClG,WAAU,SAAS,SAAS,KAAK,SAAS,SAAS,OAAM;AACvD,iBAAS,MAAM,KAAK,EAAE,QAAQ,CAAA,UAAS,KAAK,yBAAyB,KAAK,CAAC;MAC7E;IACF;IAEA,2BAA0B;AACxB,UAAI,EAAC,gBAAgB,YAAAV,YAAU,IAAI;AACnC,UAAG,eAAe,SAAS,GAAE;AAC3B,QAAAA,YAAW,kBAAkB,gBAAgB,MAAM;AACjD,yBAAe,QAAQ,CAAA,OAAM;AAC3B,gBAAI,QAAQ,YAAI,cAAc,EAAE;AAChC,gBAAG,OAAM;AAAE,cAAAA,YAAW,gBAAgB,KAAK;YAAE;AAC7C,eAAG,OAAO;UACZ,CAAC;AACD,eAAK,WAAW,wBAAwB,cAAc;QACxD,CAAC;MACH;IACF;IAEA,gBAAgB,QAAQ,MAAK;AAC3B,UAAG,EAAE,kBAAkB,sBAAsB,OAAO,UAAS;AAAE,eAAO;MAAM;AAC5E,UAAG,OAAO,QAAQ,WAAW,KAAK,QAAQ,QAAO;AAAE,eAAO;MAAK;AAG/D,WAAK,QAAQ,OAAO;AAIpB,aAAO,CAAC,OAAO,YAAY,IAAI;IACjC;IAEA,aAAY;AAAE,aAAO,KAAK;IAAS;IAEnC,eAAe,IAAG;AAChB,aAAO,GAAG,aAAa,KAAK,gBAAgB,GAAG,aAAa,QAAQ;IACtE;IAEA,mBAAmB,MAAK;AACtB,UAAG,CAAC,KAAK,WAAW,GAAE;AAAE;MAAO;AAC/B,UAAI,CAAC,OAAO,GAAG,IAAI,IAAI,YAAI,sBAAsB,KAAK,WAAW,KAAK,SAAS;AAC/E,UAAG,KAAK,WAAW,KAAK,YAAI,gBAAgB,IAAI,MAAM,GAAE;AACtD,eAAO;MACT,OAAO;AACL,eAAO,SAAS,MAAM;MACxB;IACF;IAEA,QAAQ,QAAQ,OAAM;AAAE,aAAO,MAAM,KAAK,OAAO,QAAQ,EAAE,QAAQ,KAAK;IAAE;EAC5E;ACneA,MAAM,YAAY,oBAAI,IAAI;IACxB;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;EACF,CAAC;AACD,MAAM,aAAa,oBAAI,IAAI,CAAC,KAAK,GAAI,CAAC;AAE/B,MAAI,aAAa,CAAC,MAAM,OAAO,mBAAmB;AACvD,QAAI,IAAI;AACR,QAAI,gBAAgB;AACpB,QAAI,WAAW,UAAU,KAAK,eAAe,IAAI;AAEjD,QAAI,YAAY,KAAK,MAAM,sCAAsC;AACjE,QAAG,cAAc,MAAK;AAAE,YAAM,IAAI,MAAM,kBAAkB,MAAM;IAAE;AAElE,QAAI,UAAU,CAAC,EAAE;AACjB,gBAAY,UAAU,CAAC;AACvB,UAAM,UAAU,CAAC;AACjB,oBAAgB;AAGhB,SAAI,GAAG,IAAI,KAAK,QAAQ,KAAI;AAC1B,UAAG,KAAK,OAAO,CAAC,MAAM,KAAK;AAAE;MAAM;AACnC,UAAG,KAAK,OAAO,CAAC,MAAM,KAAI;AACxB,YAAI,OAAO,KAAK,MAAM,IAAI,GAAG,CAAC,MAAM;AACpC;AACA,YAAI,OAAO,KAAK,OAAO,CAAC;AACxB,YAAG,WAAW,IAAI,IAAI,GAAE;AACtB,cAAI,eAAe;AACnB;AACA,eAAI,GAAG,IAAI,KAAK,QAAQ,KAAI;AAC1B,gBAAG,KAAK,OAAO,CAAC,MAAM,MAAK;AAAE;YAAM;UACrC;AACA,cAAG,MAAK;AACN,iBAAK,KAAK,MAAM,eAAe,GAAG,CAAC;AACnC;UACF;QACF;MACF;IACF;AAEA,QAAI,UAAU,KAAK,SAAS;AAC5B,oBAAgB;AAChB,WAAM,WAAW,UAAU,SAAS,IAAI,QAAO;AAC7C,UAAI,OAAO,KAAK,OAAO,OAAO;AAC9B,UAAG,eAAc;AACf,YAAG,SAAS,OAAO,KAAK,MAAM,UAAU,GAAG,OAAO,MAAM,OAAM;AAC5D,0BAAgB;AAChB,qBAAW;QACb,OAAO;AACL,qBAAW;QACb;MACF,WAAU,SAAS,OAAO,KAAK,MAAM,UAAU,GAAG,OAAO,MAAM,MAAK;AAClE,wBAAgB;AAChB,mBAAW;MACb,WAAU,SAAS,KAAI;AACrB;MACF,OAAO;AACL,mBAAW;MACb;IACF;AACA,eAAW,KAAK,MAAM,UAAU,GAAG,KAAK,MAAM;AAE9C,QAAI,WACF,OAAO,KAAK,KAAK,EACd,IAAI,CAAA,SAAQ,MAAM,IAAI,MAAM,OAAO,OAAO,GAAG,SAAS,MAAM,IAAI,IAAI,EACpE,KAAK,GAAG;AAEb,QAAG,gBAAe;AAEhB,UAAI,YAAY,KAAK,QAAQ,QAAQ;AACrC,UAAG,UAAU,IAAI,GAAG,GAAE;AACpB,kBAAU,IAAI,MAAM,YAAY,aAAa,KAAK,KAAK,MAAM;MAC/D,OAAO;AACL,kBAAU,IAAI,MAAM,YAAY,aAAa,KAAK,KAAK,MAAM,cAAc;MAC7E;IACF,OAAO;AACL,UAAI,OAAO,KAAK,MAAM,eAAe,UAAU,CAAC;AAChD,gBAAU,IAAI,MAAM,aAAa,KAAK,KAAK,MAAM,WAAW;IAC9D;AAEA,WAAO,CAAC,SAAS,WAAW,QAAQ;EACtC;AAEA,MAAqB,WAArB,MAA8B;IAC5B,OAAO,QAAQ,MAAK;AAClB,UAAI,EAAC,CAAC,KAAK,GAAG,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC,KAAK,GAAG,MAAK,IAAI;AACzD,aAAO,KAAK,KAAK;AACjB,aAAO,KAAK,MAAM;AAClB,aAAO,KAAK,KAAK;AACjB,aAAO,EAAC,MAAM,OAAO,OAAO,SAAS,MAAM,QAAQ,UAAU,CAAC,EAAC;IACjE;IAEA,YAAY,QAAQ,UAAS;AAC3B,WAAK,SAAS;AACd,WAAK,WAAW,CAAC;AACjB,WAAK,UAAU;AACf,WAAK,UAAU,QAAQ;IACzB;IAEA,eAAc;AAAE,aAAO,KAAK;IAAO;IAEnC,SAAS,UAAS;AAChB,UAAI,CAAC,KAAK,OAAO,IAAI,KAAK,kBAAkB,KAAK,UAAU,KAAK,SAAS,UAAU,GAAG,UAAU,MAAM,CAAC,CAAC;AACxG,aAAO,CAAC,KAAK,OAAO;IACtB;IAEA,kBAAkB,UAAU,aAAa,SAAS,UAAU,GAAG,UAAU,gBAAgB,WAAU;AACjG,iBAAW,WAAW,IAAI,IAAI,QAAQ,IAAI;AAC1C,UAAI,SAAS,EAAC,QAAQ,IAAI,YAAwB,UAAoB,SAAS,oBAAI,IAAI,EAAC;AACxF,WAAK,eAAe,UAAU,MAAM,QAAQ,gBAAgB,SAAS;AACrE,aAAO,CAAC,OAAO,QAAQ,OAAO,OAAO;IACvC;IAEA,cAAc,MAAK;AAAE,aAAO,OAAO,KAAK,KAAK,UAAU,KAAK,CAAC,CAAC,EAAE,IAAI,CAAA,MAAK,SAAS,CAAC,CAAC;IAAE;IAEtF,oBAAoB,MAAK;AACvB,UAAG,CAAC,KAAK,UAAU,GAAE;AAAE,eAAO;MAAM;AACpC,aAAO,OAAO,KAAK,IAAI,EAAE,WAAW;IACtC;IAEA,aAAa,MAAM,KAAI;AAAE,aAAO,KAAK,UAAU,EAAE,GAAG;IAAE;IAEtD,YAAY,KAAI;AAGd,UAAG,KAAK,SAAS,UAAU,EAAE,GAAG,GAAE;AAChC,aAAK,SAAS,UAAU,EAAE,GAAG,EAAE,QAAQ;MACzC;IACF;IAEA,UAAU,MAAK;AACb,UAAI,OAAO,KAAK,UAAU;AAC1B,UAAI,QAAQ,CAAC;AACb,aAAO,KAAK,UAAU;AACtB,WAAK,WAAW,KAAK,aAAa,KAAK,UAAU,IAAI;AACrD,WAAK,SAAS,UAAU,IAAI,KAAK,SAAS,UAAU,KAAK,CAAC;AAE1D,UAAG,MAAK;AACN,YAAI,OAAO,KAAK,SAAS,UAAU;AAEnC,iBAAQ,OAAO,MAAK;AAClB,eAAK,GAAG,IAAI,KAAK,oBAAoB,KAAK,KAAK,GAAG,GAAG,MAAM,MAAM,KAAK;QACxE;AAEA,iBAAQ,OAAO,MAAK;AAAE,eAAK,GAAG,IAAI,KAAK,GAAG;QAAE;AAC5C,aAAK,UAAU,IAAI;MACrB;IACF;IAEA,oBAAoB,KAAK,OAAO,MAAM,MAAM,OAAM;AAChD,UAAG,MAAM,GAAG,GAAE;AACZ,eAAO,MAAM,GAAG;MAClB,OAAO;AACL,YAAI,OAAO,MAAM,OAAO,MAAM,MAAM;AAEpC,YAAG,MAAM,IAAI,GAAE;AACb,cAAI;AAEJ,cAAG,OAAO,GAAE;AACV,oBAAQ,KAAK,oBAAoB,MAAM,KAAK,IAAI,GAAG,MAAM,MAAM,KAAK;UACtE,OAAO;AACL,oBAAQ,KAAK,CAAC,IAAI;UACpB;AAEA,iBAAO,MAAM,MAAM;AACnB,kBAAQ,KAAK,WAAW,OAAO,OAAO,IAAI;AAC1C,gBAAM,MAAM,IAAI;QAClB,OAAO;AACL,kBAAQ,MAAM,MAAM,MAAM,UAAa,KAAK,GAAG,MAAM,SACnD,QAAQ,KAAK,WAAW,KAAK,GAAG,GAAG,OAAO,KAAK;QACnD;AAEA,cAAM,GAAG,IAAI;AACb,eAAO;MACT;IACF;IAEA,aAAa,QAAQ,QAAO;AAC1B,UAAG,OAAO,MAAM,MAAM,QAAU;AAC9B,eAAO;MACT,OAAO;AACL,aAAK,eAAe,QAAQ,MAAM;AAClC,eAAO;MACT;IACF;IAEA,eAAe,QAAQ,QAAO;AAC5B,eAAQ,OAAO,QAAO;AACpB,YAAI,MAAM,OAAO,GAAG;AACpB,YAAI,YAAY,OAAO,GAAG;AAC1B,YAAI,WAAW,SAAS,GAAG;AAC3B,YAAG,YAAY,IAAI,MAAM,MAAM,UAAa,SAAS,SAAS,GAAE;AAC9D,eAAK,eAAe,WAAW,GAAG;QACpC,OAAO;AACL,iBAAO,GAAG,IAAI;QAChB;MACF;AACA,UAAG,OAAO,IAAI,GAAE;AACd,eAAO,YAAY;MACrB;IACF;;;;;;;;;IAUA,WAAW,QAAQ,QAAQ,cAAa;AACtC,UAAI,SAAS,kCAAI,SAAW;AAC5B,eAAQ,OAAO,QAAO;AACpB,YAAI,MAAM,OAAO,GAAG;AACpB,YAAI,YAAY,OAAO,GAAG;AAC1B,YAAG,SAAS,GAAG,KAAK,IAAI,MAAM,MAAM,UAAa,SAAS,SAAS,GAAE;AACnE,iBAAO,GAAG,IAAI,KAAK,WAAW,WAAW,KAAK,YAAY;QAC5D,WAAU,QAAQ,UAAa,SAAS,SAAS,GAAE;AACjD,iBAAO,GAAG,IAAI,KAAK,WAAW,WAAW,CAAC,GAAG,YAAY;QAC3D;MACF;AACA,UAAG,cAAa;AACd,eAAO,OAAO;AACd,eAAO,OAAO;MAChB,WAAU,OAAO,IAAI,GAAE;AACrB,eAAO,YAAY;MACrB;AACA,aAAO;IACT;IAEA,kBAAkB,KAAI;AACpB,UAAI,CAAC,KAAK,OAAO,IAAI,KAAK,qBAAqB,KAAK,SAAS,UAAU,GAAG,KAAK,IAAI;AACnF,UAAI,CAAC,cAAc,SAAS,MAAM,IAAI,WAAW,KAAK,CAAC,CAAC;AACxD,aAAO,CAAC,cAAc,OAAO;IAC/B;IAEA,UAAU,MAAK;AACb,WAAK,QAAQ,CAAA,QAAO,OAAO,KAAK,SAAS,UAAU,EAAE,GAAG,CAAC;IAC3D;;IAIA,MAAK;AAAE,aAAO,KAAK;IAAS;IAE5B,iBAAiB,OAAO,CAAC,GAAE;AAAE,aAAO,CAAC,CAAC,KAAK,MAAM;IAAE;IAEnD,eAAe,MAAM,WAAU;AAC7B,UAAG,OAAQ,SAAU,UAAS;AAC5B,eAAO,UAAU,IAAI;MACvB,OAAO;AACL,eAAO;MACT;IACF;IAEA,cAAa;AACX,WAAK;AACL,aAAO,IAAI,KAAK,WAAW,KAAK,aAAa;IAC/C;;;;;;IAOA,eAAe,UAAU,WAAW,QAAQ,gBAAgB,YAAY,CAAC,GAAE;AACzE,UAAG,SAAS,QAAQ,GAAE;AAAE,eAAO,KAAK,sBAAsB,UAAU,WAAW,MAAM;MAAE;AACvF,UAAI,EAAC,CAAC,MAAM,GAAG,QAAO,IAAI;AAC1B,gBAAU,KAAK,eAAe,SAAS,SAAS;AAChD,UAAI,SAAS,SAAS,IAAI;AAC1B,UAAI,aAAa,OAAO;AACxB,UAAG,QAAO;AAAE,eAAO,SAAS;MAAG;AAI/B,UAAG,kBAAkB,UAAU,CAAC,SAAS,SAAQ;AAC/C,iBAAS,YAAY;AACrB,iBAAS,UAAU,KAAK,YAAY;MACtC;AAEA,aAAO,UAAU,QAAQ,CAAC;AAC1B,eAAQ,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAI;AACrC,aAAK,gBAAgB,SAAS,IAAI,CAAC,GAAG,WAAW,QAAQ,cAAc;AACvE,eAAO,UAAU,QAAQ,CAAC;MAC5B;AAMA,UAAG,QAAO;AACR,YAAI,OAAO;AACX,YAAI;AAKJ,YAAG,kBAAkB,SAAS,SAAQ;AACpC,iBAAO,kBAAkB,CAAC,SAAS;AACnC,kBAAQ,iBAAC,CAAC,YAAY,GAAG,SAAS,WAAY;QAChD,OAAO;AACL,kBAAQ;QACV;AACA,YAAG,MAAK;AAAE,gBAAM,QAAQ,IAAI;QAAK;AACjC,YAAI,CAAC,SAAS,eAAe,YAAY,IAAI,WAAW,OAAO,QAAQ,OAAO,IAAI;AAClF,iBAAS,YAAY;AACrB,eAAO,SAAS,aAAa,gBAAgB,UAAU;MACzD;IACF;IAEA,sBAAsB,UAAU,WAAW,QAAO;AAChD,UAAI,EAAC,CAAC,QAAQ,GAAG,UAAU,CAAC,MAAM,GAAG,SAAS,CAAC,MAAM,GAAG,OAAM,IAAI;AAClE,UAAI,CAAC,MAAM,UAAU,WAAW,KAAK,IAAI,UAAU,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,IAAI;AACtE,gBAAU,KAAK,eAAe,SAAS,SAAS;AAChD,UAAI,gBAAgB,aAAa,SAAS,SAAS;AACnD,eAAQ,IAAI,GAAG,IAAI,SAAS,QAAQ,KAAI;AACtC,YAAI,UAAU,SAAS,CAAC;AACxB,eAAO,UAAU,QAAQ,CAAC;AAC1B,iBAAQ,IAAI,GAAG,IAAI,QAAQ,QAAQ,KAAI;AAKrC,cAAI,iBAAiB;AACrB,eAAK,gBAAgB,QAAQ,IAAI,CAAC,GAAG,eAAe,QAAQ,cAAc;AAC1E,iBAAO,UAAU,QAAQ,CAAC;QAC5B;MACF;AAEA,UAAG,WAAW,WAAc,SAAS,QAAQ,EAAE,SAAS,KAAK,UAAU,SAAS,KAAK,QAAO;AAC1F,eAAO,SAAS,MAAM;AACtB,iBAAS,QAAQ,IAAI,CAAC;AACtB,eAAO,QAAQ,IAAI,MAAM;MAC3B;IACF;IAEA,gBAAgB,UAAU,WAAW,QAAQ,gBAAe;AAC1D,UAAG,OAAQ,aAAc,UAAS;AAChC,YAAI,CAAC,KAAK,OAAO,IAAI,KAAK,qBAAqB,OAAO,YAAY,UAAU,OAAO,QAAQ;AAC3F,eAAO,UAAU;AACjB,eAAO,UAAU,oBAAI,IAAI,CAAC,GAAG,OAAO,SAAS,GAAG,OAAO,CAAC;MAC1D,WAAU,SAAS,QAAQ,GAAE;AAC3B,aAAK,eAAe,UAAU,WAAW,QAAQ,gBAAgB,CAAC,CAAC;MACrE,OAAO;AACL,eAAO,UAAU;MACnB;IACF;IAEA,qBAAqB,YAAY,KAAK,UAAS;AAC7C,UAAI,YAAY,WAAW,GAAG,KAAK,SAAS,wBAAwB,OAAO,UAAU;AACrF,UAAI,QAAQ,EAAC,CAAC,aAAa,GAAG,IAAG;AACjC,UAAI,OAAO,YAAY,CAAC,SAAS,IAAI,GAAG;AAsBxC,gBAAU,YAAY,CAAC;AACvB,gBAAU,UAAU,IAAI,OAAO,KAAK,aAAa;AAEjD,UAAI,iBAAiB,CAAC,UAAU;AAChC,UAAI,CAAC,MAAM,OAAO,IAAI,KAAK,kBAAkB,WAAW,YAAY,UAAU,gBAAgB,KAAK;AAEnG,aAAO,UAAU;AAEjB,aAAO,CAAC,MAAM,OAAO;IACvB;EACF;AC9ZA,MAAI,aAAa,CAAC;AAClB,MAAI,0BAA0B;AAE9B,MAAI,KAAK;;IAEP,KAAK,GAAG,WAAW,UAAU,MAAM,UAAU,UAAS;AACpD,UAAI,CAAC,aAAa,WAAW,IAAI,YAAY,CAAC,MAAM,EAAC,UAAU,YAAY,SAAS,SAAQ,CAAC;AAC7F,UAAI,WAAW,SAAS,OAAO,CAAC,MAAM,MACpC,KAAK,MAAM,QAAQ,IAAI,CAAC,CAAC,aAAa,WAAW,CAAC;AAEpD,eAAS,QAAQ,CAAC,CAAC,MAAM,IAAI,MAAM;AACjC,YAAG,SAAS,aAAY;AAEtB,iBAAO,kCAAI,cAAgB;AAC3B,eAAK,WAAW,KAAK,YAAY,YAAY;QAC/C;AACA,aAAK,YAAY,KAAK,YAAY,UAAU,IAAI,EAAE,QAAQ,CAAA,OAAM;AAC9D,eAAK,QAAQ,MAAM,EAAE,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,IAAI;QACvE,CAAC;MACH,CAAC;IACH;IAEA,UAAU,IAAG;AACX,aAAO,CAAC,EAAE,GAAG,eAAe,GAAG,gBAAgB,GAAG,eAAe,EAAE,SAAS;IAC9E;;IAGA,aAAa,IAAG;AACd,YAAM,OAAO,GAAG,sBAAsB;AACtC,YAAM,eAAe,OAAO,eAAe,SAAS,gBAAgB;AACpE,YAAM,cAAc,OAAO,cAAc,SAAS,gBAAgB;AAElE,aACE,KAAK,QAAQ,KACb,KAAK,SAAS,KACd,KAAK,OAAO,eACZ,KAAK,MAAM;IAEf;;;IAMA,UAAU,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,GAAE,GAAE;AAC/D,UAAI,YAAY,GAAG,aAAa,IAAI;AACpC,UAAG,CAAC,WAAU;AAAE,cAAM,IAAI,MAAM,YAAY,kCAAkC,KAAK;MAAE;AACrF,WAAK,WAAW,OAAO,IAAI,WAAW,SAAS;IACjD;IAEA,cAAc,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,OAAO,QAAQ,QAAO,GAAE;AACjF,eAAS,UAAU,CAAC;AACpB,aAAO,aAAa;AACpB,kBAAI,cAAc,IAAI,OAAO,EAAC,QAAQ,QAAO,CAAC;IAChD;IAEA,UAAU,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,MAAK;AACzD,UAAI,EAAC,OAAO,MAAM,QAAQ,cAAc,SAAS,OAAO,YAAY,SAAQ,IAAI;AAChF,UAAI,WAAW,EAAC,SAAS,OAAO,QAAQ,cAAc,CAAC,CAAC,aAAY;AACpE,UAAI,YAAY,cAAc,YAAY,aAAa,aAAa;AACpE,UAAI,YAAY,UAAU,UAAU,aAAa,KAAK,QAAQ,QAAQ,CAAC,KAAK;AAC5E,YAAM,UAAU,CAAC,YAAY,cAAc;AACzC,YAAG,CAAC,WAAW,YAAY,GAAE;AAAE;QAAO;AACtC,YAAG,cAAc,UAAS;AACxB,cAAI,EAAC,QAAQ,QAAO,IAAI;AACxB,oBAAU,YAAY,YAAI,YAAY,QAAQ,IAAI,SAAS,OAAO;AAClE,cAAG,SAAQ;AAAE,qBAAS,UAAU;UAAQ;AACxC,qBAAW,UAAU,UAAU,WAAW,QAAQ,SAAS,UAAU,UAAU,QAAQ;QACzF,WAAU,cAAc,UAAS;AAC/B,cAAI,EAAC,UAAS,IAAI;AAClB,qBAAW,WAAW,UAAU,WAAW,SAAS,UAAU,WAAW,UAAU,QAAQ;QAC7F,OAAO;AACL,qBAAW,UAAU,WAAW,UAAU,WAAW,SAAS,UAAU,MAAM,UAAU,QAAQ;QAClG;MACF;AAGA,UAAG,KAAK,cAAc,KAAK,WAAU;AACnC,gBAAQ,KAAK,YAAY,KAAK,SAAS;MACzC,OAAO;AACL,aAAK,cAAc,WAAW,OAAO;MACvC;IACF;IAEA,cAAc,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,QAAO,GAAE;AACxE,WAAK,WAAW,gBAAgB,GAAG,MAAM,UAAU,YAAY,QAAQ,MAAM,QAAQ;IACvF;IAEA,WAAW,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,QAAO,GAAE;AACrE,WAAK,WAAW,iBAAiB,GAAG,MAAM,UAAU,YAAY,QAAQ,QAAQ;IAClF;IAEA,WAAW,GAAG,WAAW,UAAU,MAAM,UAAU,IAAG;AACpD,mBAAK,aAAa,EAAE;AAIpB,aAAO,sBAAsB,MAAM;AACjC,eAAO,sBAAsB,MAAM,aAAK,aAAa,EAAE,CAAC;MAC1D,CAAC;IACH;IAEA,iBAAiB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAG;AAC1D,mBAAK,sBAAsB,EAAE,KAAK,aAAK,WAAW,EAAE;AAEpD,aAAO,sBAAsB,MAAM;AACjC,eAAO,sBAAsB,MAAM,aAAK,sBAAsB,EAAE,KAAK,aAAK,WAAW,EAAE,CAAC;MAC1F,CAAC;IACH;IAEA,gBAAgB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAG;AACzD,iBAAW,KAAK,MAAM,QAAQ;IAChC;IAEA,eAAe,IAAI,YAAY,WAAW,OAAO,WAAW,KAAI;AAC9D,YAAM,KAAK,WAAW,IAAI;AAC1B,UAAG,IAAG;AACJ,WAAG,MAAM;AAET,eAAO,sBAAsB,MAAM;AACjC,iBAAO,sBAAsB,MAAM,GAAG,MAAM,CAAC;QAC/C,CAAC;MACH;IACF;IAEA,eAAe,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,OAAO,YAAY,MAAM,SAAQ,GAAE;AAC7F,WAAK,mBAAmB,IAAI,OAAO,CAAC,GAAG,YAAY,MAAM,MAAM,QAAQ;IACzE;IAEA,kBAAkB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,OAAO,YAAY,MAAM,SAAQ,GAAE;AAChG,WAAK,mBAAmB,IAAI,CAAC,GAAG,OAAO,YAAY,MAAM,MAAM,QAAQ;IACzE;IAEA,kBAAkB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,OAAO,YAAY,MAAM,SAAQ,GAAE;AAChG,WAAK,cAAc,IAAI,OAAO,YAAY,MAAM,MAAM,QAAQ;IAChE;IAEA,iBAAiB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,CAAC,MAAM,MAAM,IAAI,EAAC,GAAE;AACtF,WAAK,WAAW,IAAI,MAAM,MAAM,IAAI;IACtC;IAEA,gBAAgB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,YAAY,SAAQ,GAAE;AACvF,WAAK,mBAAmB,IAAI,CAAC,GAAG,CAAC,GAAG,YAAY,MAAM,MAAM,QAAQ;IACtE;IAEA,YAAY,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,SAAS,KAAK,MAAM,MAAM,SAAQ,GAAE;AAC3F,WAAK,OAAO,WAAW,MAAM,IAAI,SAAS,KAAK,MAAM,MAAM,QAAQ;IACrE;IAEA,UAAU,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,SAAS,YAAY,MAAM,SAAQ,GAAE;AAC1F,WAAK,KAAK,WAAW,MAAM,IAAI,SAAS,YAAY,MAAM,QAAQ;IACpE;IAEA,UAAU,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,SAAS,YAAY,MAAM,SAAQ,GAAE;AAC1F,WAAK,KAAK,WAAW,MAAM,IAAI,SAAS,YAAY,MAAM,QAAQ;IACpE;IAEA,cAAc,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,MAAM,CAAC,MAAM,GAAG,EAAC,GAAE;AAC5E,WAAK,iBAAiB,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,GAAG,CAAC,CAAC;IAC7C;IAEA,iBAAiB,GAAG,WAAW,UAAU,MAAM,UAAU,IAAI,EAAC,KAAI,GAAE;AAClE,WAAK,iBAAiB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;IACtC;;IAIA,KAAK,WAAW,MAAM,IAAI,SAAS,YAAY,MAAM,UAAS;AAC5D,UAAG,CAAC,KAAK,UAAU,EAAE,GAAE;AACrB,aAAK,OAAO,WAAW,MAAM,IAAI,SAAS,YAAY,MAAM,MAAM,QAAQ;MAC5E;IACF;IAEA,KAAK,WAAW,MAAM,IAAI,SAAS,YAAY,MAAM,UAAS;AAC5D,UAAG,KAAK,UAAU,EAAE,GAAE;AACpB,aAAK,OAAO,WAAW,MAAM,IAAI,SAAS,MAAM,YAAY,MAAM,QAAQ;MAC5E;IACF;IAEA,OAAO,WAAW,MAAM,IAAI,SAAS,KAAK,MAAM,MAAM,UAAS;AAC7D,aAAO,QAAQ;AACf,UAAI,CAAC,WAAW,gBAAgB,YAAY,IAAI,OAAO,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAClE,UAAI,CAAC,YAAY,iBAAiB,aAAa,IAAI,QAAQ,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AACtE,UAAG,UAAU,SAAS,KAAK,WAAW,SAAS,GAAE;AAC/C,YAAG,KAAK,UAAU,EAAE,GAAE;AACpB,cAAI,UAAU,MAAM;AAClB,iBAAK,mBAAmB,IAAI,iBAAiB,UAAU,OAAO,cAAc,EAAE,OAAO,YAAY,CAAC;AAClG,mBAAO,sBAAsB,MAAM;AACjC,mBAAK,mBAAmB,IAAI,YAAY,CAAC,CAAC;AAC1C,qBAAO,sBAAsB,MAAM,KAAK,mBAAmB,IAAI,eAAe,eAAe,CAAC;YAChG,CAAC;UACH;AACA,cAAI,QAAQ,MAAM;AAChB,iBAAK,mBAAmB,IAAI,CAAC,GAAG,WAAW,OAAO,aAAa,CAAC;AAChE,wBAAI,UAAU,IAAI,UAAU,CAAA,cAAa,UAAU,MAAM,UAAU,MAAM;AACzE,eAAG,cAAc,IAAI,MAAM,cAAc,CAAC;UAC5C;AACA,aAAG,cAAc,IAAI,MAAM,gBAAgB,CAAC;AAC5C,cAAG,aAAa,OAAM;AACpB,oBAAQ;AACR,uBAAW,OAAO,IAAI;UACxB,OAAO;AACL,iBAAK,WAAW,MAAM,SAAS,KAAK;UACtC;QACF,OAAO;AACL,cAAG,cAAc,UAAS;AAAE;UAAO;AACnC,cAAI,UAAU,MAAM;AAClB,iBAAK,mBAAmB,IAAI,gBAAgB,WAAW,OAAO,eAAe,EAAE,OAAO,aAAa,CAAC;AACpG,kBAAM,gBAAgB,WAAW,KAAK,eAAe,EAAE;AACvD,mBAAO,sBAAsB,MAAM;AAKjC,mBAAK,mBAAmB,IAAI,WAAW,CAAC,CAAC;AAGzC,qBAAO,sBAAsB,MAAM;AACjC,4BAAI,UAAU,IAAI,UAAU,CAAA,cAAa,UAAU,MAAM,UAAU,aAAa;AAChF,qBAAK,mBAAmB,IAAI,cAAc,cAAc;cAC1D,CAAC;YACH,CAAC;UACH;AACA,cAAI,QAAQ,MAAM;AAChB,iBAAK,mBAAmB,IAAI,CAAC,GAAG,UAAU,OAAO,YAAY,CAAC;AAC9D,eAAG,cAAc,IAAI,MAAM,cAAc,CAAC;UAC5C;AACA,aAAG,cAAc,IAAI,MAAM,gBAAgB,CAAC;AAC5C,cAAG,aAAa,OAAM;AACpB,oBAAQ;AACR,uBAAW,OAAO,IAAI;UACxB,OAAO;AACL,iBAAK,WAAW,MAAM,SAAS,KAAK;UACtC;QACF;MACF,OAAO;AACL,YAAG,KAAK,UAAU,EAAE,GAAE;AACpB,iBAAO,sBAAsB,MAAM;AACjC,eAAG,cAAc,IAAI,MAAM,gBAAgB,CAAC;AAC5C,wBAAI,UAAU,IAAI,UAAU,CAAA,cAAa,UAAU,MAAM,UAAU,MAAM;AACzE,eAAG,cAAc,IAAI,MAAM,cAAc,CAAC;UAC5C,CAAC;QACH,OAAO;AACL,iBAAO,sBAAsB,MAAM;AACjC,eAAG,cAAc,IAAI,MAAM,gBAAgB,CAAC;AAC5C,gBAAI,gBAAgB,WAAW,KAAK,eAAe,EAAE;AACrD,wBAAI,UAAU,IAAI,UAAU,CAAA,cAAa,UAAU,MAAM,UAAU,aAAa;AAChF,eAAG,cAAc,IAAI,MAAM,cAAc,CAAC;UAC5C,CAAC;QACH;MACF;IACF;IAEA,cAAc,IAAI,SAAS,YAAY,MAAM,MAAM,UAAS;AAC1D,aAAO,sBAAsB,MAAM;AACjC,YAAI,CAAC,UAAU,WAAW,IAAI,YAAI,UAAU,IAAI,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;AACnE,YAAI,UAAU,QAAQ,OAAO,CAAA,SAAQ,SAAS,QAAQ,IAAI,IAAI,KAAK,CAAC,GAAG,UAAU,SAAS,IAAI,CAAC;AAC/F,YAAI,aAAa,QAAQ,OAAO,CAAA,SAAQ,YAAY,QAAQ,IAAI,IAAI,KAAK,GAAG,UAAU,SAAS,IAAI,CAAC;AACpG,aAAK,mBAAmB,IAAI,SAAS,YAAY,YAAY,MAAM,MAAM,QAAQ;MACnF,CAAC;IACH;IAEA,WAAW,IAAI,MAAM,MAAM,MAAK;AAC9B,UAAG,GAAG,aAAa,IAAI,GAAE;AACvB,YAAG,SAAS,QAAU;AAEpB,cAAG,GAAG,aAAa,IAAI,MAAM,MAAK;AAChC,iBAAK,iBAAiB,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;UAC9C,OAAO;AACL,iBAAK,iBAAiB,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;UAC9C;QACF,OAAO;AAEL,eAAK,iBAAiB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;QACtC;MACF,OAAO;AACL,aAAK,iBAAiB,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;MAC9C;IACF;IAEA,mBAAmB,IAAI,MAAM,SAAS,YAAY,MAAM,MAAM,UAAS;AACrE,aAAO,QAAQ;AACf,UAAI,CAAC,eAAe,iBAAiB,aAAa,IAAI,cAAc,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;AAC/E,UAAG,cAAc,SAAS,GAAE;AAC1B,YAAI,UAAU,MAAM;AAClB,eAAK,mBAAmB,IAAI,iBAAiB,CAAC,EAAE,OAAO,aAAa,EAAE,OAAO,aAAa,CAAC;AAC3F,iBAAO,sBAAsB,MAAM;AACjC,iBAAK,mBAAmB,IAAI,eAAe,CAAC,CAAC;AAC7C,mBAAO,sBAAsB,MAAM,KAAK,mBAAmB,IAAI,eAAe,eAAe,CAAC;UAChG,CAAC;QACH;AACA,YAAI,SAAS,MAAM,KAAK,mBAAmB,IAAI,KAAK,OAAO,aAAa,GAAG,QAAQ,OAAO,aAAa,EAAE,OAAO,eAAe,CAAC;AAChI,YAAG,aAAa,OAAM;AACpB,kBAAQ;AACR,qBAAW,QAAQ,IAAI;QACzB,OAAO;AACL,eAAK,WAAW,MAAM,SAAS,MAAM;QACvC;AACA;MACF;AAEA,aAAO,sBAAsB,MAAM;AACjC,YAAI,CAAC,UAAU,WAAW,IAAI,YAAI,UAAU,IAAI,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;AACnE,YAAI,WAAW,KAAK,OAAO,CAAA,SAAQ,SAAS,QAAQ,IAAI,IAAI,KAAK,CAAC,GAAG,UAAU,SAAS,IAAI,CAAC;AAC7F,YAAI,cAAc,QAAQ,OAAO,CAAA,SAAQ,YAAY,QAAQ,IAAI,IAAI,KAAK,GAAG,UAAU,SAAS,IAAI,CAAC;AACrG,YAAI,UAAU,SAAS,OAAO,CAAA,SAAQ,QAAQ,QAAQ,IAAI,IAAI,CAAC,EAAE,OAAO,QAAQ;AAChF,YAAI,aAAa,YAAY,OAAO,CAAA,SAAQ,KAAK,QAAQ,IAAI,IAAI,CAAC,EAAE,OAAO,WAAW;AAEtF,oBAAI,UAAU,IAAI,WAAW,CAAA,cAAa;AACxC,oBAAU,UAAU,OAAO,GAAG,UAAU;AACxC,oBAAU,UAAU,IAAI,GAAG,OAAO;AAClC,iBAAO,CAAC,SAAS,UAAU;QAC7B,CAAC;MACH,CAAC;IACH;IAEA,iBAAiB,IAAI,MAAM,SAAQ;AACjC,UAAI,CAAC,UAAU,WAAW,IAAI,YAAI,UAAU,IAAI,SAAS,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;AAEjE,UAAI,eAAe,KAAK,IAAI,CAAC,CAAC,MAAM,IAAI,MAAM,IAAI,EAAE,OAAO,OAAO;AAClE,UAAI,UAAU,SAAS,OAAO,CAAC,CAAC,MAAM,IAAI,MAAM,CAAC,aAAa,SAAS,IAAI,CAAC,EAAE,OAAO,IAAI;AACzF,UAAI,aAAa,YAAY,OAAO,CAAC,SAAS,CAAC,aAAa,SAAS,IAAI,CAAC,EAAE,OAAO,OAAO;AAE1F,kBAAI,UAAU,IAAI,SAAS,CAAA,cAAa;AACtC,mBAAW,QAAQ,CAAA,SAAQ,UAAU,gBAAgB,IAAI,CAAC;AAC1D,gBAAQ,QAAQ,CAAC,CAAC,MAAM,GAAG,MAAM,UAAU,aAAa,MAAM,GAAG,CAAC;AAClE,eAAO,CAAC,SAAS,UAAU;MAC7B,CAAC;IACH;IAEA,cAAc,IAAI,SAAQ;AAAE,aAAO,QAAQ,MAAM,CAAA,SAAQ,GAAG,UAAU,SAAS,IAAI,CAAC;IAAE;IAEtF,aAAa,IAAI,YAAW;AAC1B,aAAO,CAAC,KAAK,UAAU,EAAE,KAAK,KAAK,cAAc,IAAI,UAAU;IACjE;IAEA,YAAYA,aAAY,UAAU,EAAC,GAAE,GAAE;AACrC,UAAI,eAAe,MAAM;AACvB,YAAG,OAAO,OAAQ,UAAS;AACzB,iBAAO,SAAS,iBAAiB,EAAE;QACrC,WAAU,GAAG,SAAQ;AACnB,cAAI,OAAO,SAAS,QAAQ,GAAG,OAAO;AACtC,iBAAO,OAAO,CAAC,IAAI,IAAI,CAAC;QAC1B,WAAU,GAAG,OAAM;AACjB,iBAAO,SAAS,iBAAiB,GAAG,KAAK;QAC3C;MACF;AACA,aAAO,KAAKA,YAAW,mBAAmB,UAAU,IAAI,YAAY,IAAI,CAAC,QAAQ;IACnF;IAEA,eAAe,IAAG;AAChB,aAAO,EAAC,IAAI,aAAa,IAAI,aAAY,EAAE,GAAG,QAAQ,YAAY,CAAC,KAAK;IAC1E;IAEA,kBAAkB,KAAI;AACpB,UAAG,CAAC,KAAI;AAAE,eAAO;MAAK;AAEtB,UAAI,CAAC,OAAO,QAAQ,IAAI,IAAI,MAAM,QAAQ,GAAG,IAAI,MAAM,CAAC,IAAI,MAAM,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC;AAC9E,cAAQ,MAAM,QAAQ,KAAK,IAAI,QAAQ,MAAM,MAAM,GAAG;AACtD,eAAS,MAAM,QAAQ,MAAM,IAAI,SAAS,OAAO,MAAM,GAAG;AAC1D,aAAO,MAAM,QAAQ,IAAI,IAAI,OAAO,KAAK,MAAM,GAAG;AAClD,aAAO,CAAC,OAAO,QAAQ,IAAI;IAC7B;EACF;AAEA,MAAO,aAAQ;AC7Wf,MAAM,UAAU;AAEhB,MAAI,aAAa;AACjB,MAAqB,WAArB,MAA8B;IAC5B,OAAO,SAAQ;AAAE,aAAO;IAAa;IACrC,OAAO,UAAU,IAAG;AAAE,aAAO,YAAI,QAAQ,IAAI,OAAO;IAAE;IAEtD,YAAY,MAAM,IAAI,WAAU;AAC9B,WAAK,KAAK;AACV,WAAK,aAAa,IAAI;AACtB,WAAK,cAAc;AACnB,WAAK,cAAc,oBAAI,IAAI;AAC3B,WAAK,mBAAmB;AACxB,kBAAI,WAAW,KAAK,IAAI,SAAS,KAAK,YAAY,OAAO,CAAC;AAC1D,eAAQ,OAAO,KAAK,aAAY;AAAE,aAAK,GAAG,IAAI,KAAK,YAAY,GAAG;MAAE;IACtE;IAEA,aAAa,MAAK;AAChB,UAAG,MAAK;AACN,aAAK,SAAS,MAAM;AACpB,aAAK,aAAa,KAAK;MACzB,OAAO;AACL,aAAK,SAAS,MAAM;AAClB,gBAAM,IAAI,MAAM,yCAAyC,KAAK,GAAG,WAAW;QAC9E;AACA,aAAK,aAAa;MACpB;IACF;IAEA,YAAW;AAAE,WAAK,WAAW,KAAK,QAAQ;IAAE;IAC5C,YAAW;AAAE,WAAK,WAAW,KAAK,QAAQ;IAAE;IAC5C,iBAAgB;AAAE,WAAK,gBAAgB,KAAK,aAAa;IAAE;IAC3D,cAAa;AACX,WAAK,aAAa,KAAK,UAAU;AACjC,kBAAI,cAAc,KAAK,IAAI,OAAO;IACpC;IACA,gBAAe;AACb,UAAG,KAAK,kBAAiB;AACvB,aAAK,mBAAmB;AACxB,aAAK,eAAe,KAAK,YAAY;MACvC;IACF;IACA,iBAAgB;AACd,WAAK,mBAAmB;AACxB,WAAK,gBAAgB,KAAK,aAAa;IACzC;;;;;;;;IASA,KAAI;AACF,UAAI,OAAO;AAEX,aAAO;;;;;;QAML,KAAK,WAAU;AACb,eAAK,OAAO,EAAE,WAAW,OAAO,KAAK,IAAI,WAAW,MAAM;QAC5D;;;;;;;;;;;;QAaA,KAAK,IAAI,OAAO,CAAC,GAAE;AACjB,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,KAAK,QAAQ,OAAO,IAAI,KAAK,SAAS,KAAK,YAAY,KAAK,MAAM,KAAK,QAAQ;QACpF;;;;;;;;;;;QAYA,KAAK,IAAI,OAAO,CAAC,GAAE;AACjB,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,KAAK,QAAQ,OAAO,IAAI,MAAM,KAAK,YAAY,KAAK,MAAM,KAAK,QAAQ;QAC5E;;;;;;;;;;;;;;;;;;;;;;;;;;QA2BA,OAAO,IAAI,OAAO,CAAC,GAAE;AACnB,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,eAAK,KAAK,WAAG,kBAAkB,KAAK,EAAE;AACtC,eAAK,MAAM,WAAG,kBAAkB,KAAK,GAAG;AACxC,qBAAG,OAAO,QAAQ,OAAO,IAAI,KAAK,SAAS,KAAK,IAAI,KAAK,KAAK,KAAK,MAAM,KAAK,QAAQ;QACxF;;;;;;;;;;;;;;;;;;QAmBA,SAAS,IAAI,OAAO,OAAO,CAAC,GAAE;AAC5B,kBAAQ,MAAM,QAAQ,KAAK,IAAI,QAAQ,MAAM,MAAM,GAAG;AACtD,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,mBAAmB,IAAI,OAAO,CAAC,GAAG,KAAK,YAAY,KAAK,MAAM,OAAO,KAAK,QAAQ;QACvF;;;;;;;;;;;;;;;;;;QAmBA,YAAY,IAAI,OAAO,OAAO,CAAC,GAAE;AAC/B,eAAK,aAAa,WAAG,kBAAkB,KAAK,UAAU;AACtD,kBAAQ,MAAM,QAAQ,KAAK,IAAI,QAAQ,MAAM,MAAM,GAAG;AACtD,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,mBAAmB,IAAI,CAAC,GAAG,OAAO,KAAK,YAAY,KAAK,MAAM,OAAO,KAAK,QAAQ;QACvF;;;;;;;;;;;;;;;;;;QAmBA,YAAY,IAAI,OAAO,OAAO,CAAC,GAAE;AAC/B,eAAK,aAAa,WAAG,kBAAkB,KAAK,UAAU;AACtD,kBAAQ,MAAM,QAAQ,KAAK,IAAI,QAAQ,MAAM,MAAM,GAAG;AACtD,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,cAAc,IAAI,OAAO,KAAK,YAAY,KAAK,MAAM,OAAO,KAAK,QAAQ;QAC9E;;;;;;;;;;;;;;;;;QAkBA,WAAW,IAAI,YAAY,OAAO,CAAC,GAAE;AACnC,cAAI,QAAQ,KAAK,OAAO,EAAE,WAAW,MAAM,EAAE;AAC7C,qBAAG,mBAAmB,IAAI,CAAC,GAAG,CAAC,GAAG,WAAG,kBAAkB,UAAU,GAAG,KAAK,MAAM,OAAO,KAAK,QAAQ;QACrG;;;;;;;;QASA,aAAa,IAAI,MAAM,KAAI;AAAE,qBAAG,iBAAiB,IAAI,CAAC,CAAC,MAAM,GAAG,CAAC,GAAG,CAAC,CAAC;QAAE;;;;;;;QAQxE,gBAAgB,IAAI,MAAK;AAAE,qBAAG,iBAAiB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;QAAE;;;;;;;;;QAU/D,gBAAgB,IAAI,MAAM,MAAM,MAAK;AAAE,qBAAG,WAAW,IAAI,MAAM,MAAM,IAAI;QAAE;MAC7E;IACF;IAEA,UAAU,OAAO,UAAU,CAAC,GAAG,SAAQ;AACrC,UAAG,YAAY,QAAU;AACvB,eAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,cAAI;AACF,kBAAM,MAAM,KAAK,OAAO,EAAE,cAAc,KAAK,IAAI,MAAM,OAAO,SAAS,CAAC,OAAO,SAAS,QAAQ,KAAK,CAAC;AACtG,gBAAG,QAAQ,OAAM;AACf,qBAAO,IAAI,MAAM,mDAAmD,CAAC;YACvE;UACF,SAAS,OAAT;AACE,mBAAO,KAAK;UACd;QACF,CAAC;MACH;AACA,aAAO,KAAK,OAAO,EAAE,cAAc,KAAK,IAAI,MAAM,OAAO,SAAS,OAAO;IAC3E;IAEA,YAAY,WAAW,OAAO,UAAU,CAAC,GAAG,SAAQ;AAClD,UAAG,YAAY,QAAU;AACvB,eAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,cAAI;AACF,iBAAK,OAAO,EAAE,cAAc,WAAW,CAAC,MAAM,cAAc;AAC1D,oBAAM,MAAM,KAAK,cAAc,KAAK,IAAI,WAAW,OAAO,SAAS,CAAC,OAAO,SAAS,QAAQ,KAAK,CAAC;AAClG,kBAAG,QAAQ,OAAM;AACf,uBAAO,IAAI,MAAM,mDAAmD,CAAC;cACvE;YACF,CAAC;UACH,SAAS,OAAT;AACE,mBAAO,KAAK;UACd;QACF,CAAC;MACH;AACA,aAAO,KAAK,OAAO,EAAE,cAAc,WAAW,CAAC,MAAM,cAAc;AACjE,eAAO,KAAK,cAAc,KAAK,IAAI,WAAW,OAAO,SAAS,OAAO;MACvE,CAAC;IACH;IAEA,YAAY,OAAO,UAAS;AAC1B,UAAI,cAAc,CAAC,aAAa,WAAW,SAAS,QAAQ,SAAS,YAAY,MAAM;AACvF,aAAO,iBAAiB,OAAO,SAAS,WAAW;AACnD,WAAK,YAAY,IAAI,WAAW;AAChC,aAAO;IACT;IAEA,kBAAkB,aAAY;AAC5B,UAAI,QAAQ,YAAY,MAAM,IAAI;AAClC,aAAO,oBAAoB,OAAO,SAAS,WAAW;AACtD,WAAK,YAAY,OAAO,WAAW;IACrC;IAEA,OAAO,MAAM,OAAM;AACjB,aAAO,KAAK,OAAO,EAAE,gBAAgB,MAAM,MAAM,KAAK;IACxD;IAEA,SAAS,WAAW,MAAM,OAAM;AAC9B,aAAO,KAAK,OAAO,EAAE,cAAc,WAAW,CAAC,MAAM,cAAc;AACjE,aAAK,gBAAgB,WAAW,MAAM,KAAK;MAC7C,CAAC;IACH;IAEA,cAAa;AACX,WAAK,YAAY,QAAQ,CAAA,gBAAe,KAAK,kBAAkB,WAAW,CAAC;IAC7E;EACF;AC/PO,MAAI,qBAAqB,CAAC,KAAK,WAAW;AAC/C,QAAI,UAAU,IAAI,SAAS,IAAI;AAE/B,QAAI,UAAU,UAAU,IAAI,MAAM,GAAG,EAAE,IAAI;AAE3C,cAAU,QAAQ,QAAQ,oBAAoB,GAAG,YAAY;AAE7D,QAAG,SAAQ;AAAE,iBAAW;IAAK;AAC7B,WAAO;EACT;AAEA,MAAI,gBAAgB,CAAC,MAAM,MAAM,YAAY,CAAC,MAAM;AAClD,UAAM,EAAC,UAAS,IAAI;AAIpB,QAAI;AACJ,QAAG,aAAa,UAAU,MAAK;AAC7B,YAAM,QAAQ,SAAS,cAAc,OAAO;AAC5C,YAAM,OAAO;AAGb,YAAM,SAAS,UAAU,aAAa,MAAM;AAC5C,UAAG,QAAO;AACR,cAAM,aAAa,QAAQ,MAAM;MACnC;AACA,YAAM,OAAO,UAAU;AACvB,YAAM,QAAQ,UAAU;AACxB,gBAAU,cAAc,aAAa,OAAO,SAAS;AACrD,wBAAkB;IACpB;AAEA,UAAM,WAAW,IAAI,SAAS,IAAI;AAClC,UAAM,WAAW,CAAC;AAElB,aAAS,QAAQ,CAAC,KAAK,KAAK,WAAW;AACrC,UAAG,eAAe,MAAK;AAAE,iBAAS,KAAK,GAAG;MAAE;IAC9C,CAAC;AAGD,aAAS,QAAQ,CAAA,QAAO,SAAS,OAAO,GAAG,CAAC;AAE5C,UAAM,SAAS,IAAI,gBAAgB;AAEnC,UAAM,EAAC,cAAc,iBAAgB,IAAI,MAAM,KAAK,KAAK,QAAQ,EAAE,OAAO,CAAC,KAAK,UAAU;AACxF,YAAM,EAAC,cAAAW,eAAc,kBAAAC,kBAAgB,IAAI;AACzC,YAAM,MAAM,MAAM;AAClB,UAAG,CAAC,KAAI;AAAE,eAAO;MAAI;AAErB,UAAGD,cAAa,GAAG,MAAM,QAAU;AAAEA,sBAAa,GAAG,IAAI;MAAK;AAC9D,UAAGC,kBAAiB,GAAG,MAAM,QAAU;AAAEA,0BAAiB,GAAG,IAAI;MAAK;AAEtE,YAAM,SAAS,YAAI,QAAQ,OAAO,eAAe,KAAK,YAAI,QAAQ,OAAO,iBAAiB;AAC1F,YAAM,WAAW,MAAM,SAAS;AAChCD,oBAAa,GAAG,IAAIA,cAAa,GAAG,KAAK,CAAC;AAC1CC,wBAAiB,GAAG,IAAIA,kBAAiB,GAAG,KAAK;AAEjD,aAAO;IACT,GAAG,EAAC,cAAc,CAAC,GAAG,kBAAkB,CAAC,EAAC,CAAC;AAE3C,aAAQ,CAAC,KAAK,GAAG,KAAK,SAAS,QAAQ,GAAE;AACvC,UAAG,UAAU,WAAW,KAAK,UAAU,QAAQ,GAAG,KAAK,GAAE;AACvD,YAAI,WAAW,aAAa,GAAG;AAC/B,YAAI,SAAS,iBAAiB,GAAG;AACjC,YAAG,YAAY,EAAE,aAAa,UAAU,QAAQ,QAAQ,CAAC,QAAO;AAC9D,iBAAO,OAAO,mBAAmB,KAAK,UAAU,GAAG,EAAE;QACvD;AACA,eAAO,OAAO,KAAK,GAAG;MACxB;IACF;AAIA,QAAG,aAAa,iBAAgB;AAC9B,gBAAU,cAAc,YAAY,eAAe;IACrD;AAEA,WAAO,OAAO,SAAS;EACzB;AAEA,MAAqB,OAArB,MAAqB,MAAK;IACxB,OAAO,YAAY,IAAG;AACpB,UAAI,aAAa,GAAG,QAAQ,iBAAiB;AAC7C,aAAO,aAAa,YAAI,QAAQ,YAAY,MAAM,IAAI;IACxD;IAEA,YAAY,IAAIZ,aAAY,YAAY,OAAO,aAAY;AACzD,WAAK,SAAS;AACd,WAAK,aAAaA;AAClB,WAAK,QAAQ;AACb,WAAK,SAAS;AACd,WAAK,OAAO,aAAa,WAAW,OAAO;AAC3C,WAAK,KAAK;AACV,kBAAI,WAAW,KAAK,IAAI,QAAQ,IAAI;AACpC,WAAK,KAAK,KAAK,GAAG;AAClB,WAAK,MAAM;AACX,WAAK,aAAa;AAClB,WAAK,aAAa;AAClB,WAAK,cAAc;AACnB,WAAK,oBAAoB;AACzB,WAAK,eAAe,CAAC;AACrB,WAAK,eAAe,oBAAI,IAAI;AAC5B,WAAK,WAAW;AAChB,WAAK,OAAO;AACZ,WAAK,YAAY,KAAK,SAAS,KAAK,OAAO,YAAY,IAAI;AAC3D,WAAK,eAAe;AACpB,WAAK,cAAc;AACnB,WAAK,YAAY;AACjB,WAAK,eAAe,SAAS,QAAO;AAAE,kBAAU,OAAO;MAAE;AACzD,WAAK,eAAe,WAAU;MAAE;AAChC,WAAK,iBAAiB,KAAK,SAAS,OAAO,CAAC;AAC5C,WAAK,YAAY,CAAC;AAClB,WAAK,cAAc,CAAC;AACpB,WAAK,WAAW,KAAK,SAAS,OAAO,CAAC;AACtC,WAAK,KAAK,SAAS,KAAK,EAAE,IAAI,CAAC;AAC/B,WAAK,mBAAmB,CAAC;AACzB,WAAK,UAAU,KAAK,WAAW,QAAQ,MAAM,KAAK,MAAM,MAAM;AAC5D,YAAI,MAAM,KAAK,QAAQ,KAAK,UAAU,KAAK,IAAI;AAC/C,eAAO;UACL,UAAU,KAAK,WAAW,MAAM;UAChC,KAAK,KAAK,WAAW,SAAY,OAAO;UACxC,QAAQ,KAAK,cAAc,WAAW;UACtC,SAAS,KAAK,WAAW;UACzB,QAAQ,KAAK,UAAU;UACvB,OAAO,KAAK;UACZ,QAAQ,KAAK,GAAG,aAAa,UAAU;QACzC;MACF,CAAC;IACH;IAEA,QAAQ,MAAK;AAAE,WAAK,OAAO;IAAK;IAEhC,YAAY,MAAK;AACf,WAAK,WAAW;AAChB,WAAK,OAAO;IACd;IAEA,SAAQ;AAAE,aAAO,KAAK,GAAG,aAAa,QAAQ;IAAE;IAEhD,cAAc,aAAY;AACxB,UAAI,SAAS,KAAK,WAAW,OAAO,KAAK,EAAE;AAC3C,UAAI,WACF,YAAI,IAAI,UAAU,IAAI,KAAK,QAAQ,gBAAgB,IAAI,EACpD,IAAI,CAAA,SAAQ,KAAK,OAAO,KAAK,IAAI,EAAE,OAAO,CAAA,QAAO,OAAQ,QAAS,QAAQ;AAE/E,UAAG,SAAS,SAAS,GAAE;AAAE,eAAO,eAAe,IAAI;MAAS;AAC5D,aAAO,SAAS,IAAI,KAAK;AACzB,aAAO,iBAAiB,IAAI,KAAK;AACjC,aAAO,eAAe,IAAI;AAC1B,WAAK;AAEL,aAAO;IACT;IAEA,cAAa;AAAE,aAAO,KAAK,QAAQ,QAAQ;IAAE;IAE7C,aAAY;AAAE,aAAO,KAAK,GAAG,aAAa,WAAW;IAAE;IAEvD,YAAW;AACT,UAAI,MAAM,KAAK,GAAG,aAAa,UAAU;AACzC,aAAO,QAAQ,KAAK,OAAO;IAC7B;IAEA,QAAQ,WAAW,WAAW;IAAE,GAAE;AAChC,WAAK,mBAAmB;AACxB,WAAK,YAAY;AACjB,aAAO,KAAK,KAAK,SAAS,KAAK,EAAE;AACjC,UAAG,KAAK,QAAO;AAAE,eAAO,KAAK,KAAK,SAAS,KAAK,OAAO,EAAE,EAAE,KAAK,EAAE;MAAE;AACpE,mBAAa,KAAK,WAAW;AAC7B,UAAI,aAAa,MAAM;AACrB,iBAAS;AACT,iBAAQ,MAAM,KAAK,WAAU;AAC3B,eAAK,YAAY,KAAK,UAAU,EAAE,CAAC;QACrC;MACF;AAEA,kBAAI,sBAAsB,KAAK,EAAE;AAEjC,WAAK,IAAI,aAAa,MAAM,CAAC,4CAA4C,CAAC;AAC1E,WAAK,QAAQ,MAAM,EAChB,QAAQ,MAAM,UAAU,EACxB,QAAQ,SAAS,UAAU,EAC3B,QAAQ,WAAW,UAAU;IAClC;IAEA,uBAAuB,SAAQ;AAC7B,WAAK,GAAG,UAAU;QAChB;QACA;QACA;QACA;QACA;MACF;AACA,WAAK,GAAG,UAAU,IAAI,GAAG,OAAO;IAClC;IAEA,WAAW,SAAQ;AACjB,mBAAa,KAAK,WAAW;AAC7B,UAAG,SAAQ;AACT,aAAK,cAAc,WAAW,MAAM,KAAK,WAAW,GAAG,OAAO;MAChE,OAAO;AACL,iBAAQ,MAAM,KAAK,WAAU;AAAE,eAAK,UAAU,EAAE,EAAE,eAAe;QAAE;AACnE,aAAK,oBAAoB,iBAAiB;MAC5C;IACF;IAEA,QAAQ,SAAQ;AACd,kBAAI,IAAI,KAAK,IAAI,IAAI,YAAY,CAAA,OAAM,KAAK,WAAW,OAAO,IAAI,GAAG,aAAa,OAAO,CAAC,CAAC;IAC7F;IAEA,aAAY;AACV,mBAAa,KAAK,WAAW;AAC7B,mBAAa,KAAK,iBAAiB;AACnC,WAAK,oBAAoB,mBAAmB;AAC5C,WAAK,QAAQ,KAAK,QAAQ,WAAW,CAAC;IACxC;IAEA,qBAAoB;AAClB,eAAQ,MAAM,KAAK,WAAU;AAAE,aAAK,UAAU,EAAE,EAAE,cAAc;MAAE;IACpE;IAEA,IAAI,MAAM,aAAY;AACpB,WAAK,WAAW,IAAI,MAAM,MAAM,WAAW;IAC7C;IAEA,WAAW,MAAM,SAAS,SAAS,WAAU;IAAC,GAAE;AAC9C,WAAK,WAAW,WAAW,MAAM,SAAS,MAAM;IAClD;;;;;;;IAQA,cAAc,WAAW,UAAU,MAAM,UAAU,QAAO;AAIxD,UAAG,qBAAqB,eAAe,qBAAqB,YAAW;AACrE,eAAO,KAAK,WAAW,MAAM,WAAW,CAAA,SAAQ,SAAS,MAAM,SAAS,CAAC;MAC3E;AAEA,UAAG,MAAM,SAAS,GAAE;AAClB,YAAI,UAAU,YAAI,sBAAsB,UAAU,KAAK,IAAI,SAAS;AACpE,YAAG,QAAQ,WAAW,GAAE;AACtB,mBAAS,6CAA6C,WAAW;QACnE,OAAO;AACL,mBAAS,MAAM,SAAS,SAAS,CAAC;QACpC;MACF,OAAO;AACL,YAAI,UAAU,MAAM,KAAK,IAAI,iBAAiB,SAAS,CAAC;AACxD,YAAG,QAAQ,WAAW,GAAE;AAAE,mBAAS,mDAAmD,YAAY;QAAE;AACpG,gBAAQ,QAAQ,CAAA,WAAU,KAAK,WAAW,MAAM,QAAQ,CAAA,SAAQ,SAAS,MAAM,MAAM,CAAC,CAAC;MACzF;IACF;IAEA,UAAU,MAAM,SAAS,UAAS;AAChC,WAAK,IAAI,MAAM,MAAM,CAAC,IAAI,MAAM,OAAO,CAAC,CAAC;AACzC,UAAI,EAAC,MAAM,OAAO,QAAQ,MAAK,IAAI,SAAS,QAAQ,OAAO;AAC3D,eAAS,EAAC,MAAM,OAAO,OAAM,CAAC;AAC9B,UAAG,OAAO,UAAU,YAAY,QAAQ,SAAQ;AAAE,eAAO,sBAAsB,MAAM,YAAI,SAAS,KAAK,CAAC;MAAE;IAC5G;IAEA,OAAO,MAAK;AACV,UAAI,EAAC,UAAU,WAAW,iBAAgB,IAAI;AAC9C,UAAG,WAAU;AACX,YAAI,CAAC,KAAK,KAAK,IAAI;AACnB,aAAK,KAAK,YAAI,qBAAqB,KAAK,IAAI,KAAK,KAAK;MACxD;AACA,WAAK,aAAa;AAClB,WAAK,cAAc;AACnB,WAAK,QAAQ;AACb,UAAG,KAAK,SAAS,MAAK;AACpB,aAAK,mBAAmB,KAAK,oBAAoB;MACnD;AACA,UAAG,KAAK,OAAO,KAAK,OAAO,QAAQ,UAAU,MAAK;AAEhD,wBAAQ,UAAU,WAAW;UAC3B,MAAM;UACN,IAAI,KAAK;UACT,UAAU,KAAK,WAAW;QAC5B,CAAC;MACH;AAEA,UAAG,qBAAqB,KAAK,WAAW,QAAQ,GAAE;AAChD,gBAAQ,MAAM,uDAAuD,KAAK,WAAW,QAAQ,gBAAgB,uGAAuG;MACtN;AAEA,sBAAQ,UAAU,KAAK,WAAW,cAAc,OAAO,SAAS,UAAU,mBAAmB;AAC7F,WAAK,UAAU,SAAS,UAAU,CAAC,EAAC,MAAM,OAAM,MAAM;AACpD,aAAK,WAAW,IAAI,SAAS,KAAK,IAAI,IAAI;AAC1C,YAAI,CAAC,MAAM,OAAO,IAAI,KAAK,gBAAgB,MAAM,MAAM;AACvD,aAAK,gBAAgB;AACrB,aAAK;AACL,aAAK,eAAe;AAEpB,aAAK,kBAAkB,MAAM,MAAM;AACjC,eAAK,eAAe,MAAM,MAAM,SAAS,MAAM;QACjD,CAAC;MACH,CAAC;IACH;IAEA,kBAAiB;AACf,kBAAI,IAAI,UAAU,IAAI,gBAAgB,KAAK,OAAO,OAAO,CAAA,OAAM;AAC7D,WAAG,gBAAgB,eAAe;AAClC,WAAG,gBAAgB,WAAW;AAC9B,WAAG,gBAAgB,YAAY;MACjC,CAAC;IACH;IAEA,eAAe,EAAC,WAAU,GAAG,MAAM,SAAS,QAAO;AAGjD,UAAG,KAAK,YAAY,KAAM,KAAK,UAAU,CAAC,KAAK,OAAO,cAAc,GAAG;AACrE,eAAO,KAAK,eAAe,YAAY,MAAM,SAAS,MAAM;MAC9D;AAMA,UAAI,cAAc,YAAI,0BAA0B,MAAM,KAAK,EAAE,EAAE,OAAO,CAAA,SAAQ;AAC5E,YAAI,SAAS,KAAK,MAAM,KAAK,GAAG,cAAc,QAAQ,KAAK,MAAM;AACjE,YAAI,YAAY,UAAU,OAAO,aAAa,UAAU;AACxD,YAAG,WAAU;AAAE,eAAK,aAAa,YAAY,SAAS;QAAE;AAGxD,YAAG,QAAO;AAAE,iBAAO,aAAa,aAAa,KAAK,KAAK,EAAE;QAAE;AAC3D,eAAO,KAAK,UAAU,IAAI;MAC5B,CAAC;AAED,UAAG,YAAY,WAAW,GAAE;AAC1B,YAAG,KAAK,QAAO;AACb,eAAK,KAAK,eAAe,KAAK,CAAC,MAAM,MAAM,KAAK,eAAe,YAAY,MAAM,SAAS,MAAM,CAAC,CAAC;AAClG,eAAK,OAAO,QAAQ,IAAI;QAC1B,OAAO;AACL,eAAK,wBAAwB;AAC7B,eAAK,eAAe,YAAY,MAAM,SAAS,MAAM;QACvD;MACF,OAAO;AACL,aAAK,KAAK,eAAe,KAAK,CAAC,MAAM,MAAM,KAAK,eAAe,YAAY,MAAM,SAAS,MAAM,CAAC,CAAC;MACpG;IACF;IAEA,kBAAiB;AACf,WAAK,KAAK,YAAI,KAAK,KAAK,EAAE;AAC1B,WAAK,GAAG,aAAa,aAAa,KAAK,KAAK,EAAE;IAChD;;;;;IAMA,eAAe,SAAS,KAAK,IAAG;AAC9B,UAAI,iBAAiB,KAAK,QAAQ,gBAAgB;AAClD,UAAI,oBAAoB,KAAK,QAAQ,mBAAmB;AACxD,kBAAI,IAAI,QAAQ,IAAI,qBAAqB,sBAAsB,CAAA,WAAU;AACvE,YAAG,KAAK,YAAY,MAAM,GAAE;AAC1B,sBAAI,qBAAqB,QAAQ,QAAQ,gBAAgB,iBAAiB;AAC1E,eAAK,gBAAgB,MAAM;QAC7B;MACF,CAAC;AACD,kBAAI,IAAI,QAAQ,IAAI,KAAK,QAAQ,QAAQ,iBAAiB,aAAa,CAAA,WAAU;AAC/E,YAAG,KAAK,YAAY,MAAM,GAAE;AAC1B,eAAK,gBAAgB,MAAM;QAC7B;MACF,CAAC;AACD,kBAAI,IAAI,QAAQ,IAAI,KAAK,QAAQ,WAAW,MAAM,CAAA,OAAM;AACtD,YAAG,KAAK,YAAY,EAAE,GAAE;AACtB,eAAK,aAAa,EAAE;QACtB;MACF,CAAC;IACH;IAEA,eAAe,YAAY,MAAM,SAAS,QAAO;AAC/C,WAAK,gBAAgB;AACrB,UAAI,QAAQ,IAAI,SAAS,MAAM,KAAK,IAAI,KAAK,IAAI,MAAM,SAAS,IAAI;AACpE,YAAM,8BAA8B;AACpC,WAAK,aAAa,OAAO,OAAO,IAAI;AACpC,WAAK,gBAAgB;AACrB,WAAK,eAAe;AAEpB,WAAK,cAAc;AACnB,WAAK,WAAW,eAAe,MAAM;AACrC,WAAK,oBAAoB;AAEzB,UAAG,YAAW;AACZ,YAAI,EAAC,MAAM,GAAE,IAAI;AACjB,aAAK,WAAW,aAAa,IAAI,IAAI;MACvC;AACA,WAAK,WAAW;AAChB,UAAG,KAAK,YAAY,GAAE;AAAE,aAAK,mBAAmB;MAAE;AAClD,WAAK,aAAa;IACpB;IAEA,wBAAwB,QAAQ,MAAK;AACnC,WAAK,WAAW,WAAW,qBAAqB,CAAC,QAAQ,IAAI,CAAC;AAC9D,UAAI,OAAO,KAAK,QAAQ,MAAM;AAC9B,UAAI,YAAY,QAAQ,YAAI,UAAU,QAAQ,KAAK,QAAQ,UAAU,CAAC;AACtE,UAAG,QAAQ,CAAC,OAAO,YAAY,IAAI,KAAK,EAAE,aAAa,WAAW,OAAO,SAAS,KAAK,OAAO,IAAG;AAC/F,aAAK,eAAe;AACpB,eAAO;MACT;IACF;IAEA,aAAa,IAAG;AACd,UAAI,aAAa,GAAG,aAAa,KAAK,QAAQ,WAAW,CAAC;AAC1D,UAAI,iBAAiB,cAAc,YAAI,QAAQ,IAAI,SAAS;AAC5D,UAAG,cAAc,CAAC,gBAAe;AAC/B,aAAK,WAAW,OAAO,IAAI,UAAU;AACrC,oBAAI,WAAW,IAAI,WAAW,IAAI;MACpC;IACF;IAEA,gBAAgB,IAAG;AACjB,UAAI,UAAU,KAAK,QAAQ,EAAE;AAC7B,UAAG,SAAQ;AAAE,gBAAQ,UAAU;MAAE;IACnC;IAEA,aAAa,OAAO,WAAW,cAAc,OAAM;AACjD,UAAI,aAAa,CAAC;AAClB,UAAI,mBAAmB;AACvB,UAAI,iBAAiB,oBAAI,IAAI;AAE7B,WAAK,WAAW,WAAW,gBAAgB,CAAC,MAAM,eAAe,CAAC;AAElE,YAAM,MAAM,SAAS,CAAA,OAAM;AACzB,aAAK,WAAW,WAAW,eAAe,CAAC,EAAE,CAAC;AAC9C,YAAI,iBAAiB,KAAK,QAAQ,gBAAgB;AAClD,YAAI,oBAAoB,KAAK,QAAQ,mBAAmB;AACxD,oBAAI,qBAAqB,IAAI,IAAI,gBAAgB,iBAAiB;AAClE,aAAK,gBAAgB,EAAE;AACvB,YAAG,GAAG,cAAa;AAAE,eAAK,aAAa,EAAE;QAAE;MAC7C,CAAC;AAED,YAAM,MAAM,iBAAiB,CAAA,OAAM;AACjC,YAAG,YAAI,YAAY,EAAE,GAAE;AACrB,eAAK,WAAW,cAAc;QAChC,OAAO;AACL,6BAAmB;QACrB;MACF,CAAC;AAED,YAAM,OAAO,WAAW,CAAC,QAAQ,SAAS;AACxC,YAAI,OAAO,KAAK,wBAAwB,QAAQ,IAAI;AACpD,YAAG,MAAK;AAAE,yBAAe,IAAI,OAAO,EAAE;QAAE;MAC1C,CAAC;AAED,YAAM,MAAM,WAAW,CAAA,OAAM;AAC3B,YAAG,eAAe,IAAI,GAAG,EAAE,GAAE;AAAE,eAAK,QAAQ,EAAE,EAAE,UAAU;QAAE;MAC9D,CAAC;AAED,YAAM,MAAM,aAAa,CAAC,OAAO;AAC/B,YAAG,GAAG,aAAa,KAAK,cAAa;AAAE,qBAAW,KAAK,EAAE;QAAE;MAC7D,CAAC;AAED,YAAM,MAAM,wBAAwB,CAAA,QAAO,KAAK,qBAAqB,KAAK,SAAS,CAAC;AACpF,YAAM,QAAQ,WAAW;AACzB,WAAK,qBAAqB,YAAY,SAAS;AAE/C,WAAK,WAAW,WAAW,cAAc,CAAC,MAAM,eAAe,CAAC;AAChE,aAAO;IACT;IAEA,qBAAqB,UAAU,WAAU;AACvC,UAAI,gBAAgB,CAAC;AACrB,eAAS,QAAQ,CAAA,WAAU;AACzB,YAAI,aAAa,YAAI,IAAI,QAAQ,IAAI,gBAAgB;AACrD,YAAI,QAAQ,YAAI,IAAI,QAAQ,IAAI,KAAK,QAAQ,QAAQ,qBAAqB;AAC1E,mBAAW,OAAO,MAAM,EAAE,QAAQ,CAAA,OAAM;AACtC,cAAI,MAAM,KAAK,YAAY,EAAE;AAC7B,cAAG,MAAM,GAAG,KAAK,cAAc,QAAQ,GAAG,MAAM,IAAG;AAAE,0BAAc,KAAK,GAAG;UAAE;QAC/E,CAAC;AACD,cAAM,OAAO,MAAM,EAAE,QAAQ,CAAA,WAAU;AACrC,cAAI,OAAO,KAAK,QAAQ,MAAM;AAC9B,kBAAQ,KAAK,YAAY,IAAI;QAC/B,CAAC;MACH,CAAC;AAID,UAAG,WAAU;AACX,aAAK,6BAA6B,aAAa;MACjD;IACF;IAEA,kBAAiB;AACf,kBAAI,gBAAgB,KAAK,IAAI,KAAK,EAAE,EAAE,QAAQ,CAAA,OAAM,KAAK,UAAU,EAAE,CAAC;IACxE;IAEA,kBAAkB,MAAM,UAAS;AAC/B,YAAM,YAAY,KAAK,QAAQ,QAAQ;AACvC,YAAM,WAAW,KAAK,KAAK;AAQ3B,UAAI,WAAW,SAAS,cAAc,UAAU;AAChD,eAAS,YAAY;AAGrB,YAAM,SAAS,SAAS,QAAQ;AAChC,aAAO,KAAK,KAAK;AACjB,aAAO,aAAa,aAAa,KAAK,KAAK,EAAE;AAC7C,aAAO,aAAa,aAAa,KAAK,WAAW,CAAC;AAClD,aAAO,aAAa,YAAY,KAAK,UAAU,CAAC;AAChD,aAAO,aAAa,eAAe,KAAK,SAAS,KAAK,OAAO,KAAK,IAAI;AAKtE,YAAM;;;QAGJ,YAAI,IAAI,SAAS,SAAS,MAAM,EAE7B,OAAO,CAAA,YAAW,QAAQ,MAAM,SAAS,QAAQ,EAAE,CAAC,EAEpD,OAAO,CAAA,YAAW,CAAC,KAAK,aAAa,IAAI,QAAQ,EAAE,CAAC,EAEpD,OAAO,CAAA,YAAW,SAAS,QAAQ,EAAE,EAAE,aAAa,SAAS,MAAM,QAAQ,aAAa,SAAS,CAAC,EAClG,IAAI,CAAA,YAAW;AACd,iBAAO,CAAC,SAAS,QAAQ,EAAE,GAAG,OAAO;QACvC,CAAC;;AAEL,UAAG,eAAe,WAAW,GAAE;AAC7B,eAAO,SAAS;MAClB;AAEA,qBAAe,QAAQ,CAAC,CAAC,SAAS,OAAO,GAAG,MAAM;AAChD,aAAK,aAAa,IAAI,QAAQ,EAAE;AAKhC,aAAK,iBAAiB,SAAS,SAAS,SAAS,QAAQ,mBAAmB,MAAM;AAChF,eAAK,aAAa,OAAO,QAAQ,EAAE;AAEnC,cAAG,MAAM,eAAe,SAAS,GAAE;AACjC,qBAAS;UACX;QACF,CAAC;MACH,CAAC;IACH;IAEA,aAAa,IAAG;AAAE,aAAO,KAAK,KAAK,SAAS,KAAK,EAAE,EAAE,EAAE;IAAE;IAEzD,kBAAkB,IAAG;;AACnB,UAAG,GAAG,OAAO,KAAK,IAAG;AACnB,eAAO;MACT,OAAO;AACL,gBAAO,UAAK,SAAS,GAAG,aAAa,aAAa,CAAC,MAA5C,mBAAgD,GAAG;MAC5D;IACF;IAEA,kBAAkB,IAAG;AACnB,eAAQ,YAAY,KAAK,KAAK,UAAS;AACrC,iBAAQ,WAAW,KAAK,KAAK,SAAS,QAAQ,GAAE;AAC9C,cAAG,YAAY,IAAG;AAAE,mBAAO,KAAK,KAAK,SAAS,QAAQ,EAAE,OAAO,EAAE,QAAQ;UAAE;QAC7E;MACF;IACF;IAEA,UAAU,IAAG;AACX,UAAI,QAAQ,KAAK,aAAa,GAAG,EAAE;AACnC,UAAG,CAAC,OAAM;AACR,YAAI,OAAO,IAAI,MAAK,IAAI,KAAK,YAAY,IAAI;AAC7C,aAAK,KAAK,SAAS,KAAK,EAAE,EAAE,KAAK,EAAE,IAAI;AACvC,aAAK,KAAK;AACV,aAAK;AACL,eAAO;MACT;IACF;IAEA,gBAAe;AAAE,aAAO,KAAK;IAAY;IAEzC,QAAQ,QAAO;AACb,WAAK;AAEL,UAAG,KAAK,eAAe,GAAE;AACvB,YAAG,KAAK,QAAO;AACb,eAAK,OAAO,QAAQ,IAAI;QAC1B,OAAO;AACL,eAAK,wBAAwB;QAC/B;MACF;IACF;IAEA,0BAAyB;AAGvB,WAAK,aAAa,MAAM;AAExB,WAAK,mBAAmB,CAAC;AACzB,WAAK,aAAa,MAAM;AACtB,aAAK,eAAe,QAAQ,CAAC,CAAC,MAAM,EAAE,MAAM;AAC1C,cAAG,CAAC,KAAK,YAAY,GAAE;AAAE,eAAG;UAAE;QAChC,CAAC;AACD,aAAK,iBAAiB,CAAC;MACzB,CAAC;IACH;IAEA,OAAO,MAAM,QAAO;AAClB,UAAG,KAAK,cAAc,KAAM,KAAK,WAAW,eAAe,KAAK,KAAK,KAAK,OAAO,GAAG;AAClF,eAAO,KAAK,aAAa,KAAK,EAAC,MAAM,OAAM,CAAC;MAC9C;AAEA,WAAK,SAAS,UAAU,IAAI;AAC5B,UAAI,mBAAmB;AAKvB,UAAG,KAAK,SAAS,oBAAoB,IAAI,GAAE;AACzC,aAAK,WAAW,KAAK,4BAA4B,MAAM;AACrD,cAAI,aAAa,YAAI,uBAAuB,KAAK,IAAI,KAAK,SAAS,cAAc,IAAI,CAAC;AACtF,qBAAW,QAAQ,CAAA,cAAa;AAC9B,gBAAG,KAAK,eAAe,KAAK,SAAS,aAAa,MAAM,SAAS,GAAG,SAAS,GAAE;AAAE,iCAAmB;YAAK;UAC3G,CAAC;QACH,CAAC;MACH,WAAU,CAAC,QAAQ,IAAI,GAAE;AACvB,aAAK,WAAW,KAAK,uBAAuB,MAAM;AAChD,cAAI,CAAC,MAAM,OAAO,IAAI,KAAK,gBAAgB,MAAM,QAAQ;AACzD,cAAI,QAAQ,IAAI,SAAS,MAAM,KAAK,IAAI,KAAK,IAAI,MAAM,SAAS,IAAI;AACpE,6BAAmB,KAAK,aAAa,OAAO,IAAI;QAClD,CAAC;MACH;AAEA,WAAK,WAAW,eAAe,MAAM;AACrC,UAAG,kBAAiB;AAAE,aAAK,gBAAgB;MAAE;IAC/C;IAEA,gBAAgB,MAAM,MAAK;AACzB,aAAO,KAAK,WAAW,KAAK,kBAAkB,SAAS,MAAM;AAC3D,YAAI,MAAM,KAAK,GAAG;AAGlB,YAAI,OAAO,OAAO,KAAK,SAAS,cAAc,IAAI,IAAI;AACtD,YAAI,CAAC,MAAM,OAAO,IAAI,KAAK,SAAS,SAAS,IAAI;AACjD,eAAO,CAAC,IAAI,OAAO,SAAS,QAAQ,OAAO;MAC7C,CAAC;IACH;IAEA,eAAe,MAAM,KAAI;AACvB,UAAG,QAAQ,IAAI;AAAG,eAAO;AACzB,UAAI,CAAC,MAAM,OAAO,IAAI,KAAK,SAAS,kBAAkB,GAAG;AACzD,UAAI,QAAQ,IAAI,SAAS,MAAM,KAAK,IAAI,KAAK,IAAI,MAAM,SAAS,GAAG;AACnE,UAAI,gBAAgB,KAAK,aAAa,OAAO,IAAI;AACjD,aAAO;IACT;IAEA,QAAQ,IAAG;AAAE,aAAO,KAAK,UAAU,SAAS,UAAU,EAAE,CAAC;IAAE;IAE3D,QAAQ,IAAG;AACT,UAAI,WAAW,SAAS,UAAU,EAAE;AAGpC,UAAG,GAAG,gBAAgB,CAAC,KAAK,YAAY,EAAE,GAAE;AAAE;MAAO;AAErD,UAAG,YAAY,CAAC,KAAK,UAAU,QAAQ,GAAE;AAEvC,YAAI,OAAO,YAAI,gBAAgB,EAAE,KAAK,SAAS,qCAAqC,GAAG,IAAI;AAC3F,aAAK,UAAU,QAAQ,IAAI;AAC3B,aAAK,aAAa,IAAI;AACtB,eAAO;MACT,WACQ,YAAY,CAAC,GAAG,cAAa;AAEnC;MACF,OAAO;AAEL,YAAI,WAAW,GAAG,aAAa,YAAY,UAAU,KAAK,GAAG,aAAa,KAAK,QAAQ,QAAQ,CAAC;AAChG,YAAI,YAAY,KAAK,WAAW,iBAAiB,QAAQ;AAEzD,YAAG,WAAU;AACX,cAAG,CAAC,GAAG,IAAG;AAAE,qBAAS,uBAAuB,yDAAyD,EAAE;UAAE;AACzG,cAAI,OAAO,IAAI,SAAS,MAAM,IAAI,SAAS;AAC3C,eAAK,UAAU,SAAS,UAAU,KAAK,EAAE,CAAC,IAAI;AAC9C,iBAAO;QACT,WAAU,aAAa,MAAK;AAC1B,mBAAS,2BAA2B,aAAa,EAAE;QACrD;MACF;IACF;IAEA,YAAY,MAAK;AAGf,YAAM,SAAS,SAAS,UAAU,KAAK,EAAE;AACzC,WAAK,YAAY;AACjB,WAAK,YAAY;AACjB,aAAO,KAAK,UAAU,MAAM;IAC9B;IAEA,sBAAqB;AAMnB,UAAG,KAAK,WAAW,eAAe,KAAK,KAAK,KAAK,OAAO,GAAE;AAAE;MAAO;AACnE,WAAK,aAAa,QAAQ,CAAC,EAAC,MAAM,OAAM,MAAM,KAAK,OAAO,MAAM,MAAM,CAAC;AACvE,WAAK,eAAe,CAAC;AACrB,WAAK,UAAU,CAAA,UAAS,MAAM,oBAAoB,CAAC;IACrD;IAEA,UAAU,UAAS;AACjB,UAAI,WAAW,KAAK,KAAK,SAAS,KAAK,EAAE,KAAK,CAAC;AAC/C,eAAQ,MAAM,UAAS;AAAE,iBAAS,KAAK,aAAa,EAAE,CAAC;MAAE;IAC3D;IAEA,UAAU,OAAO,IAAG;AAClB,WAAK,WAAW,UAAU,KAAK,SAAS,OAAO,CAAA,SAAQ;AACrD,YAAG,KAAK,cAAc,GAAE;AACtB,eAAK,KAAK,eAAe,KAAK,CAAC,MAAM,MAAM,GAAG,IAAI,CAAC,CAAC;QACtD,OAAO;AACL,eAAK,WAAW,iBAAiB,MAAM,GAAG,IAAI,CAAC;QACjD;MACF,CAAC;IACH;IAEA,cAAa;AAGX,WAAK,WAAW,UAAU,KAAK,SAAS,QAAQ,CAAC,YAAY;AAC3D,aAAK,WAAW,iBAAiB,MAAM;AACrC,eAAK,UAAU,UAAU,SAAS,CAAC,EAAC,MAAM,OAAM,MAAM,KAAK,OAAO,MAAM,MAAM,CAAC;QACjF,CAAC;MACH,CAAC;AACD,WAAK,UAAU,YAAY,CAAC,EAAC,IAAI,MAAK,MAAM,KAAK,WAAW,EAAC,IAAI,MAAK,CAAC,CAAC;AACxE,WAAK,UAAU,cAAc,CAAC,UAAU,KAAK,YAAY,KAAK,CAAC;AAC/D,WAAK,UAAU,iBAAiB,CAAC,UAAU,KAAK,eAAe,KAAK,CAAC;AACrE,WAAK,QAAQ,QAAQ,CAAA,WAAU,KAAK,QAAQ,MAAM,CAAC;AACnD,WAAK,QAAQ,QAAQ,CAAA,WAAU,KAAK,QAAQ,MAAM,CAAC;IACrD;IAEA,qBAAoB;AAAE,WAAK,UAAU,CAAA,UAAS,MAAM,QAAQ,CAAC;IAAE;IAE/D,eAAe,OAAM;AACnB,UAAI,EAAC,IAAI,MAAM,MAAK,IAAI;AACxB,UAAI,MAAM,KAAK,UAAU,EAAE;AAC3B,UAAI,IAAI,IAAI,YAAY,uBAAuB,EAAC,QAAQ,EAAC,IAAI,MAAM,MAAK,EAAC,CAAC;AAC1E,WAAK,WAAW,gBAAgB,GAAG,KAAK,MAAM,KAAK;IACrD;IAEA,YAAY,OAAM;AAChB,UAAI,EAAC,IAAI,KAAI,IAAI;AACjB,WAAK,OAAO,KAAK,UAAU,EAAE;AAC7B,WAAK,WAAW,aAAa,IAAI,IAAI;IACvC;IAEA,UAAU,IAAG;AACX,aAAO,GAAG,WAAW,GAAG,IAAI,GAAG,OAAO,SAAS,aAAa,OAAO,SAAS,OAAO,OAAO;IAC5F;IAEA,WAAW,EAAC,IAAI,OAAO,YAAW,GAAE;AAAE,WAAK,WAAW,SAAS,IAAI,OAAO,WAAW;IAAE;IAEvF,cAAa;AAAE,aAAO,KAAK;IAAU;IAErC,WAAU;AAAE,WAAK,SAAS;IAAK;IAE/B,WAAU;AACR,WAAK,WAAW,KAAK,YAAY,KAAK,QAAQ,KAAK;AACnD,aAAO,KAAK;IACd;IAEA,KAAK,UAAS;AACZ,WAAK,WAAW,KAAK,WAAW,aAAa;AAC7C,WAAK,YAAY;AACjB,UAAG,KAAK,OAAO,GAAE;AACf,aAAK,eAAe,KAAK,WAAW,gBAAgB,EAAC,IAAI,KAAK,MAAM,MAAM,UAAS,CAAC;MACtF;AACA,WAAK,eAAe,CAAC,WAAW;AAC9B,iBAAS,UAAU,WAAU;QAAC;AAC9B,mBAAW,SAAS,KAAK,WAAW,MAAM,IAAI,OAAO;MACvD;AAEA,WAAK,SAAS,MAAM,KAAK,QAAQ,KAAK,GAAG;QACvC,IAAI,CAAC,SAAS,KAAK,WAAW,iBAAiB,MAAM,KAAK,OAAO,IAAI,CAAC;QACtE,OAAO,CAAC,UAAU,KAAK,YAAY,KAAK;QACxC,SAAS,MAAM,KAAK,YAAY,EAAC,QAAQ,UAAS,CAAC;MACrD,CAAC;IACH;IAEA,YAAY,MAAK;AACf,UAAG,KAAK,WAAW,UAAS;AAC1B,aAAK,IAAI,SAAS,MAAM,CAAC,qBAAqB,KAAK,uCAAuC,IAAI,CAAC;AAC/F,aAAK,WAAW,EAAC,IAAI,KAAK,KAAK,MAAM,aAAa,KAAK,MAAK,CAAC;AAC7D;MACF,WAAU,KAAK,WAAW,kBAAkB,KAAK,WAAW,SAAQ;AAClE,aAAK,IAAI,SAAS,MAAM,CAAC,4DAA4D,IAAI,CAAC;AAC1F,aAAK,WAAW,EAAC,IAAI,KAAK,KAAK,MAAM,OAAO,KAAK,MAAK,CAAC;AACvD;MACF;AACA,UAAG,KAAK,YAAY,KAAK,eAAc;AACrC,aAAK,cAAc;AACnB,aAAK,QAAQ,MAAM;MACrB;AACA,UAAG,KAAK,UAAS;AAAE,eAAO,KAAK,WAAW,KAAK,QAAQ;MAAE;AACzD,UAAG,KAAK,eAAc;AAAE,eAAO,KAAK,eAAe,KAAK,aAAa;MAAE;AACvE,WAAK,IAAI,SAAS,MAAM,CAAC,kBAAkB,IAAI,CAAC;AAChD,UAAG,KAAK,OAAO,GAAE;AACf,aAAK,aAAa,CAAC,mBAAmB,iBAAiB,sBAAsB,CAAC;AAC9E,YAAG,KAAK,WAAW,YAAY,GAAE;AAAE,eAAK,WAAW,iBAAiB,IAAI;QAAE;MAC5E,OAAO;AACL,YAAG,KAAK,gBAAgB,yBAAwB;AAE9C,eAAK,KAAK,aAAa,CAAC,mBAAmB,iBAAiB,sBAAsB,CAAC;AACnF,eAAK,IAAI,SAAS,MAAM,CAAC,mCAAmC,iCAAiC,IAAI,CAAC;AAClG,eAAK,QAAQ;QACf;AACA,YAAI,cAAc,YAAI,KAAK,KAAK,GAAG,EAAE;AACrC,YAAG,aAAY;AACb,sBAAI,WAAW,aAAa,KAAK,EAAE;AACnC,eAAK,aAAa,CAAC,mBAAmB,iBAAiB,sBAAsB,CAAC;AAC9E,eAAK,KAAK;QACZ,OAAO;AACL,eAAK,QAAQ;QACf;MACF;IACF;IAEA,QAAQ,QAAO;AACb,UAAG,KAAK,YAAY,GAAE;AAAE;MAAO;AAC/B,UAAG,KAAK,OAAO,KAAK,KAAK,WAAW,eAAe,KAAK,WAAW,SAAQ;AACzE,eAAO,KAAK,WAAW,iBAAiB,IAAI;MAC9C;AACA,WAAK,mBAAmB;AACxB,WAAK,WAAW,kBAAkB,IAAI;AAEtC,UAAG,SAAS,eAAc;AAAE,iBAAS,cAAc,KAAK;MAAE;AAC1D,UAAG,KAAK,WAAW,WAAW,GAAE;AAC9B,aAAK,WAAW,4BAA4B;MAC9C;IACF;IAEA,QAAQ,QAAO;AACb,WAAK,QAAQ,MAAM;AACnB,UAAG,KAAK,WAAW,YAAY,GAAE;AAAE,aAAK,IAAI,SAAS,MAAM,CAAC,gBAAgB,MAAM,CAAC;MAAE;AACrF,UAAG,CAAC,KAAK,WAAW,WAAW,GAAE;AAC/B,YAAG,KAAK,WAAW,YAAY,GAAE;AAC/B,eAAK,aAAa,CAAC,mBAAmB,iBAAiB,sBAAsB,CAAC;QAChF,OAAO;AACL,eAAK,aAAa,CAAC,mBAAmB,iBAAiB,sBAAsB,CAAC;QAChF;MACF;IACF;IAEA,aAAa,SAAQ;AACnB,UAAG,KAAK,OAAO,GAAE;AAAE,oBAAI,cAAc,QAAQ,0BAA0B,EAAC,QAAQ,EAAC,IAAI,KAAK,MAAM,MAAM,QAAO,EAAC,CAAC;MAAE;AACjH,WAAK,WAAW;AAChB,WAAK,oBAAoB,GAAG,OAAO;AACnC,WAAK,oBAAoB;IAC3B;IAEA,sBAAqB;AACnB,WAAK,oBAAoB,WAAW,MAAM;AACxC,aAAK,QAAQ,KAAK,QAAQ,cAAc,CAAC;MAC3C,GAAG,KAAK,WAAW,mBAAmB;IACxC;IAEA,SAAS,YAAY,UAAS;AAC5B,UAAI,UAAU,KAAK,WAAW,cAAc;AAC5C,UAAI,cAAc,UAChB,CAAC,OAAO,WAAW,MAAM,CAAC,KAAK,YAAY,KAAK,GAAG,GAAG,OAAO,IAC7D,CAAC,OAAO,CAAC,KAAK,YAAY,KAAK,GAAG;AAEpC,kBAAY,MAAM;AAChB,mBAAW,EACR,QAAQ,MAAM,CAAA,SAAQ,YAAY,MAAM,SAAS,MAAM,SAAS,GAAG,IAAI,CAAC,CAAC,EACzE,QAAQ,SAAS,CAAA,WAAU,YAAY,MAAM,SAAS,SAAS,SAAS,MAAM,MAAM,CAAC,CAAC,EACtF,QAAQ,WAAW,MAAM,YAAY,MAAM,SAAS,WAAW,SAAS,QAAQ,CAAC,CAAC;MACvF,CAAC;IACH;IAEA,cAAc,cAAc,OAAO,SAAQ;AACzC,UAAG,CAAC,KAAK,YAAY,GAAE;AAAE,eAAO,QAAQ,OAAO,EAAC,OAAO,eAAc,CAAC;MAAE;AAExE,UAAI,CAAC,KAAK,CAAC,EAAE,GAAG,IAAI,IAAI,eAAe,aAAa,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;AACrE,UAAI,eAAe,KAAK;AACxB,UAAI,gBAAgB,WAAU;MAAC;AAC/B,UAAG,KAAK,cAAa;AACnB,wBAAgB,KAAK,WAAW,gBAAgB,EAAC,MAAM,WAAW,QAAQ,GAAE,CAAC;MAC/E;AAEA,UAAG,OAAQ,QAAQ,QAAS,UAAS;AAAE,eAAO,QAAQ;MAAI;AAE1D,aAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,aAAK,SAAS,MAAM,KAAK,QAAQ,KAAK,OAAO,SAAS,YAAY,GAAG;UACnE,IAAI,CAAC,SAAS;AACZ,gBAAG,QAAQ,MAAK;AAAE,mBAAK,aAAa;YAAI;AACxC,gBAAI,SAAS,CAAC,cAAc;AAC1B,kBAAG,KAAK,UAAS;AAAE,qBAAK,WAAW,KAAK,QAAQ;cAAE;AAClD,kBAAG,KAAK,YAAW;AAAE,qBAAK,YAAY,KAAK,UAAU;cAAE;AACvD,kBAAG,KAAK,eAAc;AAAE,qBAAK,eAAe,KAAK,aAAa;cAAE;AAChE,4BAAc;AACd,sBAAQ,EAAC,MAAY,OAAO,UAAS,CAAC;YACxC;AACA,gBAAG,KAAK,MAAK;AACX,mBAAK,WAAW,iBAAiB,MAAM;AACrC,qBAAK,UAAU,UAAU,KAAK,MAAM,CAAC,EAAC,MAAM,OAAO,OAAM,MAAM;AAC7D,sBAAG,QAAQ,MAAK;AACd,yBAAK,SAAS,KAAK,QAAQ,KAAK;kBAClC;AACA,uBAAK,OAAO,MAAM,MAAM;AACxB,yBAAO,KAAK;gBACd,CAAC;cACH,CAAC;YACH,OAAO;AACL,kBAAG,QAAQ,MAAK;AAAE,qBAAK,SAAS,KAAK,QAAQ,KAAK;cAAE;AACpD,qBAAO,IAAI;YACb;UACF;UACA,OAAO,CAAC,WAAW,OAAO,EAAC,OAAO,OAAM,CAAC;UACzC,SAAS,MAAM;AACb,mBAAO,EAAC,SAAS,KAAI,CAAC;AACtB,gBAAG,KAAK,cAAc,cAAa;AACjC,mBAAK,WAAW,iBAAiB,MAAM,MAAM;AAC3C,qBAAK,IAAI,WAAW,MAAM,CAAC,6FAA6F,CAAC;cAC3H,CAAC;YACH;UACF;QACF,CAAC;MACH,CAAC;IACH;IAEA,SAAS,KAAK,UAAU,SAAQ;AAC9B,UAAG,CAAC,KAAK,YAAY,GAAE;AAAE;MAAO;AAChC,UAAI,WAAW,IAAI,gBAAgB,KAAK,OAAO;AAE/C,UAAG,SAAQ;AACT,kBAAU,IAAI,IAAI,OAAO;AACzB,oBAAI,IAAI,UAAU,UAAU,CAAA,WAAU;AACpC,cAAG,WAAW,CAAC,QAAQ,IAAI,MAAM,GAAE;AAAE;UAAO;AAE5C,sBAAI,IAAI,QAAQ,UAAU,CAAA,UAAS,KAAK,UAAU,OAAO,KAAK,QAAQ,CAAC;AACvE,eAAK,UAAU,QAAQ,KAAK,QAAQ;QACtC,CAAC;MACH,OAAO;AACL,oBAAI,IAAI,UAAU,UAAU,CAAA,OAAM,KAAK,UAAU,IAAI,KAAK,QAAQ,CAAC;MACrE;IACF;IAEA,UAAU,IAAI,KAAK,UAAS;AAC1B,UAAI,QAAQ,IAAI,WAAW,EAAE;AAE7B,YAAM,UAAU,KAAK,UAAU,CAAA,eAAc;AAG3C,YAAI,QAAQ,IAAI,SAAS,MAAM,IAAI,KAAK,IAAI,YAAY,CAAC,GAAG,MAAM,EAAC,SAAS,IAAG,CAAC;AAChF,cAAM,mBAAmB,KAAK,aAAa,OAAO,IAAI;AACtD,oBAAI,IAAI,IAAI,IAAI,gBAAgB,KAAK,OAAO,OAAO,CAAA,UAAS,KAAK,UAAU,OAAO,KAAK,QAAQ,CAAC;AAChG,YAAG,kBAAiB;AAAE,eAAK,gBAAgB;QAAE;MAC/C,CAAC;IACH;IAEA,SAAQ;AAAE,aAAO,KAAK,GAAG;IAAG;IAE5B,OAAO,UAAU,UAAU,WAAW,OAAO,CAAC,GAAE;AAC9C,UAAI,SAAS,KAAK;AAClB,UAAI,cAAc,KAAK,QAAQ,gBAAgB;AAC/C,UAAG,KAAK,SAAQ;AACd,YAAI,aAAa,YAAI,IAAI,UAAU,KAAK,OAAO,EAAE,IAAI,CAAA,OAAM;AACzD,iBAAO,EAAC,IAAI,MAAM,MAAM,SAAS,KAAI;QACvC,CAAC;AACD,mBAAW,SAAS,OAAO,UAAU;MACvC;AAEA,eAAQ,EAAC,IAAI,MAAM,QAAO,KAAK,UAAS;AACtC,YAAG,CAAC,QAAQ,CAAC,SAAQ;AAAE,gBAAM,IAAI,MAAM,iCAAiC;QAAE;AAC1E,WAAG,aAAa,aAAa,KAAK,OAAO,CAAC;AAC1C,YAAG,SAAQ;AAAE,aAAG,aAAa,iBAAiB,MAAM;QAAE;AACtD,YAAG,MAAK;AAAE,aAAG,aAAa,cAAc,MAAM;QAAE;AAEhD,YAAG,CAAC,WAAY,KAAK,aAAa,EAAE,OAAO,KAAK,aAAa,OAAO,KAAK,OAAO;AAAE;QAAS;AAE3F,YAAI,sBAAsB,IAAI,QAAQ,CAAA,YAAW;AAC/C,aAAG,iBAAiB,iBAAiB,UAAU,MAAM,QAAQ,MAAM,GAAG,EAAC,MAAM,KAAI,CAAC;QACpF,CAAC;AAED,YAAI,yBAAyB,IAAI,QAAQ,CAAA,YAAW;AAClD,aAAG,iBAAiB,oBAAoB,UAAU,MAAM,QAAQ,MAAM,GAAG,EAAC,MAAM,KAAI,CAAC;QACvF,CAAC;AAED,WAAG,UAAU,IAAI,OAAO,mBAAmB;AAC3C,YAAI,cAAc,GAAG,aAAa,WAAW;AAC7C,YAAG,gBAAgB,MAAK;AACtB,cAAG,CAAC,GAAG,aAAa,wBAAwB,GAAE;AAC5C,eAAG,aAAa,0BAA0B,GAAG,SAAS;UACxD;AACA,cAAG,gBAAgB,IAAG;AAAE,eAAG,YAAY;UAAY;AAEnD,aAAG,aAAa,cAAc,GAAG,aAAa,YAAY,KAAK,GAAG,QAAQ;AAC1E,aAAG,aAAa,YAAY,EAAE;QAChC;AAEA,YAAI,SAAS;UACX,OAAO;UACP;UACA,KAAK;UACL,WAAW;UACX,UAAU;UACV,cAAc,SAAS,OAAO,CAAC,EAAC,MAAAa,MAAI,MAAMA,KAAI,EAAE,IAAI,CAAC,EAAC,IAAAH,IAAE,MAAMA,GAAE;UAChE,iBAAiB,SAAS,OAAO,CAAC,EAAC,SAAAI,SAAO,MAAMA,QAAO,EAAE,IAAI,CAAC,EAAC,IAAAJ,IAAE,MAAMA,GAAE;UACzE,QAAQ,CAAC,QAAQ;AACf,kBAAM,MAAM,QAAQ,GAAG,IAAI,MAAM,CAAC,GAAG;AACrC,iBAAK,SAAS,QAAQ,UAAU,GAAG;UACrC;UACA,cAAc;UACd,iBAAiB;UACjB,MAAM,CAAC,WAAW;AAChB,mBAAO,IAAI,QAAQ,CAAA,YAAW;AAC5B,kBAAG,KAAK,QAAQ,MAAM,GAAE;AAAE,uBAAO,QAAQ,MAAM;cAAE;AACjD,qBAAO,aAAa,cAAc,MAAM;AACxC,qBAAO,aAAa,aAAa,KAAK,OAAO,CAAC;AAC9C,qBAAO,iBAAiB,iBAAiB,UAAU,MAAM,QAAQ,MAAM,GAAG,EAAC,MAAM,KAAI,CAAC;YACxF,CAAC;UACH;QACF;AACA,WAAG,cAAc,IAAI,YAAY,YAAY;UAC3C;UACA,SAAS;UACT,YAAY;QACd,CAAC,CAAC;AACF,YAAG,UAAS;AACV,aAAG,cAAc,IAAI,YAAY,YAAY,YAAY;YACvD;YACA,SAAS;YACT,YAAY;UACd,CAAC,CAAC;QACJ;MACF;AACA,aAAO,CAAC,QAAQ,SAAS,IAAI,CAAC,EAAC,GAAE,MAAM,EAAE,GAAG,IAAI;IAClD;IAEA,QAAQ,KAAI;AAAE,aAAO,KAAK,eAAe,QAAQ,KAAK,cAAc;IAAI;IAExE,YAAY,IAAG;AACb,UAAI,MAAM,GAAG,gBAAgB,GAAG,aAAa,aAAa;AAC1D,aAAO,MAAM,SAAS,GAAG,IAAI;IAC/B;IAEA,kBAAkB,QAAQ,WAAW,OAAO,CAAC,GAAE;AAC7C,UAAG,MAAM,SAAS,GAAE;AAAE,eAAO;MAAU;AAEvC,UAAI,gBAAgB,KAAK,UAAU,OAAO,aAAa,KAAK,QAAQ,QAAQ,CAAC;AAC7E,UAAG,MAAM,aAAa,GAAE;AACtB,eAAO,SAAS,aAAa;MAC/B,WAAU,cAAc,kBAAkB,QAAQ,KAAK,SAAQ;AAC7D,eAAO,KAAK,mBAAmB,SAAS;MAC1C,OAAO;AACL,eAAO;MACT;IACF;IAEA,mBAAmB,WAAU;AAC3B,UAAG,MAAM,SAAS,GAAE;AAClB,eAAO;MACT,WAAU,WAAU;AAClB,eAAO,MAAM,UAAU,QAAQ,IAAI,gBAAgB,GAAG,CAAA,OAAM,KAAK,YAAY,EAAE,KAAK,KAAK,YAAY,EAAE,CAAC;MAC1G,OAAO;AACL,eAAO;MACT;IACF;IAEA,cAAc,IAAI,WAAW,OAAO,SAAS,SAAQ;AACnD,UAAG,CAAC,KAAK,YAAY,GAAE;AACrB,aAAK,IAAI,QAAQ,MAAM,CAAC,qDAAqD,OAAO,OAAO,CAAC;AAC5F,eAAO;MACT;AACA,UAAI,CAAC,KAAK,KAAK,IAAI,IAAI,KAAK,OAAO,CAAC,EAAC,IAAI,SAAS,MAAM,MAAM,KAAI,CAAC,GAAG,OAAO,MAAM;AACnF,WAAK,cAAc,MAAM,CAAC,KAAK,KAAK,IAAI,GAAG,SAAS;QAClD,MAAM;QACN;QACA,OAAO;QACP,KAAK,KAAK,mBAAmB,SAAS;MACxC,CAAC,EAAE,KAAK,CAAC,EAAC,MAAM,OAAO,OAAO,UAAS,MAAM,QAAQ,WAAW,GAAG,CAAC;AAEpE,aAAO;IACT;IAEA,YAAY,IAAI,MAAM,OAAM;AAC1B,UAAI,SAAS,KAAK,QAAQ,QAAQ;AAClC,eAAQ,IAAI,GAAG,IAAI,GAAG,WAAW,QAAQ,KAAI;AAC3C,YAAG,CAAC,MAAK;AAAE,iBAAO,CAAC;QAAE;AACrB,YAAI,OAAO,GAAG,WAAW,CAAC,EAAE;AAC5B,YAAG,KAAK,WAAW,MAAM,GAAE;AAAE,eAAK,KAAK,QAAQ,QAAQ,EAAE,CAAC,IAAI,GAAG,aAAa,IAAI;QAAE;MACtF;AACA,UAAG,GAAG,UAAU,UAAa,EAAE,cAAc,kBAAiB;AAC5D,YAAG,CAAC,MAAK;AAAE,iBAAO,CAAC;QAAE;AACrB,aAAK,QAAQ,GAAG;AAEhB,YAAG,GAAG,YAAY,WAAW,iBAAiB,QAAQ,GAAG,IAAI,KAAK,KAAK,CAAC,GAAG,SAAQ;AACjF,iBAAO,KAAK;QACd;MACF;AACA,UAAG,OAAM;AACP,YAAG,CAAC,MAAK;AAAE,iBAAO,CAAC;QAAE;AACrB,iBAAQ,OAAO,OAAM;AAAE,eAAK,GAAG,IAAI,MAAM,GAAG;QAAE;MAChD;AACA,aAAO;IACT;IAEA,UAAU,MAAM,IAAI,WAAW,UAAU,MAAM,OAAO,CAAC,GAAG,SAAQ;AAChE,WAAK,cAAc,MAAM,KAAK,OAAO,CAAC,EAAC,IAAI,SAAS,MAAM,MAAM,KAAI,CAAC,GAAG,UAAU,MAAM,IAAI,GAAG,SAAS;QACtG;QACA,OAAO;QACP,OAAO,KAAK,YAAY,IAAI,MAAM,KAAK,KAAK;QAC5C,KAAK,KAAK,kBAAkB,IAAI,WAAW,IAAI;MACjD,CAAC,EACE,KAAK,CAAC,EAAC,MAAK,MAAM,WAAW,QAAQ,KAAK,CAAC,EAC3C,MAAM,CAAC,UAAU,SAAS,wBAAwB,KAAK,CAAC;IAC7D;IAEA,iBAAiB,QAAQ,UAAU,UAAU,UAAU,WAAW;IAAE,GAAE;AACpE,WAAK,WAAW,aAAa,OAAO,MAAM,CAAC,MAAM,cAAc;AAC7D,aAAK,cAAc,MAAM,YAAY;UACnC,OAAO,OAAO,aAAa,KAAK,QAAQ,YAAY,CAAC;UACrD,KAAK,OAAO,aAAa,cAAc;UACvC,WAAW;UACX;UACA,KAAK,KAAK,kBAAkB,OAAO,MAAM,SAAS;QACpD,CAAC,EACE,KAAK,CAAC,EAAC,KAAI,MAAM,QAAQ,IAAI,CAAC,EAC9B,MAAM,CAAC,UAAU,SAAS,gCAAgC,KAAK,CAAC;MACrE,CAAC;IACH;IAEA,UAAU,SAAS,WAAW,UAAU,UAAU,MAAM,UAAS;AAC/D,UAAG,CAAC,QAAQ,MAAK;AACf,cAAM,IAAI,MAAM,mDAAmD;MACrE;AAEA,UAAI;AACJ,UAAI,MAAM,MAAM,QAAQ,IAAI,WAAW,KAAK,kBAAkB,QAAQ,MAAM,WAAW,IAAI;AAC3F,UAAI,eAAe,MAAM;AACvB,eAAO,KAAK,OAAO;UACjB,EAAC,IAAI,SAAS,SAAS,MAAM,MAAM,KAAI;UACvC,EAAC,IAAI,QAAQ,MAAM,SAAS,MAAM,MAAM,KAAI;QAC9C,GAAG,UAAU,UAAU,IAAI;MAC7B;AACA,UAAI;AACJ,UAAI,OAAO,KAAK,YAAY,QAAQ,MAAM,CAAC,GAAG,KAAK,KAAK;AACxD,UAAI,gBAAgB,CAAC;AACrB,UAAG,mBAAmB,mBAAkB;AAAE,sBAAc,YAAY;MAAQ;AAC5E,UAAG,QAAQ,aAAa,KAAK,QAAQ,QAAQ,CAAC,GAAE;AAC9C,mBAAW,cAAc,QAAQ,MAAM,eAAe,CAAC,QAAQ,IAAI,CAAC;MACtE,OAAO;AACL,mBAAW,cAAc,QAAQ,MAAM,aAAa;MACtD;AACA,UAAG,YAAI,cAAc,OAAO,KAAK,QAAQ,SAAS,QAAQ,MAAM,SAAS,GAAE;AACzE,qBAAa,WAAW,SAAS,MAAM,KAAK,QAAQ,KAAK,CAAC;MAC5D;AACA,gBAAU,aAAa,iBAAiB,OAAO;AAE/C,UAAI,QAAQ;QACV,MAAM;QACN,OAAO;QACP,OAAO;QACP,MAAM;;;;;UAKJ,SAAS,KAAK,WAAW;WACtB;QAEL;QACA;MACF;AACA,WAAK,cAAc,cAAc,SAAS,KAAK,EAAE,KAAK,CAAC,EAAC,KAAI,MAAM;AAChE,YAAG,YAAI,cAAc,OAAO,KAAK,YAAI,aAAa,OAAO,GAAE;AAIzD,qBAAW,SAAS,SAAS,MAAM;AACjC,gBAAG,aAAa,uBAAuB,OAAO,EAAE,SAAS,GAAE;AACzD,kBAAI,CAAC,KAAK,IAAI,IAAI,aAAa;AAC/B,mBAAK,SAAS,KAAK,UAAU,CAAC,QAAQ,IAAI,CAAC;AAC3C,mBAAK,YAAY,QAAQ,MAAM,UAAU,WAAW,KAAK,KAAK,CAAC,aAAa;AAC1E,4BAAY,SAAS,IAAI;AACzB,qBAAK,sBAAsB,QAAQ,MAAM,QAAQ;AACjD,qBAAK,SAAS,KAAK,QAAQ;cAC7B,CAAC;YACH;UACF,CAAC;QACH,OAAO;AACL,sBAAY,SAAS,IAAI;QAC3B;MACF,CAAC,EAAE,MAAM,CAAC,UAAU,SAAS,8BAA8B,KAAK,CAAC;IACnE;IAEA,sBAAsB,QAAQ,UAAS;AACrC,UAAI,iBAAiB,KAAK,mBAAmB,MAAM;AACnD,UAAG,gBAAe;AAChB,YAAI,CAAC,KAAK,MAAM,OAAO,QAAQ,IAAI;AACnC,aAAK,aAAa,QAAQ,QAAQ;AAClC,iBAAS;MACX;IACF;IAEA,mBAAmB,QAAO;AACxB,aAAO,KAAK,YAAY,KAAK,CAAC,CAAC,IAAI,MAAM,OAAO,SAAS,MAAM,GAAG,WAAW,MAAM,CAAC;IACtF;IAEA,eAAe,QAAQ,KAAK,MAAM,UAAS;AACzC,UAAG,KAAK,mBAAmB,MAAM,GAAE;AAAE,eAAO;MAAK;AACjD,WAAK,YAAY,KAAK,CAAC,QAAQ,KAAK,MAAM,QAAQ,CAAC;IACrD;IAEA,aAAa,QAAQ,UAAS;AAC5B,WAAK,cAAc,KAAK,YAAY,OAAO,CAAC,CAAC,IAAI,KAAK,OAAO,SAAS,MAAM;AAC1E,YAAG,GAAG,WAAW,MAAM,GAAE;AACvB,eAAK,SAAS,KAAK,QAAQ;AAC3B,iBAAO;QACT,OAAO;AACL,iBAAO;QACT;MACF,CAAC;IACH;IAEA,YAAY,QAAQ,UAAU,OAAO,CAAC,GAAE;AACtC,UAAI,gBAAgB,CAAA,OAAM;AACxB,YAAI,cAAc,kBAAkB,IAAI,GAAG,KAAK,QAAQ,UAAU,YAAY,GAAG,IAAI;AACrF,eAAO,EAAE,eAAe,kBAAkB,IAAI,0BAA0B,GAAG,IAAI;MACjF;AACA,UAAI,iBAAiB,CAAA,OAAM;AACzB,eAAO,GAAG,aAAa,KAAK,QAAQ,gBAAgB,CAAC;MACvD;AACA,UAAI,eAAe,CAAA,OAAM,GAAG,WAAW;AAEvC,UAAI,cAAc,CAAA,OAAM,CAAC,SAAS,YAAY,QAAQ,EAAE,SAAS,GAAG,OAAO;AAE3E,UAAI,eAAe,MAAM,KAAK,OAAO,QAAQ;AAC7C,UAAI,WAAW,aAAa,OAAO,cAAc;AACjD,UAAI,UAAU,aAAa,OAAO,YAAY,EAAE,OAAO,aAAa;AACpE,UAAI,SAAS,aAAa,OAAO,WAAW,EAAE,OAAO,aAAa;AAElE,cAAQ,QAAQ,CAAA,WAAU;AACxB,eAAO,aAAa,cAAc,OAAO,QAAQ;AACjD,eAAO,WAAW;MACpB,CAAC;AACD,aAAO,QAAQ,CAAA,UAAS;AACtB,cAAM,aAAa,cAAc,MAAM,QAAQ;AAC/C,cAAM,WAAW;AACjB,YAAG,MAAM,OAAM;AACb,gBAAM,aAAa,cAAc,MAAM,QAAQ;AAC/C,gBAAM,WAAW;QACnB;MACF,CAAC;AACD,UAAI,UAAU,SAAS,OAAO,OAAO,EAAE,OAAO,MAAM,EAAE,IAAI,CAAA,OAAM;AAC9D,eAAO,EAAC,IAAI,SAAS,MAAM,MAAM,KAAI;MACvC,CAAC;AAID,UAAI,MAAM,CAAC,EAAC,IAAI,QAAQ,SAAS,MAAM,MAAM,MAAK,CAAC,EAAE,OAAO,OAAO,EAAE,QAAQ;AAC7E,aAAO,KAAK,OAAO,KAAK,UAAU,UAAU,IAAI;IAClD;IAEA,eAAe,QAAQ,WAAW,UAAU,WAAW,MAAM,SAAQ;AACnE,UAAI,eAAe,MAAM,KAAK,YAAY,QAAQ,UAAU,iCACvD,OADuD;QAE1D,MAAM;QACN;MACF,EAAC;AAGD,kBAAI,WAAW,QAAQ,aAAa,SAAS;AAC7C,UAAI,MAAM,KAAK,kBAAkB,QAAQ,SAAS;AAClD,UAAG,aAAa,qBAAqB,MAAM,GAAE;AAC3C,YAAI,CAAC,KAAK,IAAI,IAAI,aAAa;AAC/B,YAAI,OAAO,MAAM,KAAK,eAAe,QAAQ,WAAW,UAAU,WAAW,MAAM,OAAO;AAC1F,eAAO,KAAK,eAAe,QAAQ,KAAK,MAAM,IAAI;MACpD,WAAU,aAAa,wBAAwB,MAAM,EAAE,SAAS,GAAE;AAChE,YAAI,CAAC,KAAK,GAAG,IAAI,aAAa;AAC9B,YAAI,cAAc,MAAM,CAAC,KAAK,KAAK,IAAI;AACvC,aAAK,YAAY,QAAQ,UAAU,WAAW,KAAK,KAAK,CAAC,aAAa;AAGpE,cAAG,aAAa,wBAAwB,MAAM,EAAE,SAAS,GAAE;AACzD,mBAAO,KAAK,SAAS,KAAK,QAAQ;UACpC;AACA,cAAI,OAAO,KAAK,YAAY,QAAQ,CAAC,GAAG,KAAK,KAAK;AAClD,cAAI,WAAW,cAAc,QAAQ,EAAC,UAAS,CAAC;AAChD,eAAK,cAAc,aAAa,SAAS;YACvC,MAAM;YACN,OAAO;YACP,OAAO;YACP;YACA;UACF,CAAC,EACE,KAAK,CAAC,EAAC,KAAI,MAAM,QAAQ,IAAI,CAAC,EAC9B,MAAM,CAAC,UAAU,SAAS,8BAA8B,KAAK,CAAC;QACnE,CAAC;MACH,WAAU,EAAE,OAAO,aAAa,WAAW,KAAK,OAAO,UAAU,SAAS,oBAAoB,IAAG;AAC/F,YAAI,OAAO,KAAK,YAAY,QAAQ,CAAC,GAAG,KAAK,KAAK;AAClD,YAAI,WAAW,cAAc,QAAQ,EAAC,UAAS,CAAC;AAChD,aAAK,cAAc,cAAc,SAAS;UACxC,MAAM;UACN,OAAO;UACP,OAAO;UACP;UACA;QACF,CAAC,EACE,KAAK,CAAC,EAAC,KAAI,MAAM,QAAQ,IAAI,CAAC,EAC9B,MAAM,CAAC,UAAU,SAAS,8BAA8B,KAAK,CAAC;MACnE;IACF;IAEA,YAAY,QAAQ,UAAU,WAAW,KAAK,KAAK,YAAW;AAC5D,UAAI,oBAAoB,KAAK;AAC7B,UAAI,WAAW,aAAa,iBAAiB,MAAM;AACnD,UAAI,0BAA0B,SAAS;AAGvC,eAAS,QAAQ,CAAA,YAAW;AAC1B,YAAI,WAAW,IAAI,aAAa,SAAS,MAAM,MAAM;AACnD;AACA,cAAG,4BAA4B,GAAE;AAAE,uBAAW;UAAE;QAClD,CAAC;AAED,YAAI,UAAU,SAAS,QAAQ,EAAE,IAAI,CAAA,UAAS,MAAM,mBAAmB,CAAC;AAExE,YAAG,QAAQ,WAAW,GAAE;AACtB;AACA;QACF;AAEA,YAAI,UAAU;UACZ,KAAK,QAAQ,aAAa,cAAc;UACxC;UACA,KAAK,KAAK,kBAAkB,QAAQ,MAAM,SAAS;QACrD;AAEA,aAAK,IAAI,UAAU,MAAM,CAAC,6BAA6B,OAAO,CAAC;AAE/D,aAAK,cAAc,MAAM,gBAAgB,OAAO,EAAE,KAAK,CAAC,EAAC,KAAI,MAAM;AACjE,eAAK,IAAI,UAAU,MAAM,CAAC,0BAA0B,IAAI,CAAC;AAGzD,mBAAS,QAAQ,EAAE,QAAQ,CAAA,UAAS;AAClC,gBAAG,KAAK,WAAW,CAAC,KAAK,QAAQ,MAAM,GAAG,GAAE;AAC1C,mBAAK,2BAA2B,MAAM,KAAK,oBAAoB,QAAQ;YACzE;UACF,CAAC;AAGD,cAAG,KAAK,SAAS,OAAO,KAAK,KAAK,OAAO,EAAE,WAAW,GAAE;AACtD,iBAAK,SAAS,KAAK,QAAQ;AAC3B,gBAAI,SAAS,KAAK,SAAS,CAAC;AAC5B,mBAAO,IAAI,CAAC,CAAC,WAAW,MAAM,MAAM;AAClC,mBAAK,2BAA2B,WAAW,QAAQ,QAAQ;YAC7D,CAAC;UACH,OAAO;AACL,gBAAI,UAAU,CAAC,aAAa;AAC1B,mBAAK,QAAQ,QAAQ,MAAM;AACzB,oBAAG,KAAK,cAAc,mBAAkB;AAAE,2BAAS;gBAAE;cACvD,CAAC;YACH;AACA,qBAAS,kBAAkB,MAAM,SAAS,KAAK,UAAU;UAC3D;QACF,CAAC,EAAE,MAAM,CAAC,UAAU,SAAS,yBAAyB,KAAK,CAAC;MAC9D,CAAC;IACH;IAEA,2BAA2B,WAAW,QAAQ,UAAS;AACrD,UAAG,SAAS,aAAa,GAAE;AAEzB,YAAI,QAAQ,SAAS,QAAQ,EAAE,KAAK,CAAAK,WAASA,OAAM,QAAQ,UAAU,SAAS,CAAC;AAC/E,YAAG,OAAM;AAAE,gBAAM,OAAO;QAAE;MAC5B,OAAO;AACL,iBAAS,QAAQ,EAAE,IAAI,CAAA,UAAS,MAAM,OAAO,CAAC;MAChD;AACA,WAAK,IAAI,UAAU,MAAM,CAAC,mBAAmB,aAAa,MAAM,CAAC;IACnE;IAEA,gBAAgB,WAAW,MAAM,cAAa;AAC5C,UAAI,gBAAgB,KAAK,iBAAiB,SAAS,KAAK,KAAK;AAC7D,UAAI,SAAS,YAAI,iBAAiB,aAAa,EAAE,OAAO,CAAA,OAAM,GAAG,SAAS,IAAI;AAC9E,UAAG,OAAO,WAAW,GAAE;AAAE,iBAAS,gDAAgD,OAAO;MAAE,WACnF,OAAO,SAAS,GAAE;AAAE,iBAAS,uDAAuD,OAAO;MAAE,OAChG;AAAE,oBAAI,cAAc,OAAO,CAAC,GAAG,mBAAmB,EAAC,QAAQ,EAAC,OAAO,aAAY,EAAC,CAAC;MAAE;IAC1F;IAEA,iBAAiB,WAAU;AACzB,UAAG,MAAM,SAAS,GAAE;AAClB,YAAI,CAAC,MAAM,IAAI,YAAI,sBAAsB,KAAK,IAAI,SAAS;AAC3D,eAAO;MACT,WAAU,WAAU;AAClB,eAAO;MACT,OAAO;AACL,eAAO;MACT;IACF;IAEA,iBAAiB,SAAS,SAAS,aAAa,UAAS;AAGvD,YAAM,YAAY,KAAK,QAAQ,QAAQ;AACvC,YAAM,YAAY,QAAQ,aAAa,KAAK,QAAQ,QAAQ,CAAC,KAAK;AAClE,YAAM,WAAW,QAAQ,aAAa,KAAK,QAAQ,gBAAgB,CAAC,KAAK,QAAQ,aAAa,KAAK,QAAQ,QAAQ,CAAC;AACpH,YAAM,SAAS,MAAM,KAAK,QAAQ,QAAQ,EAAE,OAAO,CAAA,OAAM,YAAI,YAAY,EAAE,KAAK,GAAG,QAAQ,CAAC,GAAG,aAAa,SAAS,CAAC;AACtH,UAAG,OAAO,WAAW,GAAE;AACrB,iBAAS;AACT;MACF;AAGA,aAAO,QAAQ,CAAAC,WAASA,OAAM,aAAa,cAAc,KAAK,aAAa,WAAWA,MAAK,CAAC;AAG5F,UAAI,QAAQ,OAAO,KAAK,CAAA,OAAM,GAAG,SAAS,QAAQ,KAAK,OAAO,CAAC;AAI/D,UAAI,UAAU;AAEd,WAAK,cAAc,WAAW,CAAC,YAAY,cAAc;AACvD,cAAM,MAAM,KAAK,kBAAkB,SAAS,SAAS;AACrD;AACA,YAAI,IAAI,IAAI,YAAY,qBAAqB,EAAC,QAAQ,EAAC,eAAe,QAAO,EAAC,CAAC;AAC/E,mBAAG,KAAK,GAAG,UAAU,UAAU,MAAM,OAAO,CAAC,QAAQ;UACnD,SAAS,MAAM;UACf;UACA;UACA,QAAQ;UACR,UAAU,MAAM;AACd;AACA,gBAAG,YAAY,GAAE;AAAE,uBAAS;YAAE;UAChC;QACF,CAAC,CAAC;MACJ,GAAG,aAAa,WAAW;IAC7B;IAEA,cAAc,GAAG,MAAM,UAAU,UAAS;AACxC,UAAI,UAAU,KAAK,WAAW,eAAe,IAAI;AAGjD,UAAI,UAAU,EAAE,aAAa,EAAE,SAAS;AACxC,UAAI,SAAS,WAAW,MAAM,KAAK,OAAO,CAAC,EAAC,IAAI,UAAU,SAAkB,MAAM,KAAI,CAAC,GAAG,MAAM,OAAO,IAAI;AAC3G,UAAI,WAAW,MAAM,KAAK,WAAW,SAAS,OAAO,SAAS,IAAI;AAClE,UAAI,MAAM,KAAK,WAAW,GAAG,IAAI,GAAG,SAAS,aAAa,SAAS,OAAO,SAAS;AAEnF,WAAK,cAAc,QAAQ,cAAc,EAAC,IAAG,CAAC,EAAE;QAC9C,CAAC,EAAC,KAAI,MAAM;AACV,eAAK,WAAW,iBAAiB,MAAM;AACrC,gBAAG,KAAK,eAAc;AACpB,mBAAK,WAAW,YAAY,MAAM,MAAM,UAAU,OAAO;YAC3D,OAAO;AACL,kBAAG,KAAK,WAAW,kBAAkB,OAAO,GAAE;AAC5C,qBAAK,OAAO;cACd;AACA,mBAAK,oBAAoB;AACzB,0BAAY,SAAS,OAAO;YAC9B;UACF,CAAC;QACH;QACA,CAAC,EAAC,OAAO,QAAQ,SAAS,SAAQ,MAAM,SAAS;MACnD;IACF;IAEA,sBAAqB;AACnB,UAAG,KAAK,cAAc,GAAE;AAAE,eAAO,CAAC;MAAE;AAEpC,UAAI,YAAY,KAAK,QAAQ,QAAQ;AAErC,aAAO,YAAI,IAAI,KAAK,IAAI,QAAQ,YAAY,EACzC,OAAO,CAAA,SAAQ,KAAK,EAAE,EACtB,OAAO,CAAA,SAAQ,KAAK,SAAS,SAAS,CAAC,EACvC,OAAO,CAAA,SAAQ,KAAK,aAAa,KAAK,QAAQ,gBAAgB,CAAC,MAAM,QAAQ,EAC7E,IAAI,CAAA,SAAQ;AAEX,cAAM,aAAa,KAAK,UAAU,KAAK;AAGvC,oBAAI,aAAa,YAAY,IAAI;AACjC,cAAM,KAAK,KAAK,QAAQ,EAAE,QAAQ,CAAC,OAAO;AAGxC,gBAAM,WAAW,GAAG,UAAU,IAAI;AAKlC,+BAAS,UAAU,EAAE;AACrB,sBAAI,aAAa,UAAU,EAAE;AAC7B,qBAAW,YAAY,QAAQ;QACjC,CAAC;AACD,eAAO;MACT,CAAC,EACA,OAAO,CAAC,KAAK,SAAS;AACrB,YAAI,KAAK,EAAE,IAAI;AACf,eAAO;MACT,GAAG,CAAC,CAAC;IACT;IAEA,6BAA6B,eAAc;AACzC,UAAI,kBAAkB,cAAc,OAAO,CAAA,QAAO;AAChD,eAAO,YAAI,sBAAsB,KAAK,IAAI,GAAG,EAAE,WAAW;MAC5D,CAAC;AAED,UAAG,gBAAgB,SAAS,GAAE;AAG5B,wBAAgB,QAAQ,CAAA,QAAO,KAAK,SAAS,YAAY,GAAG,CAAC;AAE7D,aAAK,cAAc,MAAM,qBAAqB,EAAC,MAAM,gBAAe,CAAC,EAAE,KAAK,MAAM;AAGhF,eAAK,WAAW,iBAAiB,MAAM;AAGrC,gBAAI,wBAAwB,gBAAgB,OAAO,CAAA,QAAO;AACxD,qBAAO,YAAI,sBAAsB,KAAK,IAAI,GAAG,EAAE,WAAW;YAC5D,CAAC;AAED,gBAAG,sBAAsB,SAAS,GAAE;AAClC,mBAAK,cAAc,MAAM,kBAAkB,EAAC,MAAM,sBAAqB,CAAC,EAAE,KAAK,CAAC,EAAC,KAAI,MAAM;AACzF,qBAAK,SAAS,UAAU,KAAK,IAAI;cACnC,CAAC,EAAE,MAAM,CAAC,UAAU,SAAS,uCAAuC,KAAK,CAAC;YAC5E;UACF,CAAC;QACH,CAAC,EAAE,MAAM,CAAC,UAAU,SAAS,uCAAuC,KAAK,CAAC;MAC5E;IACF;IAEA,YAAY,IAAG;AACb,UAAI,eAAe,GAAG,QAAQ,iBAAiB;AAC/C,aAAO,GAAG,aAAa,aAAa,MAAM,KAAK,MAC5C,gBAAgB,aAAa,OAAO,KAAK,MACzC,CAAC,gBAAgB,KAAK;IAC3B;IAEA,WAAW,MAAM,WAAW,UAAU,WAAW,OAAO,CAAC,GAAE;AACzD,kBAAI,WAAW,MAAM,mBAAmB,IAAI;AAC5C,YAAM,SAAS,MAAM,KAAK,KAAK,QAAQ;AACvC,aAAO,QAAQ,CAAA,UAAS,YAAI,WAAW,OAAO,mBAAmB,IAAI,CAAC;AACtE,WAAK,WAAW,kBAAkB,IAAI;AACtC,WAAK,eAAe,MAAM,WAAW,UAAU,WAAW,MAAM,MAAM;AACpE,aAAK,WAAW,6BAA6B;MAC/C,CAAC;IACH;IAEA,QAAQ,MAAK;AAAE,aAAO,KAAK,WAAW,QAAQ,IAAI;IAAE;EACtD;AC98CA,MAAqB,aAArB,MAAgC;IAC9B,YAAY,KAAK,WAAW,OAAO,CAAC,GAAE;AACpC,WAAK,WAAW;AAChB,UAAG,CAAC,aAAa,UAAU,YAAY,SAAS,UAAS;AACvD,cAAM,IAAI,MAAM;;;;;;OAMf;MACH;AACA,WAAK,SAAS,IAAI,UAAU,KAAK,IAAI;AACrC,WAAK,gBAAgB,KAAK,iBAAiB;AAC3C,WAAK,OAAO;AACZ,WAAK,SAASC,SAAQ,KAAK,UAAU,CAAC,CAAC;AACvC,WAAK,aAAa,KAAK;AACvB,WAAK,oBAAoB,KAAK,YAAY,CAAC;AAC3C,WAAK,WAAW,OAAO,OAAO,MAAM,QAAQ,GAAG,KAAK,YAAY,CAAC,CAAC;AAClE,WAAK,gBAAgB;AACrB,WAAK,aAAa;AAClB,WAAK,WAAW;AAChB,WAAK,OAAO;AACZ,WAAK,iBAAiB;AACtB,WAAK,uBAAuB;AAC5B,WAAK,UAAU;AACf,WAAK,QAAQ,CAAC;AACd,WAAK,OAAO,OAAO,SAAS;AAC5B,WAAK,cAAc;AACnB,WAAK,kBAAkB,MAAM,OAAO,QAAQ;AAC5C,WAAK,QAAQ,KAAK,SAAS,CAAC;AAC5B,WAAK,YAAY,KAAK,aAAa,CAAC;AACpC,WAAK,gBAAgB,KAAK,iBAAiB;AAC3C,WAAK,sBAAsB,KAAK,uBAAuB;AACvD,WAAK,wBAAwB;AAC7B,WAAK,aAAa,KAAK,cAAc;AACrC,WAAK,kBAAkB,KAAK,mBAAmB;AAC/C,WAAK,kBAAkB,KAAK,mBAAmB;AAC/C,WAAK,iBAAiB,KAAK,kBAAkB;AAC7C,WAAK,eAAe,KAAK,gBAAgB,OAAO;AAChD,WAAK,iBAAiB,KAAK,kBAAkB,OAAO;AACpD,WAAK,sBAAsB;AAC3B,WAAK,kBAAkB,oBAAI,IAAI;AAC/B,WAAK,iBAAiB;AACtB,WAAK,eAAe,OAAO;QAAO;UAChC,oBAAoB;UACpB,cAAcA,SAAQ;UACtB,YAAYA,SAAQ;UACpB,aAAaA,SAAQ;UACrB,mBAAmBA,SAAQ;QAAC;QAC9B,KAAK,OAAO,CAAC;MAAC;AACd,WAAK,cAAc,IAAI,cAAc;AACrC,WAAK,yBAAyB,SAAS,KAAK,eAAe,QAAQ,uBAAuB,CAAC,KAAK;AAChG,aAAO,iBAAiB,YAAY,CAAA,OAAM;AACxC,aAAK,WAAW;MAClB,CAAC;AACD,WAAK,OAAO,OAAO,MAAM;AACvB,YAAG,KAAK,WAAW,GAAE;AAEnB,iBAAO,SAAS,OAAO;QACzB;MACF,CAAC;IACH;;IAIA,UAAS;AAAE,aAAO;IAAO;IAEzB,mBAAkB;AAAE,aAAO,KAAK,eAAe,QAAQ,cAAc,MAAM;IAAO;IAElF,iBAAgB;AAAE,aAAO,KAAK,eAAe,QAAQ,YAAY,MAAM;IAAO;IAE9E,kBAAiB;AAAE,aAAO,KAAK,eAAe,QAAQ,YAAY,MAAM;IAAQ;IAEhF,cAAa;AAAE,WAAK,eAAe,QAAQ,cAAc,MAAM;IAAE;IAEjE,kBAAiB;AAAE,WAAK,eAAe,QAAQ,gBAAgB,MAAM;IAAE;IAEvE,eAAc;AAAE,WAAK,eAAe,QAAQ,cAAc,OAAO;IAAE;IAEnE,mBAAkB;AAAE,WAAK,eAAe,WAAW,cAAc;IAAE;IAEnE,iBAAiB,cAAa;AAC5B,WAAK,YAAY;AACjB,cAAQ,IAAI,yGAAyG;AACrH,WAAK,eAAe,QAAQ,oBAAoB,YAAY;IAC9D;IAEA,oBAAmB;AAAE,WAAK,eAAe,WAAW,kBAAkB;IAAE;IAExE,gBAAe;AACb,UAAI,MAAM,KAAK,eAAe,QAAQ,kBAAkB;AACxD,aAAO,MAAM,SAAS,GAAG,IAAI;IAC/B;IAEA,YAAW;AAAE,aAAO,KAAK;IAAO;IAEhC,UAAS;AAEP,UAAG,OAAO,SAAS,aAAa,eAAe,CAAC,KAAK,gBAAgB,GAAE;AAAE,aAAK,YAAY;MAAE;AAC5F,UAAI,YAAY,MAAM;AACpB,aAAK,kBAAkB;AACvB,YAAG,KAAK,cAAc,GAAE;AACtB,eAAK,mBAAmB;AACxB,eAAK,OAAO,QAAQ;QACtB,WAAU,KAAK,MAAK;AAClB,eAAK,OAAO,QAAQ;QACtB,OAAO;AACL,eAAK,mBAAmB,EAAC,MAAM,KAAI,CAAC;QACtC;AACA,aAAK,aAAa;MACpB;AACA,UAAG,CAAC,YAAY,UAAU,aAAa,EAAE,QAAQ,SAAS,UAAU,KAAK,GAAE;AACzE,kBAAU;MACZ,OAAO;AACL,iBAAS,iBAAiB,oBAAoB,MAAM,UAAU,CAAC;MACjE;IACF;IAEA,WAAW,UAAS;AAClB,mBAAa,KAAK,qBAAqB;AAGvC,UAAG,KAAK,gBAAe;AACrB,aAAK,OAAO,IAAI,KAAK,cAAc;AACnC,aAAK,iBAAiB;MACxB;AACA,WAAK,OAAO,WAAW,QAAQ;IACjC;IAEA,iBAAiB,WAAU;AACzB,mBAAa,KAAK,qBAAqB;AACvC,WAAK,OAAO,iBAAiB,SAAS;AACtC,WAAK,QAAQ;IACf;IAEA,OAAO,IAAI,WAAW,YAAY,MAAK;AACrC,UAAI,IAAI,IAAI,YAAY,YAAY,EAAC,QAAQ,EAAC,eAAe,GAAE,EAAC,CAAC;AACjE,WAAK,MAAM,IAAI,CAAA,SAAQ,WAAG,KAAK,GAAG,WAAW,WAAW,MAAM,EAAE,CAAC;IACnE;;IAIA,eAAe,IAAI,UAAU,MAAM,UAAS;AAC1C,WAAK,aAAa,IAAI,CAAA,SAAQ;AAC5B,YAAI,IAAI,IAAI,YAAY,YAAY,EAAC,QAAQ,EAAC,eAAe,GAAE,EAAC,CAAC;AACjE,mBAAG,KAAK,GAAG,QAAQ,UAAU,MAAM,IAAI,CAAC,QAAQ,EAAC,MAAM,SAAQ,CAAC,CAAC;MACnE,CAAC;IACH;IAEA,SAAQ;AACN,UAAG,KAAK,UAAS;AAAE;MAAO;AAC1B,UAAG,KAAK,QAAQ,KAAK,YAAY,GAAE;AAAE,aAAK,IAAI,KAAK,MAAM,UAAU,MAAM,CAAC,yBAAyB,CAAC;MAAE;AACtG,WAAK,WAAW;AAChB,WAAK,gBAAgB;AACrB,WAAK,WAAW;IAClB;IAEA,WAAW,MAAM,MAAK;AAAE,WAAK,aAAa,IAAI,EAAE,GAAG,IAAI;IAAE;IAEzD,KAAK,MAAM,MAAK;AACd,UAAG,CAAC,KAAK,iBAAiB,KAAK,CAAC,QAAQ,MAAK;AAAE,eAAO,KAAK;MAAE;AAC7D,cAAQ,KAAK,IAAI;AACjB,UAAI,SAAS,KAAK;AAClB,cAAQ,QAAQ,IAAI;AACpB,aAAO;IACT;IAEA,IAAI,MAAM,MAAM,aAAY;AAC1B,UAAG,KAAK,YAAW;AACjB,YAAI,CAAC,KAAK,GAAG,IAAI,YAAY;AAC7B,aAAK,WAAW,MAAM,MAAM,KAAK,GAAG;MACtC,WAAU,KAAK,eAAe,GAAE;AAC9B,YAAI,CAAC,KAAK,GAAG,IAAI,YAAY;AAC7B,cAAM,MAAM,MAAM,KAAK,GAAG;MAC5B;IACF;IAEA,iBAAiB,UAAS;AACxB,WAAK,YAAY,MAAM,QAAQ;IACjC;IAEA,WAAW,MAAM,SAAS,SAAS,WAAU;IAAC,GAAE;AAC9C,WAAK,YAAY,cAAc,MAAM,SAAS,MAAM;IACtD;IAEA,UAAU,SAAS,OAAO,IAAG;AAC3B,cAAQ,GAAG,OAAO,CAAA,SAAQ;AACxB,YAAI,UAAU,KAAK,cAAc;AACjC,YAAG,CAAC,SAAQ;AACV,aAAG,IAAI;QACT,OAAO;AACL,qBAAW,MAAM,GAAG,IAAI,GAAG,OAAO;QACpC;MACF,CAAC;IACH;IAEA,iBAAiB,MAAM,KAAI;AACzB,mBAAa,KAAK,qBAAqB;AACvC,WAAK,WAAW;AAChB,UAAI,QAAQ,KAAK;AACjB,UAAI,QAAQ,KAAK;AACjB,UAAI,UAAU,KAAK,MAAM,KAAK,OAAO,KAAK,QAAQ,QAAQ,EAAE,IAAI;AAChE,UAAI,QAAQ,gBAAQ,YAAY,KAAK,cAAc,OAAO,SAAS,UAAU,qBAAqB,GAAG,CAAA,UAAS,QAAQ,CAAC;AACvH,UAAG,SAAS,KAAK,YAAW;AAC1B,kBAAU,KAAK;MACjB;AACA,WAAK,wBAAwB,WAAW,MAAM;AAE5C,YAAG,KAAK,YAAY,KAAK,KAAK,YAAY,GAAE;AAAE;QAAO;AACrD,aAAK,QAAQ;AACb,cAAM,IAAI,IAAI,KAAK,IAAI,MAAM,QAAQ,MAAM,CAAC,eAAe,2BAA2B,CAAC;AACvF,YAAG,SAAS,KAAK,YAAW;AAC1B,eAAK,IAAI,MAAM,QAAQ,MAAM,CAAC,YAAY,KAAK,wDAAwD,CAAC;QAC1G;AACA,YAAG,KAAK,eAAe,GAAE;AACvB,iBAAO,WAAW,KAAK;QACzB,OAAO;AACL,iBAAO,SAAS,OAAO;QACzB;MACF,GAAG,OAAO;IACZ;IAEA,iBAAiB,MAAK;AACpB,aAAO,QAAQ,KAAK,WAAW,UAAU,IAAI,cAAM,KAAK,MAAM,GAAG,EAAE,CAAC,CAAC,IAAI,KAAK,MAAM,IAAI;IAC1F;IAEA,aAAY;AAAE,aAAO,KAAK;IAAS;IAEnC,cAAa;AAAE,aAAO,KAAK,OAAO,YAAY;IAAE;IAEhD,mBAAkB;AAAE,aAAO,KAAK;IAAc;IAE9C,QAAQ,MAAK;AAAE,aAAO,GAAG,KAAK,iBAAiB,IAAI;IAAO;IAE1D,QAAQ,OAAO,QAAO;AAAE,aAAO,KAAK,OAAO,QAAQ,OAAO,MAAM;IAAE;IAElE,eAAc;AACZ,UAAI,OAAO,SAAS;AACpB,UAAG,QAAQ,CAAC,KAAK,UAAU,IAAI,KAAK,CAAC,KAAK,UAAU,SAAS,iBAAiB,GAAE;AAC9E,YAAI,OAAO,KAAK,YAAY,IAAI;AAChC,aAAK,QAAQ,KAAK,QAAQ,CAAC;AAC3B,aAAK,SAAS;AACd,YAAG,CAAC,KAAK,MAAK;AAAE,eAAK,OAAO;QAAK;AACjC,eAAO,sBAAsB,MAAM;;AACjC,eAAK,eAAe;AAEpB,eAAK,aAAY,aAAQ,UAAR,mBAAe,MAAM;QACxC,CAAC;MACH;IACF;IAEA,gBAAe;AACb,UAAI,aAAa;AACjB,kBAAI,IAAI,UAAU,GAAG,0BAA0B,mBAAmB,CAAA,WAAU;AAC1E,YAAG,CAAC,KAAK,YAAY,OAAO,EAAE,GAAE;AAC9B,cAAI,OAAO,KAAK,YAAY,MAAM;AAGlC,cAAG,CAAC,YAAI,YAAY,MAAM,GAAE;AAAE,iBAAK,QAAQ,KAAK,QAAQ,CAAC;UAAE;AAC3D,eAAK,KAAK;AACV,cAAG,OAAO,aAAa,QAAQ,GAAE;AAAE,iBAAK,OAAO;UAAK;QACtD;AACA,qBAAa;MACf,CAAC;AACD,aAAO;IACT;IAEA,SAAS,IAAI,OAAO,aAAY;AAC9B,UAAG,aAAY;AAAE,wBAAQ,UAAU,mBAAmB,aAAa,EAAE;MAAE;AACvE,WAAK,OAAO;AACZ,sBAAQ,SAAS,IAAI,KAAK;IAC5B;IAEA,YAAY,MAAM,OAAO,WAAW,MAAM,UAAU,KAAK,eAAe,IAAI,GAAE;AAC5E,YAAM,cAAc,KAAK,gBAAgB;AACzC,WAAK,iBAAiB,KAAK,kBAAkB,KAAK,KAAK;AAEvD,YAAM,WAAW,YAAI,cAAc,QAAQ,KAAK,CAAC;AACjD,YAAM,YAAY,YAAI,IAAI,KAAK,gBAAgB,IAAI,KAAK,QAAQ,QAAQ,IAAI,EACzE,OAAO,CAAA,OAAM,CAAC,YAAI,aAAa,IAAI,QAAQ,CAAC;AAE/C,YAAM,YAAY,YAAI,UAAU,KAAK,gBAAgB,EAAE;AACvD,WAAK,KAAK,WAAW,KAAK,aAAa;AACvC,WAAK,KAAK,QAAQ;AAElB,WAAK,OAAO,KAAK,YAAY,WAAW,OAAO,WAAW;AAC1D,WAAK,KAAK,YAAY,IAAI;AAC1B,WAAK,kBAAkB,SAAS;AAChC,WAAK,KAAK,KAAK,CAAC,WAAW,WAAW;AACpC,YAAG,cAAc,KAAK,KAAK,kBAAkB,OAAO,GAAE;AACpD,eAAK,iBAAiB,MAAM;AAE1B,sBAAU,QAAQ,CAAA,OAAM,GAAG,OAAO,CAAC;AACnC,qBAAS,QAAQ,CAAA,OAAM,UAAU,YAAY,EAAE,CAAC;AAChD,iBAAK,eAAe,YAAY,SAAS;AACzC,iBAAK,iBAAiB;AACtB,wBAAY,SAAS,OAAO;AAC5B,mBAAO;UACT,CAAC;QACH;MACF,CAAC;IACH;IAEA,kBAAkB,UAAU,UAAS;AACnC,UAAI,aAAa,KAAK,QAAQ,QAAQ;AACtC,UAAI,gBAAgB,CAAC,MAAM;AACzB,UAAE,eAAe;AACjB,UAAE,yBAAyB;MAC7B;AACA,eAAS,QAAQ,CAAA,OAAM;AAGrB,iBAAQ,SAAS,KAAK,iBAAgB;AACpC,aAAG,iBAAiB,OAAO,eAAe,IAAI;QAChD;AACA,aAAK,OAAO,IAAI,GAAG,aAAa,UAAU,GAAG,QAAQ;MACvD,CAAC;AAGD,WAAK,iBAAiB,MAAM;AAC1B,iBAAS,QAAQ,CAAA,OAAM;AACrB,mBAAQ,SAAS,KAAK,iBAAgB;AACpC,eAAG,oBAAoB,OAAO,eAAe,IAAI;UACnD;QACF,CAAC;AACD,oBAAY,SAAS;MACvB,CAAC;IACH;IAEA,UAAU,IAAG;AAAE,aAAO,GAAG,gBAAgB,GAAG,aAAa,WAAW,MAAM;IAAK;IAE/E,YAAY,IAAI,OAAO,aAAY;AACjC,UAAI,OAAO,IAAI,KAAK,IAAI,MAAM,MAAM,OAAO,WAAW;AACtD,WAAK,MAAM,KAAK,EAAE,IAAI;AACtB,aAAO;IACT;IAEA,MAAM,SAAS,UAAS;AACtB,UAAI;AACJ,YAAM,gBAAgB,QAAQ,QAAQ,iBAAiB;AACvD,UAAG,eAAc;AAGf,eAAO,KAAK,YAAY,aAAa;MACvC,OAAO;AACL,eAAO,KAAK;MACd;AACA,aAAO,QAAQ,WAAW,SAAS,IAAI,IAAI;IAC7C;IAEA,aAAa,SAAS,UAAS;AAC7B,WAAK,MAAM,SAAS,CAAA,SAAQ,SAAS,MAAM,OAAO,CAAC;IACrD;IAEA,YAAY,IAAG;AACb,UAAI,SAAS,GAAG,aAAa,WAAW;AACxC,aAAO,MAAM,KAAK,YAAY,MAAM,GAAG,CAAA,SAAQ,KAAK,kBAAkB,EAAE,CAAC;IAC3E;IAEA,YAAY,IAAG;AAAE,aAAO,KAAK,MAAM,EAAE;IAAE;IAEvC,kBAAiB;AACf,eAAQ,MAAM,KAAK,OAAM;AACvB,aAAK,MAAM,EAAE,EAAE,QAAQ;AACvB,eAAO,KAAK,MAAM,EAAE;MACtB;AACA,WAAK,OAAO;IACd;IAEA,gBAAgB,IAAG;AACjB,UAAI,OAAO,KAAK,YAAY,GAAG,aAAa,WAAW,CAAC;AACxD,UAAG,QAAQ,KAAK,OAAO,GAAG,IAAG;AAC3B,aAAK,QAAQ;AACb,eAAO,KAAK,MAAM,KAAK,EAAE;MAC3B,WAAU,MAAK;AACb,aAAK,kBAAkB,GAAG,EAAE;MAC9B;IACF;IAEA,mBAAkB;AAChB,aAAO,SAAS;IAClB;IAEA,kBAAkB,MAAK;AACrB,UAAG,KAAK,cAAc,KAAK,YAAY,KAAK,UAAU,GAAE;AACtD,aAAK,aAAa;MACpB;IACF;IAEA,+BAA8B;AAC5B,UAAG,KAAK,cAAc,KAAK,eAAe,SAAS,MAAK;AACtD,aAAK,WAAW,MAAM;MACxB;IACF;IAEA,oBAAmB;AACjB,WAAK,aAAa,KAAK,iBAAiB;AACxC,UAAG,KAAK,eAAe,SAAS,MAAK;AAAE,aAAK,WAAW,KAAK;MAAE;IAChE;IAEA,mBAAmB,EAAC,KAAI,IAAI,CAAC,GAAE;AAC7B,UAAG,KAAK,qBAAoB;AAAE;MAAO;AAErC,WAAK,sBAAsB;AAE3B,WAAK,iBAAiB,KAAK,OAAO,QAAQ,CAAA,UAAS;AAEjD,YAAG,SAAS,MAAM,SAAS,OAAQ,KAAK,MAAK;AAAE,iBAAO,KAAK,iBAAiB,KAAK,IAAI;QAAE;MACzF,CAAC;AACD,eAAS,KAAK,iBAAiB,SAAS,WAAW;MAAE,CAAC;AACtD,aAAO,iBAAiB,YAAY,CAAA,MAAK;AACvC,YAAG,EAAE,WAAU;AACb,eAAK,UAAU,EAAE,WAAW;AAC5B,eAAK,gBAAgB,EAAC,IAAI,OAAO,SAAS,MAAM,MAAM,WAAU,CAAC;AACjE,iBAAO,SAAS,OAAO;QACzB;MACF,GAAG,IAAI;AACP,UAAG,CAAC,MAAK;AAAE,aAAK,QAAQ;MAAE;AAC1B,WAAK,WAAW;AAChB,UAAG,CAAC,MAAK;AAAE,aAAK,UAAU;MAAE;AAC5B,WAAK,KAAK,EAAC,OAAO,SAAS,SAAS,UAAS,GAAG,CAAC,GAAG,MAAM,MAAM,UAAU,UAAU,eAAe;AACjG,YAAI,WAAW,SAAS,aAAa,KAAK,QAAQ,OAAO,CAAC;AAC1D,YAAI,aAAa,EAAE,OAAO,EAAE,IAAI,YAAY;AAC5C,YAAG,YAAY,SAAS,YAAY,MAAM,YAAW;AAAE;QAAO;AAE9D,YAAI,OAAO,iBAAC,KAAK,EAAE,OAAQ,KAAK,UAAU,MAAM,GAAG,QAAQ;AAC3D,mBAAG,KAAK,GAAG,MAAM,UAAU,MAAM,UAAU,CAAC,QAAQ,EAAC,KAAI,CAAC,CAAC;MAC7D,CAAC;AACD,WAAK,KAAK,EAAC,MAAM,YAAY,OAAO,UAAS,GAAG,CAAC,GAAG,MAAM,MAAM,UAAU,UAAU,cAAc;AAChG,YAAG,CAAC,WAAU;AACZ,cAAI,OAAO,iBAAC,KAAK,EAAE,OAAQ,KAAK,UAAU,MAAM,GAAG,QAAQ;AAC3D,qBAAG,KAAK,GAAG,MAAM,UAAU,MAAM,UAAU,CAAC,QAAQ,EAAC,KAAI,CAAC,CAAC;QAC7D;MACF,CAAC;AACD,WAAK,KAAK,EAAC,MAAM,QAAQ,OAAO,QAAO,GAAG,CAAC,GAAG,MAAM,MAAM,UAAU,UAAU,cAAc;AAE1F,YAAG,cAAc,UAAS;AACxB,cAAI,OAAO,KAAK,UAAU,MAAM,GAAG,QAAQ;AAC3C,qBAAG,KAAK,GAAG,MAAM,UAAU,MAAM,UAAU,CAAC,QAAQ,EAAC,KAAI,CAAC,CAAC;QAC7D;MACF,CAAC;AACD,WAAK,GAAG,YAAY,CAAA,MAAK,EAAE,eAAe,CAAC;AAC3C,WAAK,GAAG,QAAQ,CAAA,MAAK;AACnB,UAAE,eAAe;AACjB,YAAI,eAAe,MAAM,kBAAkB,EAAE,QAAQ,KAAK,QAAQ,eAAe,CAAC,GAAG,CAAA,eAAc;AACjG,iBAAO,WAAW,aAAa,KAAK,QAAQ,eAAe,CAAC;QAC9D,CAAC;AACD,YAAI,aAAa,gBAAgB,SAAS,eAAe,YAAY;AACrE,YAAI,QAAQ,MAAM,KAAK,EAAE,aAAa,SAAS,CAAC,CAAC;AACjD,YAAG,CAAC,cAAc,WAAW,YAAY,MAAM,WAAW,KAAK,EAAE,WAAW,iBAAiB,WAAU;AAAE;QAAO;AAEhH,qBAAa,WAAW,YAAY,OAAO,EAAE,YAAY;AACzD,mBAAW,cAAc,IAAI,MAAM,SAAS,EAAC,SAAS,KAAI,CAAC,CAAC;MAC9D,CAAC;AACD,WAAK,GAAG,mBAAmB,CAAA,MAAK;AAC9B,YAAI,eAAe,EAAE;AACrB,YAAG,CAAC,YAAI,cAAc,YAAY,GAAE;AAAE;QAAO;AAC7C,YAAI,QAAQ,MAAM,KAAK,EAAE,OAAO,SAAS,CAAC,CAAC,EAAE,OAAO,CAAA,MAAK,aAAa,QAAQ,aAAa,IAAI;AAC/F,qBAAa,WAAW,cAAc,KAAK;AAC3C,qBAAa,cAAc,IAAI,MAAM,SAAS,EAAC,SAAS,KAAI,CAAC,CAAC;MAChE,CAAC;IACH;IAEA,UAAU,WAAW,GAAG,UAAS;AAC/B,UAAI,WAAW,KAAK,kBAAkB,SAAS;AAC/C,aAAO,WAAW,SAAS,GAAG,QAAQ,IAAI,CAAC;IAC7C;IAEA,eAAe,MAAK;AAClB,WAAK;AACL,WAAK,cAAc;AACnB,WAAK,kBAAkB;AACvB,aAAO,KAAK;IACd;;;IAIA,oBAAmB;AAAE,sBAAQ,aAAa,iBAAiB;IAAE;IAE7D,kBAAkB,SAAQ;AACxB,UAAG,KAAK,YAAY,SAAQ;AAC1B,eAAO;MACT,OAAO;AACL,aAAK,OAAO,KAAK;AACjB,aAAK,cAAc;AACnB,eAAO;MACT;IACF;IAEA,UAAS;AAAE,aAAO,KAAK;IAAK;IAE5B,iBAAgB;AAAE,aAAO,CAAC,CAAC,KAAK;IAAY;IAE5C,KAAK,QAAQ,UAAS;AACpB,eAAQ,SAAS,QAAO;AACtB,YAAI,mBAAmB,OAAO,KAAK;AAEnC,aAAK,GAAG,kBAAkB,CAAA,MAAK;AAC7B,cAAI,UAAU,KAAK,QAAQ,KAAK;AAChC,cAAI,gBAAgB,KAAK,QAAQ,UAAU,OAAO;AAClD,cAAI,iBAAiB,EAAE,OAAO,gBAAgB,EAAE,OAAO,aAAa,OAAO;AAC3E,cAAG,gBAAe;AAChB,iBAAK,SAAS,EAAE,QAAQ,GAAG,kBAAkB,MAAM;AACjD,mBAAK,aAAa,EAAE,QAAQ,CAAA,SAAQ;AAClC,yBAAS,GAAG,OAAO,MAAM,EAAE,QAAQ,gBAAgB,IAAI;cACzD,CAAC;YACH,CAAC;UACH,OAAO;AACL,wBAAI,IAAI,UAAU,IAAI,kBAAkB,CAAA,OAAM;AAC5C,kBAAI,WAAW,GAAG,aAAa,aAAa;AAC5C,mBAAK,SAAS,IAAI,GAAG,kBAAkB,MAAM;AAC3C,qBAAK,aAAa,IAAI,CAAA,SAAQ;AAC5B,2BAAS,GAAG,OAAO,MAAM,IAAI,UAAU,QAAQ;gBACjD,CAAC;cACH,CAAC;YACH,CAAC;UACH;QACF,CAAC;MACH;IACF;IAEA,aAAY;AACV,WAAK,GAAG,aAAa,CAAA,MAAK,KAAK,uBAAuB,EAAE,MAAM;AAC9D,WAAK,UAAU,SAAS,OAAO;IACjC;IAEA,UAAU,WAAW,aAAY;AAC/B,UAAI,QAAQ,KAAK,QAAQ,WAAW;AACpC,aAAO,iBAAiB,WAAW,CAAA,MAAK;AACtC,YAAI,SAAS;AAGb,YAAG,EAAE,WAAW;AAAG,eAAK,uBAAuB,EAAE;AACjD,YAAI,uBAAuB,KAAK,wBAAwB,EAAE;AAG1D,iBAAS,kBAAkB,EAAE,QAAQ,KAAK;AAC1C,aAAK,kBAAkB,GAAG,oBAAoB;AAC9C,aAAK,uBAAuB;AAC5B,YAAI,WAAW,UAAU,OAAO,aAAa,KAAK;AAClD,YAAG,CAAC,UAAS;AACX,cAAG,YAAI,eAAe,GAAG,OAAO,QAAQ,GAAE;AAAE,iBAAK,OAAO;UAAE;AAC1D;QACF;AAEA,YAAG,OAAO,aAAa,MAAM,MAAM,KAAI;AAAE,YAAE,eAAe;QAAE;AAG5D,YAAG,OAAO,aAAa,WAAW,GAAE;AAAE;QAAO;AAE7C,aAAK,SAAS,QAAQ,GAAG,SAAS,MAAM;AACtC,eAAK,aAAa,QAAQ,CAAA,SAAQ;AAChC,uBAAG,KAAK,GAAG,SAAS,UAAU,MAAM,QAAQ,CAAC,QAAQ,EAAC,MAAM,KAAK,UAAU,SAAS,GAAG,MAAM,EAAC,CAAC,CAAC;UAClG,CAAC;QACH,CAAC;MACH,GAAG,KAAK;IACV;IAEA,kBAAkB,GAAG,gBAAe;AAClC,UAAI,eAAe,KAAK,QAAQ,YAAY;AAC5C,kBAAI,IAAI,UAAU,IAAI,iBAAiB,CAAA,OAAM;AAC3C,YAAG,EAAE,GAAG,WAAW,cAAc,KAAK,GAAG,SAAS,cAAc,IAAG;AACjE,eAAK,aAAa,IAAI,CAAA,SAAQ;AAC5B,gBAAI,WAAW,GAAG,aAAa,YAAY;AAC3C,gBAAG,WAAG,UAAU,EAAE,KAAK,WAAG,aAAa,EAAE,GAAE;AACzC,yBAAG,KAAK,GAAG,SAAS,UAAU,MAAM,IAAI,CAAC,QAAQ,EAAC,MAAM,KAAK,UAAU,SAAS,GAAG,EAAE,MAAM,EAAC,CAAC,CAAC;YAChG;UACF,CAAC;QACH;MACF,CAAC;IACH;IAEA,UAAS;AACP,UAAG,CAAC,gBAAQ,aAAa,GAAE;AAAE;MAAO;AACpC,UAAG,QAAQ,mBAAkB;AAAE,gBAAQ,oBAAoB;MAAS;AACpE,UAAI,cAAc;AAClB,aAAO,iBAAiB,UAAU,CAAA,OAAM;AACtC,qBAAa,WAAW;AACxB,sBAAc,WAAW,MAAM;AAC7B,0BAAQ,mBAAmB,CAAA,UAAS,OAAO,OAAO,OAAO,EAAC,QAAQ,OAAO,QAAO,CAAC,CAAC;QACpF,GAAG,GAAG;MACR,CAAC;AACD,aAAO,iBAAiB,YAAY,CAAA,UAAS;AAC3C,YAAG,CAAC,KAAK,oBAAoB,OAAO,QAAQ,GAAE;AAAE;QAAO;AACvD,YAAI,EAAC,MAAM,UAAU,IAAI,QAAQ,SAAQ,IAAI,MAAM,SAAS,CAAC;AAC7D,YAAI,OAAO,OAAO,SAAS;AAG3B,YAAI,YAAY,WAAW,KAAK;AAEhC,eAAO,YAAY,OAAQ,YAAY;AAGvC,aAAK,yBAAyB,YAAY;AAC1C,aAAK,eAAe,QAAQ,yBAAyB,KAAK,uBAAuB,SAAS,CAAC;AAE3F,oBAAI,cAAc,QAAQ,gBAAgB,EAAC,QAAQ,EAAC,MAAM,OAAO,SAAS,SAAS,KAAK,MAAM,WAAW,YAAY,YAAY,WAAU,EAAC,CAAC;AAC7I,aAAK,iBAAiB,MAAM;AAC1B,gBAAM,WAAW,MAAM;AAAE,iBAAK,YAAY,MAAM;UAAE;AAClD,cAAG,KAAK,KAAK,YAAY,MAAM,SAAS,WAAW,OAAO,KAAK,KAAK,KAAI;AACtE,iBAAK,KAAK,cAAc,OAAO,MAAM,MAAM,QAAQ;UACrD,OAAO;AACL,iBAAK,YAAY,MAAM,MAAM,QAAQ;UACvC;QACF,CAAC;MACH,GAAG,KAAK;AACR,aAAO,iBAAiB,SAAS,CAAA,MAAK;AACpC,YAAI,SAAS,kBAAkB,EAAE,QAAQ,aAAa;AACtD,YAAI,OAAO,UAAU,OAAO,aAAa,aAAa;AACtD,YAAG,CAAC,QAAQ,CAAC,KAAK,YAAY,KAAK,CAAC,KAAK,QAAQ,YAAI,YAAY,CAAC,GAAE;AAAE;QAAO;AAG7E,YAAI,OAAO,OAAO,gBAAgB,oBAAoB,OAAO,KAAK,UAAU,OAAO;AAEnF,YAAI,YAAY,OAAO,aAAa,cAAc;AAClD,UAAE,eAAe;AACjB,UAAE,yBAAyB;AAC3B,YAAG,KAAK,gBAAgB,MAAK;AAAE;QAAO;AAEtC,aAAK,iBAAiB,MAAM;AAC1B,cAAG,SAAS,SAAQ;AAClB,iBAAK,iBAAiB,GAAG,MAAM,WAAW,MAAM;UAClD,WAAU,SAAS,YAAW;AAC5B,iBAAK,gBAAgB,GAAG,MAAM,WAAW,MAAM,MAAM;UACvD,OAAO;AACL,kBAAM,IAAI,MAAM,YAAY,mDAAmD,MAAM;UACvF;AACA,cAAI,WAAW,OAAO,aAAa,KAAK,QAAQ,OAAO,CAAC;AACxD,cAAG,UAAS;AACV,iBAAK,iBAAiB,MAAM,KAAK,OAAO,QAAQ,UAAU,OAAO,CAAC;UACpE;QACF,CAAC;MACH,GAAG,KAAK;IACV;IAEA,YAAY,QAAO;AACjB,UAAG,OAAO,WAAY,UAAS;AAC7B,8BAAsB,MAAM;AAC1B,iBAAO,SAAS,GAAG,MAAM;QAC3B,CAAC;MACH;IACF;IAEA,cAAc,OAAO,UAAU,CAAC,GAAE;AAChC,kBAAI,cAAc,QAAQ,OAAO,SAAS,EAAC,QAAQ,QAAO,CAAC;IAC7D;IAEA,eAAe,QAAO;AACpB,aAAO,QAAQ,CAAC,CAAC,OAAO,OAAO,MAAM,KAAK,cAAc,OAAO,OAAO,CAAC;IACzE;IAEA,gBAAgB,MAAM,UAAS;AAC7B,kBAAI,cAAc,QAAQ,0BAA0B,EAAC,QAAQ,KAAI,CAAC;AAClE,UAAI,OAAO,MAAM,YAAI,cAAc,QAAQ,yBAAyB,EAAC,QAAQ,KAAI,CAAC;AAClF,aAAO,WAAW,SAAS,IAAI,IAAI;IACrC;IAEA,iBAAiB,GAAG,MAAM,WAAW,UAAS;AAC5C,UAAG,CAAC,KAAK,YAAY,KAAK,CAAC,KAAK,KAAK,OAAO,GAAE;AAAE,eAAO,gBAAQ,SAAS,IAAI;MAAE;AAE9E,WAAK,gBAAgB,EAAC,IAAI,MAAM,MAAM,QAAO,GAAG,CAAA,SAAQ;AACtD,aAAK,KAAK,cAAc,GAAG,MAAM,UAAU,CAAA,YAAW;AACpD,eAAK,aAAa,MAAM,WAAW,OAAO;AAC1C,eAAK;QACP,CAAC;MACH,CAAC;IACH;IAEA,aAAa,MAAM,WAAW,UAAU,KAAK,eAAe,IAAI,GAAE;AAChE,UAAG,CAAC,KAAK,kBAAkB,OAAO,GAAE;AAAE;MAAO;AAG7C,WAAK;AACL,WAAK,eAAe,QAAQ,yBAAyB,KAAK,uBAAuB,SAAS,CAAC;AAG3F,sBAAQ,mBAAmB,CAAC,UAAW,iCAAI,QAAJ,EAAW,UAAU,QAAO,EAAE;AAErE,sBAAQ,UAAU,WAAW;QAC3B,MAAM;QACN,IAAI,KAAK,KAAK;QACd,UAAU,KAAK;MACjB,GAAG,IAAI;AAEP,kBAAI,cAAc,QAAQ,gBAAgB,EAAC,QAAQ,EAAC,OAAO,MAAM,MAAM,KAAK,OAAO,WAAW,UAAS,EAAC,CAAC;AACzG,WAAK,oBAAoB,OAAO,QAAQ;IAC1C;IAEA,gBAAgB,GAAG,MAAM,WAAW,OAAO,UAAS;AAClD,YAAM,eAAe,YAAY,EAAE,aAAa,EAAE,SAAS;AAC3D,UAAG,cAAa;AAAE,iBAAS,UAAU,IAAI,mBAAmB;MAAE;AAC9D,UAAG,CAAC,KAAK,YAAY,KAAK,CAAC,KAAK,KAAK,OAAO,GAAE;AAAE,eAAO,gBAAQ,SAAS,MAAM,KAAK;MAAE;AAGrF,UAAG,oBAAoB,KAAK,IAAI,GAAE;AAChC,YAAI,EAAC,UAAU,KAAI,IAAI,OAAO;AAC9B,eAAO,GAAG,aAAa,OAAO;MAChC;AACA,UAAI,SAAS,OAAO;AACpB,WAAK,gBAAgB,EAAC,IAAI,MAAM,MAAM,WAAU,GAAG,CAAA,SAAQ;AACzD,aAAK,YAAY,MAAM,OAAO,CAAC,YAAY;AACzC,cAAG,YAAY,KAAK,SAAQ;AAE1B,iBAAK;AACL,iBAAK,eAAe,QAAQ,yBAAyB,KAAK,uBAAuB,SAAS,CAAC;AAG3F,4BAAQ,mBAAmB,CAAC,UAAW,iCAAI,QAAJ,EAAW,UAAU,WAAU,EAAE;AAExE,4BAAQ,UAAU,WAAW;cAC3B,MAAM;cACN,IAAI,KAAK,KAAK;cACd;cACA,UAAU,KAAK;YACjB,GAAG,IAAI;AAEP,wBAAI,cAAc,QAAQ,gBAAgB,EAAC,QAAQ,EAAC,MAAM,OAAO,OAAO,KAAK,OAAO,WAAW,UAAS,EAAC,CAAC;AAC1G,iBAAK,oBAAoB,OAAO,QAAQ;UAC1C;AAGA,cAAG,cAAa;AAAE,qBAAS,UAAU,OAAO,mBAAmB;UAAE;AACjE,eAAK;QACP,CAAC;MACH,CAAC;IACH;IAEA,oBAAoB,aAAY;AAC9B,UAAI,EAAC,UAAU,OAAM,IAAI,KAAK;AAC9B,UAAG,WAAW,WAAW,YAAY,WAAW,YAAY,QAAO;AACjE,eAAO;MACT,OAAO;AACL,aAAK,kBAAkB,MAAM,WAAW;AACxC,eAAO;MACT;IACF;IAEA,YAAW;AACT,UAAI,aAAa;AACjB,UAAI,wBAAwB;AAG5B,WAAK,GAAG,UAAU,CAAA,MAAK;AACrB,YAAI,YAAY,EAAE,OAAO,aAAa,KAAK,QAAQ,QAAQ,CAAC;AAC5D,YAAI,YAAY,EAAE,OAAO,aAAa,KAAK,QAAQ,QAAQ,CAAC;AAC5D,YAAG,CAAC,yBAAyB,aAAa,CAAC,WAAU;AACnD,kCAAwB;AACxB,YAAE,eAAe;AACjB,eAAK,aAAa,EAAE,QAAQ,CAAA,SAAQ;AAClC,iBAAK,YAAY,EAAE,MAAM;AAEzB,mBAAO,sBAAsB,MAAM;AACjC,kBAAG,YAAI,uBAAuB,CAAC,GAAE;AAAE,qBAAK,OAAO;cAAE;AACjD,gBAAE,OAAO,OAAO;YAClB,CAAC;UACH,CAAC;QACH;MACF,CAAC;AAED,WAAK,GAAG,UAAU,CAAA,MAAK;AACrB,YAAI,WAAW,EAAE,OAAO,aAAa,KAAK,QAAQ,QAAQ,CAAC;AAC3D,YAAG,CAAC,UAAS;AACX,cAAG,YAAI,uBAAuB,CAAC,GAAE;AAAE,iBAAK,OAAO;UAAE;AACjD;QACF;AACA,UAAE,eAAe;AACjB,UAAE,OAAO,WAAW;AACpB,aAAK,aAAa,EAAE,QAAQ,CAAA,SAAQ;AAClC,qBAAG,KAAK,GAAG,UAAU,UAAU,MAAM,EAAE,QAAQ,CAAC,QAAQ,EAAC,WAAW,EAAE,UAAS,CAAC,CAAC;QACnF,CAAC;MACH,CAAC;AAED,eAAQ,QAAQ,CAAC,UAAU,OAAO,GAAE;AAClC,aAAK,GAAG,MAAM,CAAA,MAAK;AACjB,cAAG,aAAa,eAAe,EAAE,OAAO,SAAS,QAAU;AAEzD,gBAAG,EAAE,UAAU,EAAE,OAAO,YAAW;AACjC,oBAAM,IAAI,MAAM,wBAAwB,8DAA8D;YACxG;AACA;UACF;AACA,cAAI,YAAY,KAAK,QAAQ,QAAQ;AACrC,cAAI,QAAQ,EAAE;AAKd,cAAG,EAAE,aAAY;AACf,kBAAM,MAAM,wBAAwB;AACpC,gBAAG,CAAC,YAAI,QAAQ,OAAO,GAAG,GAAE;AAC1B,0BAAI,WAAW,OAAO,KAAK,IAAI;AAC/B,oBAAM,iBAAiB,kBAAkB,MAAM;AAE7C,sBAAM,cAAc,IAAI,MAAM,MAAM,EAAC,SAAS,KAAI,CAAC,CAAC;AACpD,4BAAI,cAAc,OAAO,GAAG;cAC9B,GAAG,EAAC,MAAM,KAAI,CAAC;YACjB;AACA;UACF;AACA,cAAI,aAAa,MAAM,aAAa,SAAS;AAC7C,cAAI,YAAY,MAAM,QAAQ,MAAM,KAAK,aAAa,SAAS;AAC/D,cAAI,WAAW,cAAc;AAC7B,cAAG,CAAC,UAAS;AAAE;UAAO;AACtB,cAAG,MAAM,SAAS,YAAY,MAAM,YAAY,MAAM,SAAS,UAAS;AAAE;UAAO;AAEjF,cAAI,aAAa,aAAa,QAAQ,MAAM;AAC5C,cAAI,oBAAoB;AACxB;AACA,cAAI,EAAC,IAAQ,MAAM,SAAQ,IAAI,YAAI,QAAQ,OAAO,gBAAgB,KAAK,CAAC;AAIxE,cAAG,OAAO,oBAAoB,KAAK,SAAS,YAAY,aAAa,SAAQ;AAAE;UAAO;AAEtF,sBAAI,WAAW,OAAO,kBAAkB,EAAC,IAAI,mBAAmB,KAAU,CAAC;AAE3E,eAAK,SAAS,OAAO,GAAG,MAAM,MAAM;AAClC,iBAAK,aAAa,YAAY,CAAA,SAAQ;AACpC,0BAAI,WAAW,OAAO,iBAAiB,IAAI;AAC3C,yBAAG,KAAK,GAAG,UAAU,UAAU,MAAM,OAAO,CAAC,QAAQ,EAAC,SAAS,EAAE,OAAO,MAAM,WAAsB,CAAC,CAAC;YACxG,CAAC;UACH,CAAC;QACH,CAAC;MACH;AACA,WAAK,GAAG,SAAS,CAAC,MAAM;AACtB,YAAI,OAAO,EAAE;AACb,oBAAI,UAAU,IAAI;AAClB,YAAI,QAAQ,MAAM,KAAK,KAAK,QAAQ,EAAE,KAAK,CAAA,OAAM,GAAG,SAAS,OAAO;AACpE,YAAG,OAAM;AAEP,iBAAO,sBAAsB,MAAM;AACjC,kBAAM,cAAc,IAAI,MAAM,SAAS,EAAC,SAAS,MAAM,YAAY,MAAK,CAAC,CAAC;UAC5E,CAAC;QACH;MACF,CAAC;IACH;IAEA,SAAS,IAAI,OAAO,WAAW,UAAS;AACtC,UAAG,cAAc,UAAU,cAAc,YAAW;AAAE,eAAO,SAAS;MAAE;AAExE,UAAI,cAAc,KAAK,QAAQ,YAAY;AAC3C,UAAI,cAAc,KAAK,QAAQ,YAAY;AAC3C,UAAI,kBAAkB,KAAK,SAAS,SAAS,SAAS;AACtD,UAAI,kBAAkB,KAAK,SAAS,SAAS,SAAS;AAEtD,WAAK,aAAa,IAAI,CAAA,SAAQ;AAC5B,YAAI,cAAc,MAAM,CAAC,KAAK,YAAY,KAAK,SAAS,KAAK,SAAS,EAAE;AACxE,oBAAI,SAAS,IAAI,OAAO,aAAa,iBAAiB,aAAa,iBAAiB,aAAa,MAAM;AACrG,mBAAS;QACX,CAAC;MACH,CAAC;IACH;IAEA,cAAc,UAAS;AACrB,WAAK,WAAW;AAChB,eAAS;AACT,WAAK,WAAW;IAClB;IAEA,GAAG,OAAO,UAAS;AACjB,WAAK,gBAAgB,IAAI,KAAK;AAC9B,aAAO,iBAAiB,OAAO,CAAA,MAAK;AAClC,YAAG,CAAC,KAAK,UAAS;AAAE,mBAAS,CAAC;QAAE;MAClC,CAAC;IACH;IAEA,mBAAmB,UAAU,OAAO,cAAa;AAC/C,UAAI,MAAM,KAAK,aAAa;AAC5B,aAAO,MAAM,IAAI,UAAU,OAAO,YAAY,IAAI,aAAa;IACjE;EACF;AAEA,MAAM,gBAAN,MAAoB;IAClB,cAAa;AACX,WAAK,cAAc,oBAAI,IAAI;AAC3B,WAAK,aAAa,CAAC;IACrB;IAEA,QAAO;AACL,WAAK,YAAY,QAAQ,CAAA,UAAS;AAChC,qBAAa,KAAK;AAClB,aAAK,YAAY,OAAO,KAAK;MAC/B,CAAC;AACD,WAAK,gBAAgB;IACvB;IAEA,MAAM,UAAS;AACb,UAAG,KAAK,KAAK,MAAM,GAAE;AACnB,iBAAS;MACX,OAAO;AACL,aAAK,cAAc,QAAQ;MAC7B;IACF;IAEA,cAAc,MAAM,SAAS,QAAO;AAClC,cAAQ;AACR,UAAI,QAAQ,WAAW,MAAM;AAC3B,aAAK,YAAY,OAAO,KAAK;AAC7B,eAAO;AACP,aAAK,gBAAgB;MACvB,GAAG,IAAI;AACP,WAAK,YAAY,IAAI,KAAK;IAC5B;IAEA,cAAc,IAAG;AAAE,WAAK,WAAW,KAAK,EAAE;IAAE;IAE5C,OAAM;AAAE,aAAO,KAAK,YAAY;IAAK;IAErC,kBAAiB;AACf,UAAG,KAAK,KAAK,IAAI,GAAE;AAAE;MAAO;AAC5B,UAAI,KAAK,KAAK,WAAW,MAAM;AAC/B,UAAG,IAAG;AACJ,WAAG;AACH,aAAK,gBAAgB;MACvB;IACF;EACF;;;AE3/BA,sBAAmB;AAEnB,MAAI,YAAY,SAAS,cAAc,yBAAyB,EAAE,aAAa,SAAS;AACxF,MAAI,aAAa,IAAI,WAAW,SAAS,QAAQ;AAAA,IAC/C,oBAAoB;AAAA,IACpB,QAAQ,EAAC,aAAa,UAAS;AAAA,EACjC,CAAC;AAGD,gBAAAC,QAAO,OAAO,EAAC,WAAW,EAAC,GAAG,OAAM,GAAG,aAAa,oBAAmB,CAAC;AACxE,SAAO,iBAAiB,0BAA0B,WAAS,cAAAA,QAAO,KAAK,GAAG,CAAC;AAC3E,SAAO,iBAAiB,yBAAyB,WAAS,cAAAA,QAAO,KAAK,CAAC;AAGvE,aAAW,QAAQ;AAMnB,SAAO,aAAa;",
  "names": ["window", "document", "topbar", "CustomEvent", "closure", "liveSocket", "closure", "e", "isEmpty", "file", "morphAttrs", "morphdom", "childrenOnly", "targetContainer", "clone", "el", "inputsUnused", "onlyHiddenInputs", "lock", "loading", "entry", "input", "closure", "topbar"]
}
