dev: automated commit - 2025-06-16 09:37:04
This commit is contained in:
		
							parent
							
								
									682f25edcd
								
							
						
					
					
						commit
						d71a7e37c5
					
				
					 7 changed files with 1513 additions and 430 deletions
				
			
		|  | @ -1,11 +1,39 @@ | |||
| package templates | ||||
| 
 | ||||
| // IndexProps contains the properties for the Index component | ||||
| import "fmt" | ||||
| import "strings" | ||||
| 
 | ||||
| type IndexProps struct { | ||||
| 	Title string | ||||
| 	Title        string | ||||
| 	IsConfigured bool | ||||
| 	CurrentIP    string | ||||
| 	Config       ConfigData | ||||
| 	Records      []DNSRecord | ||||
| 	UpdateFreqs  []UpdateFrequency | ||||
| } | ||||
| 
 | ||||
| type ConfigData struct { | ||||
| 	ZoneID       string | ||||
| 	Domain       string | ||||
| 	UpdatePeriod string | ||||
| 	ApiToken     string | ||||
| } | ||||
| 
 | ||||
| type DNSRecord struct { | ||||
| 	ID        string | ||||
| 	Type      string | ||||
| 	Name      string | ||||
| 	Content   string | ||||
| 	TTL       int | ||||
| 	Proxied   bool | ||||
| 	CreatedOn string | ||||
| } | ||||
| 
 | ||||
| type UpdateFrequency struct { | ||||
| 	Label string | ||||
| 	Value string | ||||
| } | ||||
| 
 | ||||
| // Index is the main page component | ||||
| templ Index(props IndexProps) { | ||||
| 	@Layout(props.Title) { | ||||
| 		<div class="container"> | ||||
|  | @ -15,122 +43,214 @@ templ Index(props IndexProps) { | |||
| 						<h1>{ props.Title }</h1> | ||||
| 						<div class="current-ip d-flex align-items-center"> | ||||
| 							<span class="me-2">Current IP:</span> | ||||
| 							<span id="current-ip" class="fw-bold"></span> | ||||
| 							<span id="current-ip" class="fw-bold">{ props.CurrentIP }</span> | ||||
| 							<button | ||||
| 								id="refresh-ip" | ||||
| 								class="btn btn-sm btn-outline-secondary ms-2" | ||||
| 								title="Refresh current IP" | ||||
| 								hx-get="/refresh-ip" | ||||
| 								hx-target="#current-ip" | ||||
| 								hx-indicator="#refresh-spinner" | ||||
| 							> | ||||
| 								<i class="bi bi-arrow-clockwise"></i> | ||||
| 								<span id="refresh-spinner" class="htmx-indicator spinner-border spinner-border-sm ms-1"></span> | ||||
| 							</button> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					<!-- Configuration Warning --> | ||||
| 					<div | ||||
| 						id="config-warning" | ||||
| 						class="alert alert-warning config-warning" | ||||
| 						style="display: none" | ||||
| 					> | ||||
| 						<h4>Configuration Required</h4> | ||||
| 						<p> | ||||
| 							Please configure your Cloudflare API credentials to manage your | ||||
| 							DNS records. | ||||
| 						</p> | ||||
| 						<button | ||||
| 							class="btn btn-primary" | ||||
| 							data-bs-toggle="modal" | ||||
| 							data-bs-target="#configModal" | ||||
| 						> | ||||
| 							Configure Now | ||||
| 						</button> | ||||
| 					</div> | ||||
| 					<!-- Configuration Status --> | ||||
| 					<div id="config-status" class="card mb-4" style="display: none"> | ||||
| 						<div | ||||
| 							class="card-header d-flex justify-content-between align-items-center" | ||||
| 						> | ||||
| 							<h5 class="mb-0">Configuration</h5> | ||||
| 							<button | ||||
| 								class="btn btn-sm btn-outline-primary" | ||||
| 								data-bs-toggle="modal" | ||||
| 								data-bs-target="#configModal" | ||||
| 							> | ||||
| 								Edit | ||||
| 							</button> | ||||
| 						</div> | ||||
| 						<div class="card-body"> | ||||
| 							<div class="row"> | ||||
| 								<div class="col-md-4"> | ||||
| 									<strong>Domain:</strong> <span id="domain-name">mz.uy</span> | ||||
| 								</div> | ||||
| 								<div class="col-md-4"> | ||||
| 									<strong>Zone ID:</strong> <span id="zone-id"></span> | ||||
| 								</div> | ||||
| 								<div class="col-md-4"> | ||||
| 									<strong>IP Update Schedule:</strong> | ||||
| 									<span id="update-schedule"></span> | ||||
| 								</div> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					<!-- DNS Records Section --> | ||||
| 					<div id="dns-records-section" style="display: none"> | ||||
| 						<div class="card"> | ||||
| 							<div | ||||
| 								class="card-header d-flex justify-content-between align-items-center" | ||||
| 							> | ||||
| 								<h5 class="mb-0">DNS Records</h5> | ||||
| 								<div> | ||||
| 									<button | ||||
| 										id="update-all-records" | ||||
| 										class="btn btn-sm btn-success me-2" | ||||
| 									> | ||||
| 										<i class="bi bi-arrow-repeat"></i> Update All to Current IP | ||||
| 									</button> | ||||
| 									<button | ||||
| 										class="btn btn-sm btn-primary" | ||||
| 										data-bs-toggle="modal" | ||||
| 										data-bs-target="#recordModal" | ||||
| 									> | ||||
| 										<i class="bi bi-plus-lg"></i> Add Record | ||||
| 									</button> | ||||
| 								</div> | ||||
| 							</div> | ||||
| 							<div class="card-body p-0"> | ||||
| 								<div class="table-responsive"> | ||||
| 									<table class="table table-striped table-hover mb-0"> | ||||
| 										<thead> | ||||
| 											<tr> | ||||
| 												<th>Type</th> | ||||
| 												<th>Name</th> | ||||
| 												<th>Content</th> | ||||
| 												<th>TTL</th> | ||||
| 												<th>Proxied</th> | ||||
| 												<th>Actions</th> | ||||
| 											</tr> | ||||
| 										</thead> | ||||
| 										<tbody id="dns-records"> | ||||
| 											<!-- DNS records will be inserted here --> | ||||
| 										</tbody> | ||||
| 									</table> | ||||
| 								</div> | ||||
| 							</div> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					<!-- Loading Indicator --> | ||||
| 					<div id="loading" class="text-center my-5"> | ||||
| 						<div class="spinner-border" role="status"> | ||||
| 							<span class="visually-hidden">Loading...</span> | ||||
| 						</div> | ||||
| 						<p class="mt-2">Loading...</p> | ||||
| 					</div> | ||||
| 					if !props.IsConfigured { | ||||
| 						@ConfigWarning() | ||||
| 					} else { | ||||
| 						@ConfigStatus(props.Config) | ||||
| 						@DNSRecordsSection(props.Records, props.CurrentIP) | ||||
| 					} | ||||
| 					@ConfigModal(props.Config, props.UpdateFreqs) | ||||
| 					@RecordModal(props.Config.Domain) | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		@ConfigModal() | ||||
| 		@RecordModal() | ||||
| 		<!-- Toast container for notifications --> | ||||
| 		<div class="toast-container"></div> | ||||
| 		<script src="/assets/js/app.js"></script> | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| templ ConfigWarning() { | ||||
| 	<div class="alert alert-warning config-warning"> | ||||
| 		<h4>Configuration Required</h4> | ||||
| 		<p>Please configure your Cloudflare API credentials to manage your DNS records.</p> | ||||
| 		<button | ||||
| 			class="btn btn-primary" | ||||
| 			data-bs-toggle="modal" | ||||
| 			data-bs-target="#configModal" | ||||
| 		> | ||||
| 			Configure Now | ||||
| 		</button> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| templ ConfigStatus(config ConfigData) { | ||||
| 	<div class="card mb-4"> | ||||
| 		<div class="card-header d-flex justify-content-between align-items-center"> | ||||
| 			<h5 class="mb-0">Configuration</h5> | ||||
| 			<button | ||||
| 				class="btn btn-sm btn-outline-primary" | ||||
| 				data-bs-toggle="modal" | ||||
| 				data-bs-target="#configModal" | ||||
| 			> | ||||
| 				Edit | ||||
| 			</button> | ||||
| 		</div> | ||||
| 		<div class="card-body"> | ||||
| 			<div class="row"> | ||||
| 				<div class="col-md-4"> | ||||
| 					<strong>Domain:</strong> <span>{ config.Domain }</span> | ||||
| 				</div> | ||||
| 				<div class="col-md-4"> | ||||
| 					<strong>Zone ID:</strong> <span>{ config.ZoneID }</span> | ||||
| 				</div> | ||||
| 				<div class="col-md-4"> | ||||
| 					<strong>IP Update Schedule:</strong> | ||||
| 					<span>{ formatUpdateSchedule(config.UpdatePeriod) }</span> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| templ DNSRecordsSection(records []DNSRecord, currentIP string) { | ||||
| 	<div class="card"> | ||||
| 		<div class="card-header d-flex justify-content-between align-items-center"> | ||||
| 			<h5 class="mb-0">DNS Records</h5> | ||||
| 			<div> | ||||
| 				<button | ||||
| 					class="btn btn-sm btn-success me-2" | ||||
| 					hx-post="/update-all-records" | ||||
| 					hx-target="#dns-records-table" | ||||
| 					hx-confirm="Are you sure you want to update all A records to your current IP?" | ||||
| 					hx-indicator="#update-all-spinner" | ||||
| 				> | ||||
| 					<i class="bi bi-arrow-repeat"></i> Update All to Current IP | ||||
| 					<span id="update-all-spinner" class="htmx-indicator spinner-border spinner-border-sm ms-1"></span> | ||||
| 				</button> | ||||
| 				<button | ||||
| 					class="btn btn-sm btn-primary" | ||||
| 					data-bs-toggle="modal" | ||||
| 					data-bs-target="#recordModal" | ||||
| 					onclick="resetRecordForm()" | ||||
| 				> | ||||
| 					<i class="bi bi-plus-lg"></i> Add Record | ||||
| 				</button> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="card-body p-0"> | ||||
| 			@DNSRecordsTable(records, currentIP) | ||||
| 		</div> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| templ DNSRecordsTable(records []DNSRecord, currentIP string) { | ||||
| 	<div id="dns-records-table" class="table-responsive"> | ||||
| 		<table class="table table-striped table-hover mb-0"> | ||||
| 			<thead> | ||||
| 				<tr> | ||||
| 					<th>Type</th> | ||||
| 					<th>Name</th> | ||||
| 					<th>Content</th> | ||||
| 					<th>TTL</th> | ||||
| 					<th>Proxied</th> | ||||
| 					<th>Actions</th> | ||||
| 				</tr> | ||||
| 			</thead> | ||||
| 			<tbody> | ||||
| 				if len(records) == 0 { | ||||
| 					<tr> | ||||
| 						<td colspan="6" class="text-center">No DNS records found</td> | ||||
| 					</tr> | ||||
| 				} else { | ||||
| 					for _, record := range records { | ||||
| 						@DNSRecordRow(record, currentIP) | ||||
| 					} | ||||
| 				} | ||||
| 			</tbody> | ||||
| 		</table> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| templ DNSRecordRow(record DNSRecord, currentIP string) { | ||||
| 	<tr> | ||||
| 		<td>{ record.Type }</td> | ||||
| 		<td>{ record.Name }</td> | ||||
| 		<td> | ||||
| 			{ record.Content } | ||||
| 			if record.Type == "A" { | ||||
| 				if record.Content == currentIP { | ||||
| 					<span class="badge bg-success update-badge">Current IP</span> | ||||
| 				} else { | ||||
| 					<span class="badge bg-warning update-badge">Outdated IP</span> | ||||
| 				} | ||||
| 			} | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			if record.TTL == 1 { | ||||
| 				Auto | ||||
| 			} else { | ||||
| 				{ fmt.Sprintf("%ds", record.TTL) } | ||||
| 			} | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			if record.Proxied { | ||||
| 				<i class="bi bi-check-lg text-success"></i> | ||||
| 			} | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<button | ||||
| 				class="btn btn-sm btn-outline-primary me-1" | ||||
| 				hx-get={ fmt.Sprintf("/edit-record/%s", record.ID) } | ||||
| 				hx-target="#record-modal-content" | ||||
| 				data-bs-toggle="modal" | ||||
| 				data-bs-target="#recordModal" | ||||
| 			> | ||||
| 				<i class="bi bi-pencil"></i> | ||||
| 			</button> | ||||
| 			<button | ||||
| 				class="btn btn-sm btn-outline-danger" | ||||
| 				hx-delete={ fmt.Sprintf("/records/%s", record.ID) } | ||||
| 				hx-target="closest tr" | ||||
| 				hx-swap="outerHTML" | ||||
| 				hx-confirm={ fmt.Sprintf("Are you sure you want to delete the record for \"%s\"?", record.Name) } | ||||
| 			> | ||||
| 				<i class="bi bi-trash"></i> | ||||
| 			</button> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| } | ||||
| 
 | ||||
| // Helper function to format update schedule | ||||
| func formatUpdateSchedule(period string) string { | ||||
| 	switch period { | ||||
| 	case "*/1 * * * *": | ||||
| 		return "Every minute" | ||||
| 	case "*/5 * * * *": | ||||
| 		return "Every 5 minutes" | ||||
| 	case "*/30 * * * *": | ||||
| 		return "Every 30 minutes" | ||||
| 	case "0 * * * *": | ||||
| 		return "Hourly" | ||||
| 	case "0 */6 * * *": | ||||
| 		return "Every 6 hours" | ||||
| 	case "0 0 * * *": | ||||
| 		return "Daily" | ||||
| 	case "": | ||||
| 		return "Manual updates only" | ||||
| 	default: | ||||
| 		return period | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Helper function to extract subdomain name | ||||
| func getRecordName(fullName, domain string) string { | ||||
| 	if fullName == domain { | ||||
| 		return "@" | ||||
| 	} | ||||
| 	if strings.HasSuffix(fullName, "."+domain) { | ||||
| 		return fullName[:len(fullName)-len(domain)-1] | ||||
| 	} | ||||
| 	return fullName | ||||
| } | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| // Code generated by templ - DO NOT EDIT. | ||||
| 
 | ||||
| // templ: version: v0.3.857 | ||||
| // templ: version: v0.3.898 | ||||
| package templates | ||||
| 
 | ||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||
|  | @ -8,12 +8,40 @@ package templates | |||
| import "github.com/a-h/templ" | ||||
| import templruntime "github.com/a-h/templ/runtime" | ||||
| 
 | ||||
| // IndexProps contains the properties for the Index component | ||||
| import "fmt" | ||||
| import "strings" | ||||
| 
 | ||||
| type IndexProps struct { | ||||
| 	Title string | ||||
| 	Title        string | ||||
| 	IsConfigured bool | ||||
| 	CurrentIP    string | ||||
| 	Config       ConfigData | ||||
| 	Records      []DNSRecord | ||||
| 	UpdateFreqs  []UpdateFrequency | ||||
| } | ||||
| 
 | ||||
| type ConfigData struct { | ||||
| 	ZoneID       string | ||||
| 	Domain       string | ||||
| 	UpdatePeriod string | ||||
| 	ApiToken     string | ||||
| } | ||||
| 
 | ||||
| type DNSRecord struct { | ||||
| 	ID        string | ||||
| 	Type      string | ||||
| 	Name      string | ||||
| 	Content   string | ||||
| 	TTL       int | ||||
| 	Proxied   bool | ||||
| 	CreatedOn string | ||||
| } | ||||
| 
 | ||||
| type UpdateFrequency struct { | ||||
| 	Label string | ||||
| 	Value string | ||||
| } | ||||
| 
 | ||||
| // Index is the main page component | ||||
| func Index(props IndexProps) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
|  | @ -54,29 +82,57 @@ func Index(props IndexProps) templ.Component { | |||
| 			var templ_7745c5c3_Var3 string | ||||
| 			templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(props.Title) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 15, Col: 23} | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 43, Col: 23} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "</h1><div class=\"current-ip d-flex align-items-center\"><span class=\"me-2\">Current IP:</span> <span id=\"current-ip\" class=\"fw-bold\"></span> <button id=\"refresh-ip\" class=\"btn btn-sm btn-outline-secondary ms-2\" title=\"Refresh current IP\"><i class=\"bi bi-arrow-clockwise\"></i></button></div></div><!-- Configuration Warning --><div id=\"config-warning\" class=\"alert alert-warning config-warning\" style=\"display: none\"><h4>Configuration Required</h4><p>Please configure your Cloudflare API credentials to manage your DNS records.</p><button class=\"btn btn-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#configModal\">Configure Now</button></div><!-- Configuration Status --><div id=\"config-status\" class=\"card mb-4\" style=\"display: none\"><div class=\"card-header d-flex justify-content-between align-items-center\"><h5 class=\"mb-0\">Configuration</h5><button class=\"btn btn-sm btn-outline-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#configModal\">Edit</button></div><div class=\"card-body\"><div class=\"row\"><div class=\"col-md-4\"><strong>Domain:</strong> <span id=\"domain-name\">mz.uy</span></div><div class=\"col-md-4\"><strong>Zone ID:</strong> <span id=\"zone-id\"></span></div><div class=\"col-md-4\"><strong>IP Update Schedule:</strong> <span id=\"update-schedule\"></span></div></div></div></div><!-- DNS Records Section --><div id=\"dns-records-section\" style=\"display: none\"><div class=\"card\"><div class=\"card-header d-flex justify-content-between align-items-center\"><h5 class=\"mb-0\">DNS Records</h5><div><button id=\"update-all-records\" class=\"btn btn-sm btn-success me-2\"><i class=\"bi bi-arrow-repeat\"></i> Update All to Current IP</button> <button class=\"btn btn-sm btn-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#recordModal\"><i class=\"bi bi-plus-lg\"></i> Add Record</button></div></div><div class=\"card-body p-0\"><div class=\"table-responsive\"><table class=\"table table-striped table-hover mb-0\"><thead><tr><th>Type</th><th>Name</th><th>Content</th><th>TTL</th><th>Proxied</th><th>Actions</th></tr></thead> <tbody id=\"dns-records\"><!-- DNS records will be inserted here --></tbody></table></div></div></div></div><!-- Loading Indicator --><div id=\"loading\" class=\"text-center my-5\"><div class=\"spinner-border\" role=\"status\"><span class=\"visually-hidden\">Loading...</span></div><p class=\"mt-2\">Loading...</p></div></div></div></div>") | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "</h1><div class=\"current-ip d-flex align-items-center\"><span class=\"me-2\">Current IP:</span> <span id=\"current-ip\" class=\"fw-bold\">") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = ConfigModal().Render(ctx, templ_7745c5c3_Buffer) | ||||
| 			var templ_7745c5c3_Var4 string | ||||
| 			templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(props.CurrentIP) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 46, Col: 62} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, " ") | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "</span> <button class=\"btn btn-sm btn-outline-secondary ms-2\" title=\"Refresh current IP\" hx-get=\"/refresh-ip\" hx-target=\"#current-ip\" hx-indicator=\"#refresh-spinner\"><i class=\"bi bi-arrow-clockwise\"></i> <span id=\"refresh-spinner\" class=\"htmx-indicator spinner-border spinner-border-sm ms-1\"></span></button></div></div>") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = RecordModal().Render(ctx, templ_7745c5c3_Buffer) | ||||
| 			if !props.IsConfigured { | ||||
| 				templ_7745c5c3_Err = ConfigWarning().Render(ctx, templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} else { | ||||
| 				templ_7745c5c3_Err = ConfigStatus(props.Config).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, " ") | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 				templ_7745c5c3_Err = DNSRecordsSection(props.Records, props.CurrentIP).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = ConfigModal(props.Config, props.UpdateFreqs).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, " <!-- Toast container for notifications --> <div class=\"toast-container\"></div><script src=\"/assets/js/app.js\"></script>") | ||||
| 			templ_7745c5c3_Err = RecordModal(props.Config.Domain).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "</div></div></div><div class=\"toast-container\"></div>") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
|  | @ -90,4 +146,371 @@ func Index(props IndexProps) templ.Component { | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func ConfigWarning() templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var5 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var5 == nil { | ||||
| 			templ_7745c5c3_Var5 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "<div class=\"alert alert-warning config-warning\"><h4>Configuration Required</h4><p>Please configure your Cloudflare API credentials to manage your DNS records.</p><button class=\"btn btn-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#configModal\">Configure Now</button></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func ConfigStatus(config ConfigData) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var6 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var6 == nil { | ||||
| 			templ_7745c5c3_Var6 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "<div class=\"card mb-4\"><div class=\"card-header d-flex justify-content-between align-items-center\"><h5 class=\"mb-0\">Configuration</h5><button class=\"btn btn-sm btn-outline-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#configModal\">Edit</button></div><div class=\"card-body\"><div class=\"row\"><div class=\"col-md-4\"><strong>Domain:</strong> <span>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var7 string | ||||
| 		templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(config.Domain) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 103, Col: 51} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "</span></div><div class=\"col-md-4\"><strong>Zone ID:</strong> <span>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var8 string | ||||
| 		templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(config.ZoneID) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 106, Col: 52} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "</span></div><div class=\"col-md-4\"><strong>IP Update Schedule:</strong> <span>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var9 string | ||||
| 		templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(formatUpdateSchedule(config.UpdatePeriod)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 110, Col: 54} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "</span></div></div></div></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func DNSRecordsSection(records []DNSRecord, currentIP string) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var10 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var10 == nil { | ||||
| 			templ_7745c5c3_Var10 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<div class=\"card\"><div class=\"card-header d-flex justify-content-between align-items-center\"><h5 class=\"mb-0\">DNS Records</h5><div><button class=\"btn btn-sm btn-success me-2\" hx-post=\"/update-all-records\" hx-target=\"#dns-records-table\" hx-confirm=\"Are you sure you want to update all A records to your current IP?\" hx-indicator=\"#update-all-spinner\"><i class=\"bi bi-arrow-repeat\"></i> Update All to Current IP <span id=\"update-all-spinner\" class=\"htmx-indicator spinner-border spinner-border-sm ms-1\"></span></button> <button class=\"btn btn-sm btn-primary\" data-bs-toggle=\"modal\" data-bs-target=\"#recordModal\" onclick=\"resetRecordForm()\"><i class=\"bi bi-plus-lg\"></i> Add Record</button></div></div><div class=\"card-body p-0\">") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = DNSRecordsTable(records, currentIP).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "</div></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func DNSRecordsTable(records []DNSRecord, currentIP string) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var11 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var11 == nil { | ||||
| 			templ_7745c5c3_Var11 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "<div id=\"dns-records-table\" class=\"table-responsive\"><table class=\"table table-striped table-hover mb-0\"><thead><tr><th>Type</th><th>Name</th><th>Content</th><th>TTL</th><th>Proxied</th><th>Actions</th></tr></thead> <tbody>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if len(records) == 0 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "<tr><td colspan=\"6\" class=\"text-center\">No DNS records found</td></tr>") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} else { | ||||
| 			for _, record := range records { | ||||
| 				templ_7745c5c3_Err = DNSRecordRow(record, currentIP).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "</tbody></table></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func DNSRecordRow(record DNSRecord, currentIP string) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var12 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var12 == nil { | ||||
| 			templ_7745c5c3_Var12 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "<tr><td>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var13 string | ||||
| 		templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(record.Type) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 178, Col: 19} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "</td><td>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var14 string | ||||
| 		templ_7745c5c3_Var14, templ_7745c5c3_Err = templ.JoinStringErrs(record.Name) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 179, Col: 19} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var14)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "</td><td>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var15 string | ||||
| 		templ_7745c5c3_Var15, templ_7745c5c3_Err = templ.JoinStringErrs(record.Content) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 181, Col: 19} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var15)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, " ") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "A" { | ||||
| 			if record.Content == currentIP { | ||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "<span class=\"badge bg-success update-badge\">Current IP</span>") | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} else { | ||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "<span class=\"badge bg-warning update-badge\">Outdated IP</span>") | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, "</td><td>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 1 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "Auto") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} else { | ||||
| 			var templ_7745c5c3_Var16 string | ||||
| 			templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%ds", record.TTL)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 194, Col: 36} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "</td><td>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Proxied { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "<i class=\"bi bi-check-lg text-success\"></i>") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "</td><td><button class=\"btn btn-sm btn-outline-primary me-1\" hx-get=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var17 string | ||||
| 		templ_7745c5c3_Var17, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("/edit-record/%s", record.ID)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 205, Col: 54} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var17)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, "\" hx-target=\"#record-modal-content\" data-bs-toggle=\"modal\" data-bs-target=\"#recordModal\"><i class=\"bi bi-pencil\"></i></button> <button class=\"btn btn-sm btn-outline-danger\" hx-delete=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var18 string | ||||
| 		templ_7745c5c3_Var18, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("/records/%s", record.ID)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 214, Col: 53} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var18)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "\" hx-target=\"closest tr\" hx-swap=\"outerHTML\" hx-confirm=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var19 string | ||||
| 		templ_7745c5c3_Var19, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("Are you sure you want to delete the record for \"%s\"?", record.Name)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/index.templ`, Line: 217, Col: 99} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var19)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, "\"><i class=\"bi bi-trash\"></i></button></td></tr>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // Helper function to format update schedule | ||||
| func formatUpdateSchedule(period string) string { | ||||
| 	switch period { | ||||
| 	case "*/1 * * * *": | ||||
| 		return "Every minute" | ||||
| 	case "*/5 * * * *": | ||||
| 		return "Every 5 minutes" | ||||
| 	case "*/30 * * * *": | ||||
| 		return "Every 30 minutes" | ||||
| 	case "0 * * * *": | ||||
| 		return "Hourly" | ||||
| 	case "0 */6 * * *": | ||||
| 		return "Every 6 hours" | ||||
| 	case "0 0 * * *": | ||||
| 		return "Daily" | ||||
| 	case "": | ||||
| 		return "Manual updates only" | ||||
| 	default: | ||||
| 		return period | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Helper function to extract subdomain name | ||||
| func getRecordName(fullName, domain string) string { | ||||
| 	if fullName == domain { | ||||
| 		return "@" | ||||
| 	} | ||||
| 	if strings.HasSuffix(fullName, "."+domain) { | ||||
| 		return fullName[:len(fullName)-len(domain)-1] | ||||
| 	} | ||||
| 	return fullName | ||||
| } | ||||
| 
 | ||||
| var _ = templruntime.GeneratedTemplate | ||||
|  |  | |||
|  | @ -24,6 +24,7 @@ templ Layout(title string) { | |||
| 				rel="stylesheet" | ||||
| 				href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css" | ||||
| 			/> | ||||
| 			<script src="https://unpkg.com/htmx.org@1.9.10"></script> | ||||
| 			<style> | ||||
| 				body { | ||||
| 					padding-top: 20px; | ||||
|  | @ -46,11 +47,62 @@ templ Layout(title string) { | |||
| 					font-size: 0.8em; | ||||
| 					margin-left: 10px; | ||||
| 				} | ||||
| 				.htmx-indicator { | ||||
| 					opacity: 0; | ||||
| 					transition: opacity 300ms ease-in; | ||||
| 				} | ||||
| 				.htmx-request .htmx-indicator { | ||||
| 					opacity: 1; | ||||
| 				} | ||||
| 				.htmx-request.htmx-indicator { | ||||
| 					opacity: 1; | ||||
| 				} | ||||
| 
 | ||||
| 
 | ||||
| 			</style> | ||||
| 		</head> | ||||
| 		<body> | ||||
| 			{ children... } | ||||
| 			<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script> | ||||
| 			<script> | ||||
| 				// Simple toast function for HTMX responses | ||||
| 				document.body.addEventListener('htmx:afterRequest', function(evt) { | ||||
| 					const xhr = evt.detail.xhr; | ||||
| 					if (xhr.status >= 200 && xhr.status < 300) { | ||||
| 						const successMessage = xhr.getResponseHeader('HX-Success-Message'); | ||||
| 						if (successMessage) { | ||||
| 							showToast(successMessage, 'success'); | ||||
| 						} | ||||
| 					} else { | ||||
| 						const errorMessage = xhr.getResponseHeader('HX-Error-Message') || 'An error occurred'; | ||||
| 						showToast(errorMessage, 'danger'); | ||||
| 					} | ||||
| 				}); | ||||
| 
 | ||||
| 				function showToast(message, type = "success") { | ||||
| 					const toastContainer = document.querySelector(".toast-container"); | ||||
| 					const toast = document.createElement("div"); | ||||
| 					toast.className = `toast align-items-center text-white bg-${type}`; | ||||
| 					toast.setAttribute("role", "alert"); | ||||
| 					toast.setAttribute("aria-live", "assertive"); | ||||
| 					toast.setAttribute("aria-atomic", "true"); | ||||
| 
 | ||||
| 					toast.innerHTML = ` | ||||
| 						<div class="d-flex"> | ||||
| 							<div class="toast-body">${message}</div> | ||||
| 							<button type="button" class="btn-close btn-close-white me-2 m-auto" data-bs-dismiss="toast" aria-label="Close"></button> | ||||
| 						</div> | ||||
| 					`; | ||||
| 
 | ||||
| 					toastContainer.appendChild(toast); | ||||
| 					const bsToast = new bootstrap.Toast(toast); | ||||
| 					bsToast.show(); | ||||
| 
 | ||||
| 					toast.addEventListener("hidden.bs.toast", () => { | ||||
| 						toast.remove(); | ||||
| 					}); | ||||
| 				} | ||||
| 			</script> | ||||
| 		</body> | ||||
| 	</html> | ||||
| } | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| // Code generated by templ - DO NOT EDIT. | ||||
| 
 | ||||
| // templ: version: v0.3.857 | ||||
| // templ: version: v0.3.898 | ||||
| package templates | ||||
| 
 | ||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||
|  | @ -51,7 +51,7 @@ func Layout(title string) templ.Component { | |||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "</title><link href=\"https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css\" rel=\"stylesheet\"><link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css\"><style>\n\t\t\t\tbody {\n\t\t\t\t\tpadding-top: 20px;\n\t\t\t\t\tbackground-color: #f8f9fa;\n\t\t\t\t}\n\t\t\t\t.card {\n\t\t\t\t\tmargin-bottom: 20px;\n\t\t\t\t\tbox-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);\n\t\t\t\t}\n\t\t\t\t.config-warning {\n\t\t\t\t\tmargin-bottom: 20px;\n\t\t\t\t}\n\t\t\t\t.toast-container {\n\t\t\t\t\tposition: fixed;\n\t\t\t\t\ttop: 20px;\n\t\t\t\t\tright: 20px;\n\t\t\t\t\tz-index: 1050;\n\t\t\t\t}\n\t\t\t\t.update-badge {\n\t\t\t\t\tfont-size: 0.8em;\n\t\t\t\t\tmargin-left: 10px;\n\t\t\t\t}\n\t\t\t</style></head><body>") | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "</title><link href=\"https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css\" rel=\"stylesheet\"><link rel=\"stylesheet\" href=\"https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.1/font/bootstrap-icons.css\"><script src=\"https://unpkg.com/htmx.org@1.9.10\"></script><style>\n\t\t\t\tbody {\n\t\t\t\t\tpadding-top: 20px;\n\t\t\t\t\tbackground-color: #f8f9fa;\n\t\t\t\t}\n\t\t\t\t.card {\n\t\t\t\t\tmargin-bottom: 20px;\n\t\t\t\t\tbox-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);\n\t\t\t\t}\n\t\t\t\t.config-warning {\n\t\t\t\t\tmargin-bottom: 20px;\n\t\t\t\t}\n\t\t\t\t.toast-container {\n\t\t\t\t\tposition: fixed;\n\t\t\t\t\ttop: 20px;\n\t\t\t\t\tright: 20px;\n\t\t\t\t\tz-index: 1050;\n\t\t\t\t}\n\t\t\t\t.update-badge {\n\t\t\t\t\tfont-size: 0.8em;\n\t\t\t\t\tmargin-left: 10px;\n\t\t\t\t}\n\t\t\t\t.htmx-indicator {\n\t\t\t\t\topacity: 0;\n\t\t\t\t\ttransition: opacity 300ms ease-in;\n\t\t\t\t}\n\t\t\t\t.htmx-request .htmx-indicator {\n\t\t\t\t\topacity: 1;\n\t\t\t\t}\n\t\t\t\t.htmx-request.htmx-indicator {\n\t\t\t\t\topacity: 1;\n\t\t\t\t}\n\n\n\t\t\t</style></head><body>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
|  | @ -59,7 +59,7 @@ func Layout(title string) templ.Component { | |||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "<script src=\"https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js\"></script></body></html>") | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "<script src=\"https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js\"></script><script>\n\t\t\t\t// Simple toast function for HTMX responses\n\t\t\t\tdocument.body.addEventListener('htmx:afterRequest', function(evt) {\n\t\t\t\t\tconst xhr = evt.detail.xhr;\n\t\t\t\t\tif (xhr.status >= 200 && xhr.status < 300) {\n\t\t\t\t\t\tconst successMessage = xhr.getResponseHeader('HX-Success-Message');\n\t\t\t\t\t\tif (successMessage) {\n\t\t\t\t\t\t\tshowToast(successMessage, 'success');\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tconst errorMessage = xhr.getResponseHeader('HX-Error-Message') || 'An error occurred';\n\t\t\t\t\t\tshowToast(errorMessage, 'danger');\n\t\t\t\t\t}\n\t\t\t\t});\n\n\t\t\t\tfunction showToast(message, type = \"success\") {\n\t\t\t\t\tconst toastContainer = document.querySelector(\".toast-container\");\n\t\t\t\t\tconst toast = document.createElement(\"div\");\n\t\t\t\t\ttoast.className = `toast align-items-center text-white bg-${type}`;\n\t\t\t\t\ttoast.setAttribute(\"role\", \"alert\");\n\t\t\t\t\ttoast.setAttribute(\"aria-live\", \"assertive\");\n\t\t\t\t\ttoast.setAttribute(\"aria-atomic\", \"true\");\n\n\t\t\t\t\ttoast.innerHTML = `\n\t\t\t\t\t\t<div class=\"d-flex\">\n\t\t\t\t\t\t\t<div class=\"toast-body\">${message}</div>\n\t\t\t\t\t\t\t<button type=\"button\" class=\"btn-close btn-close-white me-2 m-auto\" data-bs-dismiss=\"toast\" aria-label=\"Close\"></button>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t`;\n\n\t\t\t\t\ttoastContainer.appendChild(toast);\n\t\t\t\t\tconst bsToast = new bootstrap.Toast(toast);\n\t\t\t\t\tbsToast.show();\n\n\t\t\t\t\ttoast.addEventListener(\"hidden.bs.toast\", () => {\n\t\t\t\t\t\ttoast.remove();\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t</script></body></html>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
|  |  | |||
|  | @ -1,7 +1,8 @@ | |||
| package templates | ||||
| 
 | ||||
| // ConfigModal is the configuration dialog component | ||||
| templ ConfigModal() { | ||||
| import "fmt" | ||||
| 
 | ||||
| templ ConfigModal(config ConfigData, frequencies []UpdateFrequency) { | ||||
| 	<div | ||||
| 		class="modal fade" | ||||
| 		id="configModal" | ||||
|  | @ -20,14 +21,16 @@ templ ConfigModal() { | |||
| 						aria-label="Close" | ||||
| 					></button> | ||||
| 				</div> | ||||
| 				<div class="modal-body"> | ||||
| 					<form id="config-form"> | ||||
| 				<form hx-post="/config" hx-target="body" hx-swap="outerHTML"> | ||||
| 					<div class="modal-body"> | ||||
| 						<div class="mb-3"> | ||||
| 							<label for="api-token" class="form-label">Cloudflare API Token</label> | ||||
| 							<input | ||||
| 								type="password" | ||||
| 								class="form-control" | ||||
| 								id="api-token" | ||||
| 								name="api_token" | ||||
| 								value={ config.ApiToken } | ||||
| 								required | ||||
| 							/> | ||||
| 							<div class="form-text"> | ||||
|  | @ -41,11 +44,12 @@ templ ConfigModal() { | |||
| 								type="text" | ||||
| 								class="form-control" | ||||
| 								id="zone-id-input" | ||||
| 								name="zone_id" | ||||
| 								value={ config.ZoneID } | ||||
| 								required | ||||
| 							/> | ||||
| 							<div class="form-text"> | ||||
| 								Found in the Cloudflare dashboard under your domain's overview | ||||
| 								page. | ||||
| 								Found in the Cloudflare dashboard under your domain's overview page. | ||||
| 							</div> | ||||
| 						</div> | ||||
| 						<div class="mb-3"> | ||||
|  | @ -54,37 +58,42 @@ templ ConfigModal() { | |||
| 								type="text" | ||||
| 								class="form-control" | ||||
| 								id="domain-input" | ||||
| 								value="mz.uy" | ||||
| 								name="domain" | ||||
| 								value={ config.Domain } | ||||
| 								required | ||||
| 							/> | ||||
| 						</div> | ||||
| 						<div class="mb-3"> | ||||
| 							<label for="update-period" class="form-label">Update Frequency</label> | ||||
| 							<select class="form-select" id="update-period"> | ||||
| 								<!-- Options will be loaded from API --> | ||||
| 							<select class="form-select" id="update-period" name="update_period"> | ||||
| 								for _, freq := range frequencies { | ||||
| 									<option value={ freq.Value } selected?={ freq.Value == config.UpdatePeriod }> | ||||
| 										{ freq.Label } | ||||
| 									</option> | ||||
| 								} | ||||
| 							</select> | ||||
| 						</div> | ||||
| 					</form> | ||||
| 				</div> | ||||
| 				<div class="modal-footer"> | ||||
| 					<button | ||||
| 						type="button" | ||||
| 						class="btn btn-secondary" | ||||
| 						data-bs-dismiss="modal" | ||||
| 					> | ||||
| 						Cancel | ||||
| 					</button> | ||||
| 					<button type="button" class="btn btn-primary" id="save-config"> | ||||
| 						Save | ||||
| 					</button> | ||||
| 				</div> | ||||
| 					</div> | ||||
| 					<div class="modal-footer"> | ||||
| 						<button | ||||
| 							type="button" | ||||
| 							class="btn btn-secondary" | ||||
| 							data-bs-dismiss="modal" | ||||
| 						> | ||||
| 							Cancel | ||||
| 						</button> | ||||
| 						<button type="submit" class="btn btn-primary"> | ||||
| 							Save | ||||
| 							<span class="htmx-indicator spinner-border spinner-border-sm ms-1"></span> | ||||
| 						</button> | ||||
| 					</div> | ||||
| 				</form> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| // RecordModal is the DNS record dialog component | ||||
| templ RecordModal() { | ||||
| templ RecordModal(domain string) { | ||||
| 	<div | ||||
| 		class="modal fade" | ||||
| 		id="recordModal" | ||||
|  | @ -93,98 +102,154 @@ templ RecordModal() { | |||
| 		aria-hidden="true" | ||||
| 	> | ||||
| 		<div class="modal-dialog"> | ||||
| 			<div class="modal-content"> | ||||
| 				<div class="modal-header"> | ||||
| 					<h5 class="modal-title" id="recordModalLabel">Add DNS Record</h5> | ||||
| 					<button | ||||
| 						type="button" | ||||
| 						class="btn-close" | ||||
| 						data-bs-dismiss="modal" | ||||
| 						aria-label="Close" | ||||
| 					></button> | ||||
| 				</div> | ||||
| 				<div class="modal-body"> | ||||
| 					<form id="record-form"> | ||||
| 						<input type="hidden" id="record-id"/> | ||||
| 						<div class="mb-3"> | ||||
| 							<label for="record-name" class="form-label">Name</label> | ||||
| 							<div class="input-group"> | ||||
| 								<input | ||||
| 									type="text" | ||||
| 									class="form-control" | ||||
| 									id="record-name" | ||||
| 									placeholder="subdomain" | ||||
| 									required | ||||
| 								/> | ||||
| 								<span class="input-group-text" id="domain-suffix">.mz.uy</span> | ||||
| 							</div> | ||||
| 							<div class="form-text">Use @ for the root domain</div> | ||||
| 						</div> | ||||
| 						<div class="mb-3"> | ||||
| 							<label for="record-type" class="form-label">Type</label> | ||||
| 							<select class="form-select" id="record-type"> | ||||
| 								<option value="A">A</option> | ||||
| 								<option value="AAAA">AAAA</option> | ||||
| 								<option value="CNAME">CNAME</option> | ||||
| 								<option value="TXT">TXT</option> | ||||
| 								<option value="MX">MX</option> | ||||
| 							</select> | ||||
| 						</div> | ||||
| 						<div class="mb-3" id="content-group"> | ||||
| 							<label for="record-content" class="form-label">Content</label> | ||||
| 							<input | ||||
| 								type="text" | ||||
| 								class="form-control" | ||||
| 								id="record-content" | ||||
| 								required | ||||
| 							/> | ||||
| 						</div> | ||||
| 						<div class="mb-3 form-check" id="use-my-ip-group"> | ||||
| 							<input | ||||
| 								type="checkbox" | ||||
| 								class="form-check-input" | ||||
| 								id="use-my-ip" | ||||
| 							/> | ||||
| 							<label class="form-check-label" for="use-my-ip">Use my current IP address</label> | ||||
| 						</div> | ||||
| 						<div class="mb-3"> | ||||
| 							<label for="record-ttl" class="form-label">TTL</label> | ||||
| 							<select class="form-select" id="record-ttl"> | ||||
| 								<option value="1">Auto</option> | ||||
| 								<option value="120">2 minutes</option> | ||||
| 								<option value="300">5 minutes</option> | ||||
| 								<option value="600">10 minutes</option> | ||||
| 								<option value="1800">30 minutes</option> | ||||
| 								<option value="3600">1 hour</option> | ||||
| 								<option value="7200">2 hours</option> | ||||
| 								<option value="18000">5 hours</option> | ||||
| 								<option value="43200">12 hours</option> | ||||
| 								<option value="86400">1 day</option> | ||||
| 							</select> | ||||
| 						</div> | ||||
| 						<div class="mb-3 form-check"> | ||||
| 							<input | ||||
| 								type="checkbox" | ||||
| 								class="form-check-input" | ||||
| 								id="record-proxied" | ||||
| 							/> | ||||
| 							<label class="form-check-label" for="record-proxied">Proxied through Cloudflare</label> | ||||
| 						</div> | ||||
| 					</form> | ||||
| 				</div> | ||||
| 				<div class="modal-footer"> | ||||
| 					<button | ||||
| 						type="button" | ||||
| 						class="btn btn-secondary" | ||||
| 						data-bs-dismiss="modal" | ||||
| 					> | ||||
| 						Cancel | ||||
| 					</button> | ||||
| 					<button type="button" class="btn btn-primary" id="save-record"> | ||||
| 						Save | ||||
| 					</button> | ||||
| 				</div> | ||||
| 			<div id="record-modal-content" class="modal-content"> | ||||
| 				@RecordForm("Add DNS Record", "", domain, DNSRecord{Type: "A", TTL: 1}) | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| } | ||||
| 
 | ||||
| templ RecordForm(title, recordID, domain string, record DNSRecord) { | ||||
| 	<div class="modal-header"> | ||||
| 		<h5 class="modal-title">{ title }</h5> | ||||
| 		<button | ||||
| 			type="button" | ||||
| 			class="btn-close" | ||||
| 			data-bs-dismiss="modal" | ||||
| 			aria-label="Close" | ||||
| 		></button> | ||||
| 	</div> | ||||
| 	<form | ||||
| 		if recordID != "" { | ||||
| 			hx-put={ fmt.Sprintf("/records/%s", recordID) } | ||||
| 		} else { | ||||
| 			hx-post="/records" | ||||
| 		} | ||||
| 		hx-target="#dns-records-table" | ||||
| 		hx-on::after-request="if(event.detail.successful) bootstrap.Modal.getInstance(document.getElementById('recordModal')).hide()" | ||||
| 	> | ||||
| 		<div class="modal-body"> | ||||
| 			<div class="mb-3"> | ||||
| 				<label for="record-name" class="form-label">Name</label> | ||||
| 				<div class="input-group"> | ||||
| 					<input | ||||
| 						type="text" | ||||
| 						class="form-control" | ||||
| 						id="record-name" | ||||
| 						name="name" | ||||
| 						placeholder="subdomain" | ||||
| 						value={ getRecordName(record.Name, domain) } | ||||
| 						required | ||||
| 					/> | ||||
| 					<span class="input-group-text">.{ domain }</span> | ||||
| 				</div> | ||||
| 				<div class="form-text">Use @ for the root domain</div> | ||||
| 			</div> | ||||
| 			<div class="mb-3"> | ||||
| 				<label for="record-type" class="form-label">Type</label> | ||||
| 				<select | ||||
| 					class="form-select" | ||||
| 					id="record-type" | ||||
| 					name="type" | ||||
| 					onchange="toggleMyIPOption()" | ||||
| 				> | ||||
| 					<option value="A" selected?={ record.Type == "A" }>A</option> | ||||
| 					<option value="AAAA" selected?={ record.Type == "AAAA" }>AAAA</option> | ||||
| 					<option value="CNAME" selected?={ record.Type == "CNAME" }>CNAME</option> | ||||
| 					<option value="TXT" selected?={ record.Type == "TXT" }>TXT</option> | ||||
| 					<option value="MX" selected?={ record.Type == "MX" }>MX</option> | ||||
| 				</select> | ||||
| 			</div> | ||||
| 			<div class="mb-3" id="content-group"> | ||||
| 				<label for="record-content" class="form-label">Content</label> | ||||
| 				<input | ||||
| 					type="text" | ||||
| 					class="form-control" | ||||
| 					id="record-content" | ||||
| 					name="content" | ||||
| 					value={ record.Content } | ||||
| 					required | ||||
| 				/> | ||||
| 			</div> | ||||
| 			<div class="mb-3 form-check" id="use-my-ip-group" style="display: none;"> | ||||
| 				<input | ||||
| 					type="checkbox" | ||||
| 					class="form-check-input" | ||||
| 					id="use-my-ip" | ||||
| 					name="use_my_ip" | ||||
| 					onchange="toggleContentField()" | ||||
| 				/> | ||||
| 				<label class="form-check-label" for="use-my-ip">Use my current IP address</label> | ||||
| 			</div> | ||||
| 			<div class="mb-3"> | ||||
| 				<label for="record-ttl" class="form-label">TTL</label> | ||||
| 				<select class="form-select" id="record-ttl" name="ttl"> | ||||
| 					<option value="1" selected?={ record.TTL == 1 }>Auto</option> | ||||
| 					<option value="120" selected?={ record.TTL == 120 }>2 minutes</option> | ||||
| 					<option value="300" selected?={ record.TTL == 300 }>5 minutes</option> | ||||
| 					<option value="600" selected?={ record.TTL == 600 }>10 minutes</option> | ||||
| 					<option value="1800" selected?={ record.TTL == 1800 }>30 minutes</option> | ||||
| 					<option value="3600" selected?={ record.TTL == 3600 }>1 hour</option> | ||||
| 					<option value="7200" selected?={ record.TTL == 7200 }>2 hours</option> | ||||
| 					<option value="18000" selected?={ record.TTL == 18000 }>5 hours</option> | ||||
| 					<option value="43200" selected?={ record.TTL == 43200 }>12 hours</option> | ||||
| 					<option value="86400" selected?={ record.TTL == 86400 }>1 day</option> | ||||
| 				</select> | ||||
| 			</div> | ||||
| 			<div class="mb-3 form-check"> | ||||
| 				<input | ||||
| 					type="checkbox" | ||||
| 					class="form-check-input" | ||||
| 					id="record-proxied" | ||||
| 					name="proxied" | ||||
| 					checked?={ record.Proxied } | ||||
| 				/> | ||||
| 				<label class="form-check-label" for="record-proxied">Proxied through Cloudflare</label> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="modal-footer"> | ||||
| 			<button | ||||
| 				type="button" | ||||
| 				class="btn btn-secondary" | ||||
| 				data-bs-dismiss="modal" | ||||
| 			> | ||||
| 				Cancel | ||||
| 			</button> | ||||
| 			<button type="submit" class="btn btn-primary"> | ||||
| 				Save | ||||
| 				<span class="htmx-indicator spinner-border spinner-border-sm ms-1"></span> | ||||
| 			</button> | ||||
| 		</div> | ||||
| 	</form> | ||||
| 	<script> | ||||
| 		function toggleMyIPOption() { | ||||
| 			const recordType = document.getElementById('record-type').value; | ||||
| 			const useMyIPGroup = document.getElementById('use-my-ip-group'); | ||||
| 			const contentGroup = document.getElementById('content-group'); | ||||
| 
 | ||||
| 			if (recordType === 'A') { | ||||
| 				useMyIPGroup.style.display = 'block'; | ||||
| 			} else { | ||||
| 				useMyIPGroup.style.display = 'none'; | ||||
| 				contentGroup.style.display = 'block'; | ||||
| 				document.getElementById('use-my-ip').checked = false; | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		function toggleContentField() { | ||||
| 			const useMyIP = document.getElementById('use-my-ip').checked; | ||||
| 			const contentGroup = document.getElementById('content-group'); | ||||
| 			contentGroup.style.display = useMyIP ? 'none' : 'block'; | ||||
| 		} | ||||
| 
 | ||||
| 		function resetRecordForm() { | ||||
| 			setTimeout(() => { | ||||
| 				document.getElementById('record-type').value = 'A'; | ||||
| 				toggleMyIPOption(); | ||||
| 			}, 100); | ||||
| 		} | ||||
| 
 | ||||
| 		// Initialize the form state | ||||
| 		toggleMyIPOption(); | ||||
| 	</script> | ||||
| } | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| // Code generated by templ - DO NOT EDIT. | ||||
| 
 | ||||
| // templ: version: v0.3.857 | ||||
| // templ: version: v0.3.898 | ||||
| package templates | ||||
| 
 | ||||
| //lint:file-ignore SA4006 This context is only used if a nested component is present. | ||||
|  | @ -8,8 +8,9 @@ package templates | |||
| import "github.com/a-h/templ" | ||||
| import templruntime "github.com/a-h/templ/runtime" | ||||
| 
 | ||||
| // ConfigModal is the configuration dialog component | ||||
| func ConfigModal() templ.Component { | ||||
| import "fmt" | ||||
| 
 | ||||
| func ConfigModal(config ConfigData, frequencies []UpdateFrequency) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
|  | @ -30,7 +31,92 @@ func ConfigModal() templ.Component { | |||
| 			templ_7745c5c3_Var1 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"modal fade\" id=\"configModal\" tabindex=\"-1\" aria-labelledby=\"configModalLabel\" aria-hidden=\"true\"><div class=\"modal-dialog\"><div class=\"modal-content\"><div class=\"modal-header\"><h5 class=\"modal-title\" id=\"configModalLabel\">Configuration</h5><button type=\"button\" class=\"btn-close\" data-bs-dismiss=\"modal\" aria-label=\"Close\"></button></div><div class=\"modal-body\"><form id=\"config-form\"><div class=\"mb-3\"><label for=\"api-token\" class=\"form-label\">Cloudflare API Token</label> <input type=\"password\" class=\"form-control\" id=\"api-token\" required><div class=\"form-text\">Create a token with <code>Zone.DNS:Edit</code> permissions in the Cloudflare dashboard.</div></div><div class=\"mb-3\"><label for=\"zone-id-input\" class=\"form-label\">Zone ID</label> <input type=\"text\" class=\"form-control\" id=\"zone-id-input\" required><div class=\"form-text\">Found in the Cloudflare dashboard under your domain's overview page.</div></div><div class=\"mb-3\"><label for=\"domain-input\" class=\"form-label\">Domain</label> <input type=\"text\" class=\"form-control\" id=\"domain-input\" value=\"mz.uy\" required></div><div class=\"mb-3\"><label for=\"update-period\" class=\"form-label\">Update Frequency</label> <select class=\"form-select\" id=\"update-period\"><!-- Options will be loaded from API --></select></div></form></div><div class=\"modal-footer\"><button type=\"button\" class=\"btn btn-secondary\" data-bs-dismiss=\"modal\">Cancel</button> <button type=\"button\" class=\"btn btn-primary\" id=\"save-config\">Save</button></div></div></div></div>") | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "<div class=\"modal fade\" id=\"configModal\" tabindex=\"-1\" aria-labelledby=\"configModalLabel\" aria-hidden=\"true\"><div class=\"modal-dialog\"><div class=\"modal-content\"><div class=\"modal-header\"><h5 class=\"modal-title\" id=\"configModalLabel\">Configuration</h5><button type=\"button\" class=\"btn-close\" data-bs-dismiss=\"modal\" aria-label=\"Close\"></button></div><form hx-post=\"/config\" hx-target=\"body\" hx-swap=\"outerHTML\"><div class=\"modal-body\"><div class=\"mb-3\"><label for=\"api-token\" class=\"form-label\">Cloudflare API Token</label> <input type=\"password\" class=\"form-control\" id=\"api-token\" name=\"api_token\" value=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var2 string | ||||
| 		templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(config.ApiToken) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 33, Col: 31} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "\" required><div class=\"form-text\">Create a token with <code>Zone.DNS:Edit</code> permissions in the Cloudflare dashboard.</div></div><div class=\"mb-3\"><label for=\"zone-id-input\" class=\"form-label\">Zone ID</label> <input type=\"text\" class=\"form-control\" id=\"zone-id-input\" name=\"zone_id\" value=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var3 string | ||||
| 		templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(config.ZoneID) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 48, Col: 29} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "\" required><div class=\"form-text\">Found in the Cloudflare dashboard under your domain's overview page.</div></div><div class=\"mb-3\"><label for=\"domain-input\" class=\"form-label\">Domain</label> <input type=\"text\" class=\"form-control\" id=\"domain-input\" name=\"domain\" value=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var4 string | ||||
| 		templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(config.Domain) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 62, Col: 29} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "\" required></div><div class=\"mb-3\"><label for=\"update-period\" class=\"form-label\">Update Frequency</label> <select class=\"form-select\" id=\"update-period\" name=\"update_period\">") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		for _, freq := range frequencies { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "<option value=\"") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			var templ_7745c5c3_Var5 string | ||||
| 			templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(freq.Value) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 70, Col: 35} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "\"") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			if freq.Value == config.UpdatePeriod { | ||||
| 				templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, " selected") | ||||
| 				if templ_7745c5c3_Err != nil { | ||||
| 					return templ_7745c5c3_Err | ||||
| 				} | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, ">") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			var templ_7745c5c3_Var6 string | ||||
| 			templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(freq.Label) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 71, Col: 22} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "</option>") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "</select></div></div><div class=\"modal-footer\"><button type=\"button\" class=\"btn btn-secondary\" data-bs-dismiss=\"modal\">Cancel</button> <button type=\"submit\" class=\"btn btn-primary\">Save <span class=\"htmx-indicator spinner-border spinner-border-sm ms-1\"></span></button></div></form></div></div></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
|  | @ -38,8 +124,7 @@ func ConfigModal() templ.Component { | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // RecordModal is the DNS record dialog component | ||||
| func RecordModal() templ.Component { | ||||
| func RecordModal(domain string) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
|  | @ -55,12 +140,289 @@ func RecordModal() templ.Component { | |||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var2 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var2 == nil { | ||||
| 			templ_7745c5c3_Var2 = templ.NopComponent | ||||
| 		templ_7745c5c3_Var7 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var7 == nil { | ||||
| 			templ_7745c5c3_Var7 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "<div class=\"modal fade\" id=\"recordModal\" tabindex=\"-1\" aria-labelledby=\"recordModalLabel\" aria-hidden=\"true\"><div class=\"modal-dialog\"><div class=\"modal-content\"><div class=\"modal-header\"><h5 class=\"modal-title\" id=\"recordModalLabel\">Add DNS Record</h5><button type=\"button\" class=\"btn-close\" data-bs-dismiss=\"modal\" aria-label=\"Close\"></button></div><div class=\"modal-body\"><form id=\"record-form\"><input type=\"hidden\" id=\"record-id\"><div class=\"mb-3\"><label for=\"record-name\" class=\"form-label\">Name</label><div class=\"input-group\"><input type=\"text\" class=\"form-control\" id=\"record-name\" placeholder=\"subdomain\" required> <span class=\"input-group-text\" id=\"domain-suffix\">.mz.uy</span></div><div class=\"form-text\">Use @ for the root domain</div></div><div class=\"mb-3\"><label for=\"record-type\" class=\"form-label\">Type</label> <select class=\"form-select\" id=\"record-type\"><option value=\"A\">A</option> <option value=\"AAAA\">AAAA</option> <option value=\"CNAME\">CNAME</option> <option value=\"TXT\">TXT</option> <option value=\"MX\">MX</option></select></div><div class=\"mb-3\" id=\"content-group\"><label for=\"record-content\" class=\"form-label\">Content</label> <input type=\"text\" class=\"form-control\" id=\"record-content\" required></div><div class=\"mb-3 form-check\" id=\"use-my-ip-group\"><input type=\"checkbox\" class=\"form-check-input\" id=\"use-my-ip\"> <label class=\"form-check-label\" for=\"use-my-ip\">Use my current IP address</label></div><div class=\"mb-3\"><label for=\"record-ttl\" class=\"form-label\">TTL</label> <select class=\"form-select\" id=\"record-ttl\"><option value=\"1\">Auto</option> <option value=\"120\">2 minutes</option> <option value=\"300\">5 minutes</option> <option value=\"600\">10 minutes</option> <option value=\"1800\">30 minutes</option> <option value=\"3600\">1 hour</option> <option value=\"7200\">2 hours</option> <option value=\"18000\">5 hours</option> <option value=\"43200\">12 hours</option> <option value=\"86400\">1 day</option></select></div><div class=\"mb-3 form-check\"><input type=\"checkbox\" class=\"form-check-input\" id=\"record-proxied\"> <label class=\"form-check-label\" for=\"record-proxied\">Proxied through Cloudflare</label></div></form></div><div class=\"modal-footer\"><button type=\"button\" class=\"btn btn-secondary\" data-bs-dismiss=\"modal\">Cancel</button> <button type=\"button\" class=\"btn btn-primary\" id=\"save-record\">Save</button></div></div></div></div>") | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "<div class=\"modal fade\" id=\"recordModal\" tabindex=\"-1\" aria-labelledby=\"recordModalLabel\" aria-hidden=\"true\"><div class=\"modal-dialog\"><div id=\"record-modal-content\" class=\"modal-content\">") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = RecordForm("Add DNS Record", "", domain, DNSRecord{Type: "A", TTL: 1}).Render(ctx, templ_7745c5c3_Buffer) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "</div></div></div>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func RecordForm(title, recordID, domain string, record DNSRecord) templ.Component { | ||||
| 	return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) { | ||||
| 		templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context | ||||
| 		if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil { | ||||
| 			return templ_7745c5c3_CtxErr | ||||
| 		} | ||||
| 		templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W) | ||||
| 		if !templ_7745c5c3_IsBuffer { | ||||
| 			defer func() { | ||||
| 				templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer) | ||||
| 				if templ_7745c5c3_Err == nil { | ||||
| 					templ_7745c5c3_Err = templ_7745c5c3_BufErr | ||||
| 				} | ||||
| 			}() | ||||
| 		} | ||||
| 		ctx = templ.InitializeContext(ctx) | ||||
| 		templ_7745c5c3_Var8 := templ.GetChildren(ctx) | ||||
| 		if templ_7745c5c3_Var8 == nil { | ||||
| 			templ_7745c5c3_Var8 = templ.NopComponent | ||||
| 		} | ||||
| 		ctx = templ.ClearChildren(ctx) | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "<div class=\"modal-header\"><h5 class=\"modal-title\">") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var9 string | ||||
| 		templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(title) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 114, Col: 33} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "</h5><button type=\"button\" class=\"btn-close\" data-bs-dismiss=\"modal\" aria-label=\"Close\"></button></div><form") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if recordID != "" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, " hx-put=\"") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			var templ_7745c5c3_Var10 string | ||||
| 			templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("/records/%s", recordID)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 124, Col: 48} | ||||
| 			} | ||||
| 			_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10)) | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "\"") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} else { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, " hx-post=\"/records\"") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, " hx-target=\"#dns-records-table\" hx-on::after-request=\"if(event.detail.successful) bootstrap.Modal.getInstance(document.getElementById('recordModal')).hide()\"><div class=\"modal-body\"><div class=\"mb-3\"><label for=\"record-name\" class=\"form-label\">Name</label><div class=\"input-group\"><input type=\"text\" class=\"form-control\" id=\"record-name\" name=\"name\" placeholder=\"subdomain\" value=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var11 string | ||||
| 		templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(getRecordName(record.Name, domain)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 141, Col: 48} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "\" required> <span class=\"input-group-text\">.") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var12 string | ||||
| 		templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(domain) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 144, Col: 45} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "</span></div><div class=\"form-text\">Use @ for the root domain</div></div><div class=\"mb-3\"><label for=\"record-type\" class=\"form-label\">Type</label> <select class=\"form-select\" id=\"record-type\" name=\"type\" onchange=\"toggleMyIPOption()\"><option value=\"A\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "A" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 22, ">A</option> <option value=\"AAAA\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "AAAA" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, ">AAAA</option> <option value=\"CNAME\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "CNAME" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, ">CNAME</option> <option value=\"TXT\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "TXT" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, ">TXT</option> <option value=\"MX\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Type == "MX" { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, ">MX</option></select></div><div class=\"mb-3\" id=\"content-group\"><label for=\"record-content\" class=\"form-label\">Content</label> <input type=\"text\" class=\"form-control\" id=\"record-content\" name=\"content\" value=\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		var templ_7745c5c3_Var13 string | ||||
| 		templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(record.Content) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ.Error{Err: templ_7745c5c3_Err, FileName: `templates/modal.templ`, Line: 170, Col: 27} | ||||
| 		} | ||||
| 		_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13)) | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "\" required></div><div class=\"mb-3 form-check\" id=\"use-my-ip-group\" style=\"display: none;\"><input type=\"checkbox\" class=\"form-check-input\" id=\"use-my-ip\" name=\"use_my_ip\" onchange=\"toggleContentField()\"> <label class=\"form-check-label\" for=\"use-my-ip\">Use my current IP address</label></div><div class=\"mb-3\"><label for=\"record-ttl\" class=\"form-label\">TTL</label> <select class=\"form-select\" id=\"record-ttl\" name=\"ttl\"><option value=\"1\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 1 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, ">Auto</option> <option value=\"120\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 120 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, ">2 minutes</option> <option value=\"300\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 300 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 36, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 37, ">5 minutes</option> <option value=\"600\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 600 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 38, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 39, ">10 minutes</option> <option value=\"1800\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 1800 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 40, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 41, ">30 minutes</option> <option value=\"3600\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 3600 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 42, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 43, ">1 hour</option> <option value=\"7200\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 7200 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 44, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 45, ">2 hours</option> <option value=\"18000\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 18000 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 46, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 47, ">5 hours</option> <option value=\"43200\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 43200 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 48, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 49, ">12 hours</option> <option value=\"86400\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.TTL == 86400 { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 50, " selected") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 51, ">1 day</option></select></div><div class=\"mb-3 form-check\"><input type=\"checkbox\" class=\"form-check-input\" id=\"record-proxied\" name=\"proxied\"") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
| 		if record.Proxied { | ||||
| 			templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 52, " checked") | ||||
| 			if templ_7745c5c3_Err != nil { | ||||
| 				return templ_7745c5c3_Err | ||||
| 			} | ||||
| 		} | ||||
| 		templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 53, "> <label class=\"form-check-label\" for=\"record-proxied\">Proxied through Cloudflare</label></div></div><div class=\"modal-footer\"><button type=\"button\" class=\"btn btn-secondary\" data-bs-dismiss=\"modal\">Cancel</button> <button type=\"submit\" class=\"btn btn-primary\">Save <span class=\"htmx-indicator spinner-border spinner-border-sm ms-1\"></span></button></div></form><script>\n\t\tfunction toggleMyIPOption() {\n\t\t\tconst recordType = document.getElementById('record-type').value;\n\t\t\tconst useMyIPGroup = document.getElementById('use-my-ip-group');\n\t\t\tconst contentGroup = document.getElementById('content-group');\n\n\t\t\tif (recordType === 'A') {\n\t\t\t\tuseMyIPGroup.style.display = 'block';\n\t\t\t} else {\n\t\t\t\tuseMyIPGroup.style.display = 'none';\n\t\t\t\tcontentGroup.style.display = 'block';\n\t\t\t\tdocument.getElementById('use-my-ip').checked = false;\n\t\t\t}\n\t\t}\n\n\t\tfunction toggleContentField() {\n\t\t\tconst useMyIP = document.getElementById('use-my-ip').checked;\n\t\t\tconst contentGroup = document.getElementById('content-group');\n\t\t\tcontentGroup.style.display = useMyIP ? 'none' : 'block';\n\t\t}\n\n\t\tfunction resetRecordForm() {\n\t\t\tsetTimeout(() => {\n\t\t\t\tdocument.getElementById('record-type').value = 'A';\n\t\t\t\ttoggleMyIPOption();\n\t\t\t}, 100);\n\t\t}\n\n\t\t// Initialize the form state\n\t\ttoggleMyIPOption();\n\t</script>") | ||||
| 		if templ_7745c5c3_Err != nil { | ||||
| 			return templ_7745c5c3_Err | ||||
| 		} | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue