blonk is a radar for your web, where you follow vibes for cool blips on the radar

more docs, abandon typescript fully, move elixir in

Changed files
+11122 -1029
assets
config
elixir_blonk
lib
priv
test
+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
··· 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

This is a binary file and will not be displayed.

-6
elixir_blonk/.formatter.exs
··· 1 - [ 2 - import_deps: [:ecto, :ecto_sql, :phoenix], 3 - subdirectories: ["priv/*/migrations"], 4 - plugins: [Phoenix.LiveView.HTMLFormatter], 5 - inputs: ["*.{heex,ex,exs}", "{config,lib,test}/**/*.{heex,ex,exs}", "priv/*/seeds.exs"] 6 - ]
-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/js/app.js assets/js/app.js
elixir_blonk/assets/tailwind.config.js assets/tailwind.config.js
elixir_blonk/assets/vendor/topbar.js assets/vendor/topbar.js
elixir_blonk/config/config.exs config/config.exs
elixir_blonk/config/dev.exs config/dev.exs
elixir_blonk/config/prod.exs config/prod.exs
elixir_blonk/config/runtime.exs config/runtime.exs
elixir_blonk/config/test.exs config/test.exs
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/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_token.ex lib/elixir_blonk/accounts/user_token.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
··· 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/session_manager.ex lib/elixir_blonk/atproto/session_manager.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/blip_tags.ex lib/elixir_blonk/blip_tags.ex
elixir_blonk/lib/elixir_blonk/blip_tags/blip_tag.ex lib/elixir_blonk/blip_tags/blip_tag.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/comment.ex lib/elixir_blonk/blips/comment.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
-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
-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
··· 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
··· 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/repo.ex lib/elixir_blonk/repo.ex
elixir_blonk/lib/elixir_blonk/tags.ex lib/elixir_blonk/tags.ex
elixir_blonk/lib/elixir_blonk/tags/tag.ex lib/elixir_blonk/tags/tag.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_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_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/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/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/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/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/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_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/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_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_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/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_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/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_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/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/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.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.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.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.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/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/mix.exs mix.exs
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/errors.pot priv/gettext/errors.pot
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/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/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/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/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/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/20250620043347_create_tags.exs priv/repo/migrations/20250620043347_create_tags.exs
elixir_blonk/priv/repo/migrations/20250620043400_create_blip_tags.exs priv/repo/migrations/20250620043400_create_blip_tags.exs
elixir_blonk/priv/repo/migrations/20250620044014_update_tags_remove_author_constraint.exs priv/repo/migrations/20250620044014_update_tags_remove_author_constraint.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/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_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/images/logo.svg priv/static/images/logo.svg
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_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/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_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_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/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/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/test_helper.exs test/test_helper.exs
elixir_blonk/test_firehose.exs test_firehose.exs
+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
··· 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
··· 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
··· 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
··· 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
··· 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
··· 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
··· 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": "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",
  "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"]
}
