Skip to content

Instantly share code, notes, and snippets.

@mkitti
Last active December 7, 2022 02:43
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save mkitti/f051b4df7e054bea53c0903e0fa8d5bb to your computer and use it in GitHub Desktop.
Save mkitti/f051b4df7e054bea53c0903e0fa8d5bb to your computer and use it in GitHub Desktop.
SVG Demo of Fourier Transform
<!DOCTYPE html><html lang="en"><head><meta name="viewport" content="width=device-width"><meta charset="utf-8">
<meta name="pluto-insertion-spot-meta">
<meta name="theme-color" media="(prefers-color-scheme: light)" content="white"><meta name="theme-color" media="(prefers-color-scheme: dark)" content="#2a2928"><meta name="color-scheme" content="light dark"><link rel="icon" type="image/png" sizes="16x16" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/favicon-16x16.347d2855.png" integrity="sha384-3qsGeVLdddzV9oIkj3PhXXQX2CZCjOD/CiyrPQOX6InOWw3HAHClrsQhPfX9uRAj" crossorigin="anonymous"><link rel="icon" type="image/png" sizes="32x32" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/favicon-32x32.8789add4.png" integrity="sha384-cOe5vSoBIgKNgkUL27p9RpsGVY0uBg9PejLccDy+fR8ZD1Iv5dF1MGHjIZAIZwm6" crossorigin="anonymous"><link rel="icon" type="image/png" sizes="96x96" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/favicon-96x96.48689391.png" integrity="sha384-TN49cYb8GyNmrZT14bsYXXo4l1x1NJeJ/EHuVAauAKsNPopPHLojijs9jFT4Vs8c" crossorigin="anonymous"><link rel="pluto-logo-big" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/logo.004c1d7c.svg" integrity="sha384-GkQkODcGxsrSRJCkeakBXihum0GUM44cwBgKyutDimectXCbCgj6Vu3jlrueqEcN" crossorigin="anonymous"><link rel="pluto-logo-small" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/favicon_unsaturated.d1387b25.svg" integrity="sha384-omwjH+Qy3hpAVf5FYd/pkaDBuVAfsEDRN7eBxEA8Ek00OAWP+aiV+GpEYk3I7lyo" crossorigin="anonymous"><script type="module" src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.a387c700.js" integrity="sha384-QIR9QyVw/4Jjo61jdv+/mCwdQswxAKPDEp9b4ZO4iitvft7o7DizyKCKnv5Z2gp5" crossorigin="anonymous"></script><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/juliamono.a2a5b30d.css" integrity="sha384-w9GV7UZkK3QBeqyAZWUEUfw3PoTu18mc0hv4YHtsbAmJztljHyviMYTzE31mbfYz" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.068d9aa1.css" integrity="sha384-aHmbK3lRzFq1wFrbmy+lGOAI6NqK4W9YtrUt5Hsly9hxAUYU8tJ9JNBnixenGWPR" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/vollkorn.089565a8.css" integrity="sha384-jnV/84VtSgBLF70H+s2rxJcOUZIMDR+X/ElFZA83i9ZtZSWiIMFAgPyrWkOJV08q" crossorigin="anonymous"><script defer="">console.log("Pluto.jl, by Fons van der Plas (https://github.com/fonsp), Mikołaj Bochenski (https://github.com/malyvsen), Michiel Dral (https://github.com/dralletje) and friends 🌈");</script><script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.4b96dd74.js" defer="" integrity="sha384-Y+UtWANyXLeovyghpAI8j9/VZXWboD6G5ZCckOsJO99PawAh/utJxFikf62WBaSu" crossorigin="anonymous"></script><script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.9f9dc874.js" defer="" integrity="sha384-tkFo1EK72I9JvoTmHFa199dfRzW8mkXPUkHb/N7UhYI+bxKzX3Kh8LNCZz1ltsFF" crossorigin="anonymous"></script><script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.90ede145.js" defer="" integrity="sha384-CuNU9gQg6fa/yynNqNWjHWzPm4nj+d7O6+HXsNGSqClhs/bYQIbBC3Lw/kh8Ukui" crossorigin="anonymous"></script><script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.dbeed08a.js" defer="" integrity="sha384-1BEdQwXfZi4ZpsNV8w1X8pQcVK1/DS/+/M8OTo3gol7mdEspSN7nT6llX57NQCSt" crossorigin="anonymous"></script><script id="iframe-resizer-content-window-script" src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.6386bd9d.js" crossorigin="anonymous" defer="" integrity="sha384-tgN2a0VDi/lCYwZuDqT7L+A/Y/9kpxf3HV7zv2BJ5Fu7zW0EClq0nM4crfK3TRPs"></script><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.8158fa1d.css" type="text/css" integrity="sha384-GPV/oHrMyayhHoGAwB87CuP4Z/qQDdQDpASH4hYmPhuXXq7f0V9nCWpru4ZBn9Xi" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.9fb5a9e0.css" type="text/css" media="all" data-pluto-file="hide-ui" integrity="sha384-7uAgyRtq5iffqlxCL5SnLdJGDJvclXx1F88O8PTht8UrVz9PXuytcPcxgxC6zeCQ" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.d3307b27.css" type="text/css" integrity="sha384-3x2qZDazgbcSpu1KwtC7K1OB1Tkfg6mRNoxs0wf+u5qGDUVfPW9p0dgiiuz/vnWq" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.e3402eb2.css" type="text/css" integrity="sha384-B+5EZ65SoErJCxHBqVwY2DKgNzKHjf/49SFYB2EfFS8GNvMy/aFGjAPMDiVq+QpM" crossorigin="anonymous"><link rel="preload" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/juliamono.a2a5b30d.css" as="style" integrity="sha384-w9GV7UZkK3QBeqyAZWUEUfw3PoTu18mc0hv4YHtsbAmJztljHyviMYTzE31mbfYz" crossorigin="anonymous"><link rel="preload" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/vollkorn.089565a8.css" as="style" integrity="sha384-jnV/84VtSgBLF70H+s2rxJcOUZIMDR+X/ElFZA83i9ZtZSWiIMFAgPyrWkOJV08q" crossorigin="anonymous"><link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.e82e08bd.css" type="text/css" integrity="sha384-7YN+h8b6N4N65qk8TG/J2KPF95D8z3sGNd06rokz4CX9oWu0KnRAF5cVWu3BkkaN" crossorigin="anonymous"><script data-pluto-file="launch-parameters">
window.pluto_notebook_id = undefined;
window.pluto_isolated_cell_ids = undefined;
window.pluto_notebookfile = "data:text/julia;charset=utf-8;base64,### A Pluto.jl notebook ###
# v0.19.16

using Markdown
using InteractiveUtils

# This Pluto notebook uses @bind for interactivity. When running this notebook outside of Pluto, the following 'mock version' of @bind gives bound variables a default value (instead of an error).
macro bind(def, element)
    quote
        local iv = try Base.loaded_modules[Base.PkgId(Base.UUID("6e696c72-6542-2067-7265-42206c756150"), "AbstractPlutoDingetjes")].Bonds.initial_value catch; b -> missing; end
        local el = $(esc(element))
        global $(esc(def)) = Core.applicable(Base.get, el) ? Base.get(el) : iv(el)
        el
    end
end

# ╔═╡ 362714b0-75d8-11ed-22c0-49cceb161bb5
begin
	using FFTW
	using HypertextLiteral
	θ = range(-π, stop=π, length=101)
end

# ╔═╡ c815e8e1-ffb1-4b93-8f3e-a8fd0c149597
begin
	# flattened
	#svg_x = 100*cos.(θ) .* R_circle_cos.(θ) .+ 200
	#svg_y = 100*sin.(θ) .* R_circle_cos.(θ) .+ 200

	# circle
	svg_x = 100*cos.(θ) .+ 200
	svg_y = 100*sin.(θ) .+ 200
	buf = IOBuffer()
	for (x, y) in zip(svg_x, svg_y)
		print(buf, x ,", ", y, ",")
	end
	str = String(take!(buf)[1:end-1])
	@bind mysvg @htl("""
	<svg id="svg-edit-demo" viewBox="0 0 400 400" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" value="3">
	
	<defs>
	    <circle id="point-handle"
	        r="10" x="0" y="0"
	        stroke-width="4"
	        fill="#fff"
	        fill-opacity="0.4"
	        stroke="#fff"/>
	</defs>
	<circle cx="200" cy="200" r="10" />
	<polygon id="edit-star"
	    stroke="#29e"
	    stroke-width="20"
	    stroke-linejoin="round"
	    fill="none"
	    points="$(str)"/>
	</svg>
	<script src="https://cdn.jsdelivr.net/npm/interactjs@1.10.17/dist/interact.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/fourier/fourier.min.js"></script>
	<script>
	console.log("SVG Script")
  const sns = 'http://www.w3.org/2000/svg'
  const xns = 'http://www.w3.org/1999/xlink'
  const root = document.getElementById('svg-edit-demo')
  const star = document.getElementById('edit-star')
  let rootMatrix
  const originalPoints = []
  let transformedPoints = []

  for (let i = 0, len = star.points.numberOfItems; i < len-20; i += 20) {
    const handle = document.createElementNS(sns, 'use')
    const point = star.points.getItem(i)
    const newPoint = root.createSVGPoint()

    handle.setAttributeNS(xns, 'href', '#point-handle')
    handle.setAttribute('class', 'point-handle')

    handle.x.baseVal.value = newPoint.x = point.x
    handle.y.baseVal.value = newPoint.y = point.y

    handle.setAttribute('data-index', i)

    originalPoints.push(newPoint)

    root.appendChild(handle)
  }

  function applyTransforms (event) {
    rootMatrix = root.getScreenCTM()

    transformedPoints = originalPoints.map((point) => {
      return point.matrixTransform(rootMatrix)
    })

    interact('.point-handle').draggable({
      snap: {
        targets: transformedPoints,
        range: 20 * Math.max(rootMatrix.a, rootMatrix.d),
      },
    })
  }


  function updatePoints() {
    const star = document.querySelector("#edit-star")
	const controlPoints = document.querySelectorAll(".point-handle")
	const num_cp = 5
	const num_pt = 101
	let x = new Array(num_cp)
	let y = new Array(num_cp)
    for(var i = 0; i < num_cp; i++) {
     x[i] = controlPoints[i].x.baseVal.value
     y[i] = controlPoints[i].y.baseVal.value
    }

	//let xhat = new Array(101)
	//let yhat = new Array(101)
	const [xhat, yhat] = fourier.dft(x ,y)
	
    let xlonghat = new Array(num_pt).fill(0.0)
    let ylonghat = new Array(num_pt).fill(0.0)

	xlonghat[0] = xhat[0] * num_pt / num_cp
	xlonghat[1] = xhat[1] * num_pt / num_cp
	xlonghat[2] = xhat[2] * num_pt / num_cp
	xlonghat[99] = xhat[3] * num_pt / num_cp
	xlonghat[100] = xhat[4] * num_pt / num_cp

	ylonghat[0] = yhat[0] * num_pt / num_cp
	ylonghat[1] = yhat[1] * num_pt / num_cp
	ylonghat[2] = yhat[2] * num_pt / num_cp
	ylonghat[99] = yhat[3] * num_pt / num_cp
	ylonghat[100] = yhat[4] * num_pt / num_cp

	//let xlong = new Array(101)
	//let ylong = new Array(101)
	const [xlong, ylong] = fourier.idft(xlonghat, ylonghat)
	
    for (var i=0; i < num_pt; i++) {
      star.points[i].x = xlong[i]
      star.points[i].y = ylong[i]
    }
	
  }

  function updatePointsRadial() {
    const star = document.querySelector("#edit-star")
	const controlPoints = document.querySelectorAll(".point-handle")
	const num_cp = 5
	const num_pt = 100
	let x = new Array(num_cp)
	let y = new Array(num_cp)
	let r = new Array(num_cp)
    for(var i = 0; i < num_cp; i++) {
     x[i] = controlPoints[i].x.baseVal.value
     y[i] = controlPoints[i].y.baseVal.value
	 r[i] = Math.hypot(x[i]-200, y[i]-200)
	 //console.log(r[i])
    }
	

	let z = new Array(num_cp).fill(0.0)
	const [rhat, ihat] = fourier.dft(r, z)

	let rlonghat = new Array(num_pt).fill(0.0)
	let ilonghat = new Array(num_pt).fill(0.0)

	rlonghat[0] = rhat[0] * num_pt / num_cp
	rlonghat[1] = rhat[1] * num_pt / num_cp
	rlonghat[2] = rhat[2] * num_pt / num_cp
	rlonghat[99] = rhat[3] * num_pt / num_cp
	rlonghat[100] = rhat[4] * num_pt / num_cp

	ilonghat[0] = ihat[0] * num_pt / num_cp
	ilonghat[1] = ihat[1] * num_pt / num_cp
	ilonghat[2] = ihat[2] * num_pt / num_cp
	ilonghat[99] = ihat[3] * num_pt / num_cp
	ilonghat[100] = ihat[4] * num_pt / num_cp

	const [rlong, _] = fourier.idft(rlonghat, ilonghat)
	
	//let xlong = new Array(num_pt)
	//let ylong = new Array(num_pt)

	let theta = -Math.PI

	for(var i=0; i < num_pt; i++) {
	   //console.log(rlong[i])
	   star.points[i].x = rlong[i] * Math.cos(theta) + 200
	   star.points[i].y = rlong[i] * Math.sin(theta) + 200
	   theta += 2 * Math.PI / (num_pt)
	}

	star.points[num_pt].x = star.points[0].x
	star.points[num_pt].y = star.points[0].y

	
  }

  interact(root).on('mousedown', applyTransforms).on('touchstart', applyTransforms)

  interact('.point-handle')
    .draggable({
      onstart: function (event) {
        root.setAttribute('class', 'dragging')
      },
      onmove: function (event) {
        const i = event.target.getAttribute('data-index') | 0
        const point = star.points.getItem(i)

		//Get angle and radius
		const theta = Math.atan2(point.y - 200, point.x - 200)

	    //unconstrained
        point.x += event.dx / rootMatrix.a
        point.y += event.dy / rootMatrix.d

	    //radially constrained
		const radial = Math.hypot(point.y - 200, point.x - 200)	    
	    point.x = radial * Math.cos(theta) + 200
	    point.y = radial * Math.sin(theta) + 200

        event.target.x.baseVal.value = point.x
        event.target.y.baseVal.value = point.y

		//updatePointsRadial()
	    //updatePoints()
      },
      onend: function (event) {
        root.setAttribute('class', '')
	    updatePointsRadial()
      },
      snap: {
        targets: originalPoints,
        range: 10,
        relativePoints: [{ x: 0.5, y: 0.5 }],
      },
      restrict: { restriction: document.rootElement },
    })
    .styleCursor(false)

  document.addEventListener('dragstart', (event) => {
    event.preventDefault()
  })
</script>
	""")
end

# ╔═╡ 00000000-0000-0000-0000-000000000001
PLUTO_PROJECT_TOML_CONTENTS = """
[deps]
FFTW = "7a1cc6ca-52ef-59f5-83cd-3a7055c09341"
HypertextLiteral = "ac1192a8-f4b3-4bfe-ba22-af5b92cd3ab2"

[compat]
FFTW = "~1.5.0"
HypertextLiteral = "~0.9.4"
"""

# ╔═╡ 00000000-0000-0000-0000-000000000002
PLUTO_MANIFEST_TOML_CONTENTS = """
# This file is machine-generated - editing it directly is not advised

julia_version = "1.8.3"
manifest_format = "2.0"
project_hash = "c053c543fca201396c653279bf46b8ff8eeff446"

[[deps.AbstractFFTs]]
deps = ["ChainRulesCore", "LinearAlgebra"]
git-tree-sha1 = "69f7020bd72f069c219b5e8c236c1fa90d2cb409"
uuid = "621f4979-c628-5d54-868e-fcf4e3e8185c"
version = "1.2.1"

[[deps.ArgTools]]
uuid = "0dad84c5-d112-42e6-8d28-ef12dabb789f"
version = "1.1.1"

[[deps.Artifacts]]
uuid = "56f22d72-fd6d-98f1-02f0-08ddc0907c33"

[[deps.Base64]]
uuid = "2a0f44e3-6c83-55bd-87e4-b1978d98bd5f"

[[deps.ChainRulesCore]]
deps = ["Compat", "LinearAlgebra", "SparseArrays"]
git-tree-sha1 = "e7ff6cadf743c098e08fca25c91103ee4303c9bb"
uuid = "d360d2e6-b24c-11e9-a2a3-2a2ae2dbcce4"
version = "1.15.6"

[[deps.Compat]]
deps = ["Dates", "LinearAlgebra", "UUIDs"]
git-tree-sha1 = "00a2cccc7f098ff3b66806862d275ca3db9e6e5a"
uuid = "34da2185-b29b-5c13-b0c7-acf172513d20"
version = "4.5.0"

[[deps.CompilerSupportLibraries_jll]]
deps = ["Artifacts", "Libdl"]
uuid = "e66e0078-7015-5450-92f7-15fbd957f2ae"
version = "0.5.2+0"

[[deps.Dates]]
deps = ["Printf"]
uuid = "ade2ca70-3891-5945-98fb-dc099432e06a"

[[deps.Downloads]]
deps = ["ArgTools", "FileWatching", "LibCURL", "NetworkOptions"]
uuid = "f43a241f-c20a-4ad4-852c-f6b1247861c6"
version = "1.6.0"

[[deps.FFTW]]
deps = ["AbstractFFTs", "FFTW_jll", "LinearAlgebra", "MKL_jll", "Preferences", "Reexport"]
git-tree-sha1 = "90630efff0894f8142308e334473eba54c433549"
uuid = "7a1cc6ca-52ef-59f5-83cd-3a7055c09341"
version = "1.5.0"

[[deps.FFTW_jll]]
deps = ["Artifacts", "JLLWrappers", "Libdl", "Pkg"]
git-tree-sha1 = "c6033cc3892d0ef5bb9cd29b7f2f0331ea5184ea"
uuid = "f5851436-0d7a-5f13-b9de-f02708fd171a"
version = "3.3.10+0"

[[deps.FileWatching]]
uuid = "7b1f6079-737a-58dc-b8bc-7a2ca5c1b5ee"

[[deps.HypertextLiteral]]
deps = ["Tricks"]
git-tree-sha1 = "c47c5fa4c5308f27ccaac35504858d8914e102f9"
uuid = "ac1192a8-f4b3-4bfe-ba22-af5b92cd3ab2"
version = "0.9.4"

[[deps.IntelOpenMP_jll]]
deps = ["Artifacts", "JLLWrappers", "Libdl", "Pkg"]
git-tree-sha1 = "d979e54b71da82f3a65b62553da4fc3d18c9004c"
uuid = "1d5cc7b8-4909-519e-a0f8-d0f5ad9712d0"
version = "2018.0.3+2"

[[deps.InteractiveUtils]]
deps = ["Markdown"]
uuid = "b77e0a4c-d291-57a0-90e8-8db25a27a240"

[[deps.JLLWrappers]]
deps = ["Preferences"]
git-tree-sha1 = "abc9885a7ca2052a736a600f7fa66209f96506e1"
uuid = "692b3bcd-3c85-4b1f-b108-f13ce0eb3210"
version = "1.4.1"

[[deps.LazyArtifacts]]
deps = ["Artifacts", "Pkg"]
uuid = "4af54fe1-eca0-43a8-85a7-787d91b784e3"

[[deps.LibCURL]]
deps = ["LibCURL_jll", "MozillaCACerts_jll"]
uuid = "b27032c2-a3e7-50c8-80cd-2d36dbcbfd21"
version = "0.6.3"

[[deps.LibCURL_jll]]
deps = ["Artifacts", "LibSSH2_jll", "Libdl", "MbedTLS_jll", "Zlib_jll", "nghttp2_jll"]
uuid = "deac9b47-8bc7-5906-a0fe-35ac56dc84c0"
version = "7.84.0+0"

[[deps.LibGit2]]
deps = ["Base64", "NetworkOptions", "Printf", "SHA"]
uuid = "76f85450-5226-5b5a-8eaa-529ad045b433"

[[deps.LibSSH2_jll]]
deps = ["Artifacts", "Libdl", "MbedTLS_jll"]
uuid = "29816b5a-b9ab-546f-933c-edad1886dfa8"
version = "1.10.2+0"

[[deps.Libdl]]
uuid = "8f399da3-3557-5675-b5ff-fb832c97cbdb"

[[deps.LinearAlgebra]]
deps = ["Libdl", "libblastrampoline_jll"]
uuid = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e"

[[deps.Logging]]
uuid = "56ddb016-857b-54e1-b83d-db4d58db5568"

[[deps.MKL_jll]]
deps = ["Artifacts", "IntelOpenMP_jll", "JLLWrappers", "LazyArtifacts", "Libdl", "Pkg"]
git-tree-sha1 = "2ce8695e1e699b68702c03402672a69f54b8aca9"
uuid = "856f044c-d86e-5d09-b602-aeab76dc8ba7"
version = "2022.2.0+0"

[[deps.Markdown]]
deps = ["Base64"]
uuid = "d6f4376e-aef5-505a-96c1-9c027394607a"

[[deps.MbedTLS_jll]]
deps = ["Artifacts", "Libdl"]
uuid = "c8ffd9c3-330d-5841-b78e-0817d7145fa1"
version = "2.28.0+0"

[[deps.MozillaCACerts_jll]]
uuid = "14a3606d-f60d-562e-9121-12d972cd8159"
version = "2022.2.1"

[[deps.NetworkOptions]]
uuid = "ca575930-c2e3-43a9-ace4-1e988b2c1908"
version = "1.2.0"

[[deps.OpenBLAS_jll]]
deps = ["Artifacts", "CompilerSupportLibraries_jll", "Libdl"]
uuid = "4536629a-c528-5b80-bd46-f80d51c5b363"
version = "0.3.20+0"

[[deps.Pkg]]
deps = ["Artifacts", "Dates", "Downloads", "LibGit2", "Libdl", "Logging", "Markdown", "Printf", "REPL", "Random", "SHA", "Serialization", "TOML", "Tar", "UUIDs", "p7zip_jll"]
uuid = "44cfe95a-1eb2-52ea-b672-e2afdf69b78f"
version = "1.8.0"

[[deps.Preferences]]
deps = ["TOML"]
git-tree-sha1 = "47e5f437cc0e7ef2ce8406ce1e7e24d44915f88d"
uuid = "21216c6a-2e73-6563-6e65-726566657250"
version = "1.3.0"

[[deps.Printf]]
deps = ["Unicode"]
uuid = "de0858da-6303-5e67-8744-51eddeeeb8d7"

[[deps.REPL]]
deps = ["InteractiveUtils", "Markdown", "Sockets", "Unicode"]
uuid = "3fa0cd96-eef1-5676-8a61-b3b8758bbffb"

[[deps.Random]]
deps = ["SHA", "Serialization"]
uuid = "9a3f8284-a2c9-5f02-9a11-845980a1fd5c"

[[deps.Reexport]]
git-tree-sha1 = "45e428421666073eab6f2da5c9d310d99bb12f9b"
uuid = "189a3867-3050-52da-a836-e630ba90ab69"
version = "1.2.2"

[[deps.SHA]]
uuid = "ea8e919c-243c-51af-8825-aaa63cd721ce"
version = "0.7.0"

[[deps.Serialization]]
uuid = "9e88b42a-f829-5b0c-bbe9-9e923198166b"

[[deps.Sockets]]
uuid = "6462fe0b-24de-5631-8697-dd941f90decc"

[[deps.SparseArrays]]
deps = ["LinearAlgebra", "Random"]
uuid = "2f01184e-e22b-5df5-ae63-d93ebab69eaf"

[[deps.TOML]]
deps = ["Dates"]
uuid = "fa267f1f-6049-4f14-aa54-33bafae1ed76"
version = "1.0.0"

[[deps.Tar]]
deps = ["ArgTools", "SHA"]
uuid = "a4e569a6-e804-4fa4-b0f3-eef7a1d5b13e"
version = "1.10.1"

[[deps.Tricks]]
git-tree-sha1 = "6bac775f2d42a611cdfcd1fb217ee719630c4175"
uuid = "410a4b4d-49e4-4fbc-ab6d-cb71b17b3775"
version = "0.1.6"

[[deps.UUIDs]]
deps = ["Random", "SHA"]
uuid = "cf7118a7-6976-5b1a-9a39-7adc72f591a4"

[[deps.Unicode]]
uuid = "4ec0a83e-493e-50e2-b9ac-8f72acf5a8f5"

[[deps.Zlib_jll]]
deps = ["Libdl"]
uuid = "83775a58-1f1d-513f-b197-d71354ab007a"
version = "1.2.12+3"

[[deps.libblastrampoline_jll]]
deps = ["Artifacts", "Libdl", "OpenBLAS_jll"]
uuid = "8e850b90-86db-534c-a0d3-1478176c7d93"
version = "5.1.1+0"

[[deps.nghttp2_jll]]
deps = ["Artifacts", "Libdl"]
uuid = "8e850ede-7688-5339-a07c-302acd2aaf8d"
version = "1.48.0+0"

[[deps.p7zip_jll]]
deps = ["Artifacts", "Libdl"]
uuid = "3f19e933-33d8-53b3-aaab-bd5110c3b7a0"
version = "17.4.0+0"
"""

# ╔═╡ Cell order:
# ╠═362714b0-75d8-11ed-22c0-49cceb161bb5
# ╠═c815e8e1-ffb1-4b93-8f3e-a8fd0c149597
# ╟─00000000-0000-0000-0000-000000000001
# ╟─00000000-0000-0000-0000-000000000002
";
window.pluto_disable_ui = true;
window.pluto_slider_server_url = undefined;
window.pluto_binder_url = "https://mybinder.org/v2/gh/fonsp/pluto-on-binder/v0.19.16";
window.pluto_statefile = "data:;base64,3gAQpWJvbmRzgaVteXN2Z4GldmFsdWXArGNlbGxfcmVzdWx0c4LZJDM2MjcxNGIwLTc1ZDgtMTFlZC0yMmMwLTQ5Y2NlYjE2MWJiNYqmcXVldWVkwqRsb2dzkKdydW5uaW5nwqZvdXRwdXSGpGJvZHnZOC0zLjE0MTU5MjY1MzU4OTc5MzowLjA2MjgzMTg1MzA3MTc5NTg3OjMuMTQxNTkyNjUzNTg5NzkzpG1pbWWqdGV4dC9wbGFpbqxyb290YXNzaWduZWXAsmxhc3RfcnVuX3RpbWVzdGFtcMtB2OP/NxFoc7BwZXJzaXN0X2pzX3N0YXRlwrdoYXNfcGx1dG9faG9va19mZWF0dXJlc8KnY2VsbF9pZNkkMzYyNzE0YjAtNzVkOC0xMWVkLTIyYzAtNDljY2ViMTYxYmI1uWRlcGVuZHNfb25fZGlzYWJsZWRfY2VsbHPCp3J1bnRpbWXOAOyNbLVwdWJsaXNoZWRfb2JqZWN0X2tleXOQuGRlcGVuZHNfb25fc2tpcHBlZF9jZWxsc8KnZXJyb3JlZMLZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N4qmcXVldWVkwqRsb2dzkKdydW5uaW5nwqZvdXRwdXSGpGJvZHnaJds8Ym9uZCBkZWY9Im15c3ZnIiB1bmlxdWVfaWQ9InduUiYjNDM7dWwvbHBjYnkiPgk8c3ZnIGlkPSJzdmctZWRpdC1kZW1vIiB2aWV3Qm94PSIwIDAgNDAwIDQwMCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmFsdWU9IjMiPgoJCgk8ZGVmcz4KCSAgICA8Y2lyY2xlIGlkPSJwb2ludC1oYW5kbGUiCgkgICAgICAgIHI9IjEwIiB4PSIwIiB5PSIwIgoJICAgICAgICBzdHJva2Utd2lkdGg9IjQiCgkgICAgICAgIGZpbGw9IiNmZmYiCgkgICAgICAgIGZpbGwtb3BhY2l0eT0iMC40IgoJICAgICAgICBzdHJva2U9IiNmZmYiLz4KCTwvZGVmcz4KCTxjaXJjbGUgY3g9IjIwMCIgY3k9IjIwMCIgcj0iMTAiIC8+Cgk8cG9seWdvbiBpZD0iZWRpdC1zdGFyIgoJICAgIHN0cm9rZT0iIzI5ZSIKCSAgICBzdHJva2Utd2lkdGg9IjIwIgoJICAgIHN0cm9rZS1saW5lam9pbj0icm91bmQiCgkgICAgZmlsbD0ibm9uZSIKCSAgICBwb2ludHM9IjEwMC4wLCAyMDAuMCwxMDAuMTk3MzI3MTU3MTcyODUsIDE5My43MjA5NDgwNDcwNjg2NSwxMDAuNzg4NTI5ODY4NTUyMjIsIDE4Ny40NjY2NzY2NDM1Njk1NCwxMDEuNzcxMjc0OTI3MTMxMTMsIDE4MS4yNjE4Njg1NDE0Mjc1MywxMDMuMTQxNjgzODg3MTM2ODksIDE3NS4xMzEwMTEyODM1MTQ1LDEwNC44OTQzNDgzNzA0ODQ2NSwgMTY5LjA5ODMwMDU2MjUwNTI0LDEwNy4wMjIzNTE0MTExNzQ4NywgMTYzLjE4NzU0NDczMTUzMjE3LDEwOS41MTcyOTQ3NTMzOTgwNiwgMTU3LjQyMjA3MDg0MzQ5MjcyLDExMi4zNjkzMzE5OTU2MTM2MywgMTUxLjgyNDYzMjU4OTgyODQ3LDExNS41NjcyMDc0NDk3OTg0OSwgMTQ2LjQxNzMyMDUwMjEwMDMyLDExOS4wOTgzMDA1NjI1MDUyNywgMTQxLjIyMTQ3NDc3MDc1MjY4LDEyMi45NDg2NzU3MjI0MjEwOSwgMTM2LjI1NzYwMTAyNTEzMSwxMjcuMTAzMTM3MjU3ODU4ODYsIDEzMS41NDUyODk0MDcxMzExMiwxMzEuNTQ1Mjg5NDA3MTMxMTIsIDEyNy4xMDMxMzcyNTc4NTg4NiwxMzYuMjU3NjAxMDI1MTMxLCAxMjIuOTQ4Njc1NzIyNDIxMDcsMTQxLjIyMTQ3NDc3MDc1MjcsIDExOS4wOTgzMDA1NjI1MDUyNiwxNDYuNDE3MzIwNTAyMTAwMzUsIDExNS41NjcyMDc0NDk3OTg0OSwxNTEuODI0NjMyNTg5ODI4NSwgMTEyLjM2OTMzMTk5NTYxMzYzLDE1Ny40MjIwNzA4NDM0OTI3MiwgMTA5LjUxNzI5NDc1MzM5ODA1LDE2My4xODc1NDQ3MzE1MzIyLCAxMDcuMDIyMzUxNDExMTc0ODUsMTY5LjA5ODMwMDU2MjUwNTI3LCAxMDQuODk0MzQ4MzcwNDg0NjQsMTc1LjEzMTAxMTI4MzUxNDU0LCAxMDMuMTQxNjgzODg3MTM2ODgsMTgxLjI2MTg2ODU0MTQyNzUzLCAxMDEuNzcxMjc0OTI3MTMxMTMsMTg3LjQ2NjY3NjY0MzU2OTYsIDEwMC43ODg1Mjk4Njg1NTIyMiwxOTMuNzIwOTQ4MDQ3MDY4NjUsIDEwMC4xOTczMjcxNTcxNzI4NSwyMDAuMCwgMTAwLjAsMjA2LjI3OTA1MTk1MjkzMTM1LCAxMDAuMTk3MzI3MTU3MTcyODUsMjEyLjUzMzMyMzM1NjQzMDQzLCAxMDAuNzg4NTI5ODY4NTUyMjIsMjE4LjczODEzMTQ1ODU3MjQ3LCAxMDEuNzcxMjc0OTI3MTMxMTUsMjI0Ljg2ODk4ODcxNjQ4NTUsIDEwMy4xNDE2ODM4ODcxMzY4OSwyMzAuOTAxNjk5NDM3NDk0NzYsIDEwNC44OTQzNDgzNzA0ODQ2NSwyMzYuODEyNDU1MjY4NDY3OCwgMTA3LjAyMjM1MTQxMTE3NDg3LDI0Mi41Nzc5MjkxNTY1MDcyNSwgMTA5LjUxNzI5NDc1MzM5ODA1LDI0OC4xNzUzNjc0MTAxNzE1MywgMTEyLjM2OTMzMTk5NTYxMzYzLDI1My41ODI2Nzk0OTc4OTk2NSwgMTE1LjU2NzIwNzQ0OTc5ODQ5LDI1OC43Nzg1MjUyMjkyNDczLCAxMTkuMDk4MzAwNTYyNTA1MjYsMjYzLjc0MjM5ODk3NDg2OSwgMTIyLjk0ODY3NTcyMjQyMTA3LDI2OC40NTQ3MTA1OTI4Njg5LCAxMjcuMTAzMTM3MjU3ODU4ODUsMjcyLjg5Njg2Mjc0MjE0MTE0LCAxMzEuNTQ1Mjg5NDA3MTMxMTIsMjc3LjA1MTMyNDI3NzU3ODk0LCAxMzYuMjU3NjAxMDI1MTMxMDQsMjgwLjkwMTY5OTQzNzQ5NDc2LCAxNDEuMjIxNDc0NzcwNzUyNjgsMjg0LjQzMjc5MjU1MDIwMTUsIDE0Ni40MTczMjA1MDIxMDAzMiwyODcuNjMwNjY4MDA0Mzg2NCwgMTUxLjgyNDYzMjU4OTgyODQ3LDI5MC40ODI3MDUyNDY2MDE5NCwgMTU3LjQyMjA3MDg0MzQ5Mjc1LDI5Mi45Nzc2NDg1ODg4MjUxNiwgMTYzLjE4NzU0NDczMTUzMjIsMjk1LjEwNTY1MTYyOTUxNTM2LCAxNjkuMDk4MzAwNTYyNTA1MjcsMjk2Ljg1ODMxNjExMjg2MzEsIDE3NS4xMzEwMTEyODM1MTQ1LDI5OC4yMjg3MjUwNzI4Njg5LCAxODEuMjYxODY4NTQxNDI3NTMsMjk5LjIxMTQ3MDEzMTQ0Nzc1LCAxODcuNDY2Njc2NjQzNTY5NTcsMjk5LjgwMjY3Mjg0MjgyNzIsIDE5My43MjA5NDgwNDcwNjg2NywzMDAuMCwgMjAwLjAsMjk5LjgwMjY3Mjg0MjgyNzIsIDIwNi4yNzkwNTE5NTI5MzEzMywyOTkuMjExNDcwMTMxNDQ3NzUsIDIxMi41MzMzMjMzNTY0MzA0MywyOTguMjI4NzI1MDcyODY4OSwgMjE4LjczODEzMTQ1ODU3MjQ3LDI5Ni44NTgzMTYxMTI4NjMxLCAyMjQuODY4OTg4NzE2NDg1NSwyOTUuMTA1NjUxNjI5NTE1MzYsIDIzMC45MDE2OTk0Mzc0OTQ3MywyOTIuOTc3NjQ4NTg4ODI1MTYsIDIzNi44MTI0NTUyNjg0Njc4LDI5MC40ODI3MDUyNDY2MDE5NCwgMjQyLjU3NzkyOTE1NjUwNzI1LDI4Ny42MzA2NjgwMDQzODY0LCAyNDguMTc1MzY3NDEwMTcxNTMsMjg0LjQzMjc5MjU1MDIwMTUsIDI1My41ODI2Nzk0OTc4OTk2OCwyODAuOTAxNjk5NDM3NDk0NzYsIDI1OC43Nzg1MjUyMjkyNDczLDI3Ny4wNTEzMjQyNzc1Nzg5NCwgMjYzLjc0MjM5ODk3NDg2ODkzLDI3Mi44OTY4NjI3NDIxNDExNCwgMjY4LjQ1NDcxMDU5Mjg2ODksMjY4LjQ1NDcxMDU5Mjg2ODksIDI3Mi44OTY4NjI3NDIxNDExNCwyNjMuNzQyMzk4OTc0ODY5LCAyNzcuMDUxMzI0Mjc3NTc4OTQsMjU4Ljc3ODUyNTIyOTI0NzMsIDI4MC45MDE2OTk0Mzc0OTQ3NiwyNTMuNTgyNjc5NDk3ODk5NjUsIDI4NC40MzI3OTI1NTAyMDE1LDI0OC4xNzUzNjc0MTAxNzE1MywgMjg3LjYzMDY2ODAwNDM4NjQsMjQyLjU3NzkyOTE1NjUwNzI1LCAyOTAuNDgyNzA1MjQ2NjAxOTQsMjM2LjgxMjQ1NTI2ODQ2NzgsIDI5Mi45Nzc2NDg1ODg4MjUxLDIzMC45MDE2OTk0Mzc0OTQ3NiwgMjk1LjEwNTY1MTYyOTUxNTM2LDIyNC44Njg5ODg3MTY0ODU1LCAyOTYuODU4MzE2MTEyODYzMSwyMTguNzM4MTMxNDU4NTcyNDcsIDI5OC4yMjg3MjUwNzI4Njg5LDIxMi41MzMzMjMzNTY0MzA0MywgMjk5LjIxMTQ3MDEzMTQ0Nzc1LDIwNi4yNzkwNTE5NTI5MzEzNSwgMjk5LjgwMjY3Mjg0MjgyNzIsMjAwLjAsIDMwMC4wLDE5My43MjA5NDgwNDcwNjg2NSwgMjk5LjgwMjY3Mjg0MjgyNzIsMTg3LjQ2NjY3NjY0MzU2OTYsIDI5OS4yMTE0NzAxMzE0NDc3NSwxODEuMjYxODY4NTQxNDI3NTMsIDI5OC4yMjg3MjUwNzI4Njg5LDE3NS4xMzEwMTEyODM1MTQ1NCwgMjk2Ljg1ODMxNjExMjg2MzEsMTY5LjA5ODMwMDU2MjUwNTI3LCAyOTUuMTA1NjUxNjI5NTE1MzYsMTYzLjE4NzU0NDczMTUzMjIsIDI5Mi45Nzc2NDg1ODg4MjUxNiwxNTcuNDIyMDcwODQzNDkyNzIsIDI5MC40ODI3MDUyNDY2MDE5NCwxNTEuODI0NjMyNTg5ODI4NSwgMjg3LjYzMDY2ODAwNDM4NjQsMTQ2LjQxNzMyMDUwMjEwMDM1LCAyODQuNDMyNzkyNTUwMjAxNSwxNDEuMjIxNDc0NzcwNzUyNywgMjgwLjkwMTY5OTQzNzQ5NDc2LDEzNi4yNTc2MDEwMjUxMzEsIDI3Ny4wNTEzMjQyNzc1Nzg5NCwxMzEuNTQ1Mjg5NDA3MTMxMTIsIDI3Mi44OTY4NjI3NDIxNDExNCwxMjcuMTAzMTM3MjU3ODU4ODYsIDI2OC40NTQ3MTA1OTI4Njg5LDEyMi45NDg2NzU3MjI0MjEwOSwgMjYzLjc0MjM5ODk3NDg2OSwxMTkuMDk4MzAwNTYyNTA1MjcsIDI1OC43Nzg1MjUyMjkyNDczLDExNS41NjcyMDc0NDk3OTg0OSwgMjUzLjU4MjY3OTQ5Nzg5OTY4LDExMi4zNjkzMzE5OTU2MTM2MywgMjQ4LjE3NTM2NzQxMDE3MTUzLDEwOS41MTcyOTQ3NTMzOTgwNiwgMjQyLjU3NzkyOTE1NjUwNzI4LDEwNy4wMjIzNTE0MTExNzQ4NywgMjM2LjgxMjQ1NTI2ODQ2NzgzLDEwNC44OTQzNDgzNzA0ODQ2NSwgMjMwLjkwMTY5OTQzNzQ5NDc2LDEwMy4xNDE2ODM4ODcxMzY4OSwgMjI0Ljg2ODk4ODcxNjQ4NTUsMTAxLjc3MTI3NDkyNzEzMTEzLCAyMTguNzM4MTMxNDU4NTcyNDcsMTAwLjc4ODUyOTg2ODU1MjIyLCAyMTIuNTMzMzIzMzU2NDMwNDYsMTAwLjE5NzMyNzE1NzE3Mjg1LCAyMDYuMjc5MDUxOTUyOTMxMzUsMTAwLjAsIDIwMC4wIi8+Cgk8L3N2Zz4KCTxzY3JpcHQgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2ludGVyYWN0anNAMS4xMC4xNy9kaXN0L2ludGVyYWN0Lm1pbi5qcyI+PC9zY3JpcHQ+CjxzY3JpcHQgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2ZvdXJpZXIvZm91cmllci5taW4uanMiPjwvc2NyaXB0PgoJPHNjcmlwdD4KCWNvbnNvbGUubG9nKCJTVkcgU2NyaXB0IikKICBjb25zdCBzbnMgPSAnaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnCiAgY29uc3QgeG5zID0gJ2h0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsnCiAgY29uc3Qgcm9vdCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdzdmctZWRpdC1kZW1vJykKICBjb25zdCBzdGFyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2VkaXQtc3RhcicpCiAgbGV0IHJvb3RNYXRyaXgKICBjb25zdCBvcmlnaW5hbFBvaW50cyA9IFtdCiAgbGV0IHRyYW5zZm9ybWVkUG9pbnRzID0gW10KCiAgZm9yIChsZXQgaSA9IDAsIGxlbiA9IHN0YXIucG9pbnRzLm51bWJlck9mSXRlbXM7IGkgPCBsZW4tMjA7IGkgKz0gMjApIHsKICAgIGNvbnN0IGhhbmRsZSA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzbnMsICd1c2UnKQogICAgY29uc3QgcG9pbnQgPSBzdGFyLnBvaW50cy5nZXRJdGVtKGkpCiAgICBjb25zdCBuZXdQb2ludCA9IHJvb3QuY3JlYXRlU1ZHUG9pbnQoKQoKICAgIGhhbmRsZS5zZXRBdHRyaWJ1dGVOUyh4bnMsICdocmVmJywgJyNwb2ludC1oYW5kbGUnKQogICAgaGFuZGxlLnNldEF0dHJpYnV0ZSgnY2xhc3MnLCAncG9pbnQtaGFuZGxlJykKCiAgICBoYW5kbGUueC5iYXNlVmFsLnZhbHVlID0gbmV3UG9pbnQueCA9IHBvaW50LngKICAgIGhhbmRsZS55LmJhc2VWYWwudmFsdWUgPSBuZXdQb2ludC55ID0gcG9pbnQueQoKICAgIGhhbmRsZS5zZXRBdHRyaWJ1dGUoJ2RhdGEtaW5kZXgnLCBpKQoKICAgIG9yaWdpbmFsUG9pbnRzLnB1c2gobmV3UG9pbnQpCgogICAgcm9vdC5hcHBlbmRDaGlsZChoYW5kbGUpCiAgfQoKICBmdW5jdGlvbiBhcHBseVRyYW5zZm9ybXMgKGV2ZW50KSB7CiAgICByb290TWF0cml4ID0gcm9vdC5nZXRTY3JlZW5DVE0oKQoKICAgIHRyYW5zZm9ybWVkUG9pbnRzID0gb3JpZ2luYWxQb2ludHMubWFwKChwb2ludCkgPT4gewogICAgICByZXR1cm4gcG9pbnQubWF0cml4VHJhbnNmb3JtKHJvb3RNYXRyaXgpCiAgICB9KQoKICAgIGludGVyYWN0KCcucG9pbnQtaGFuZGxlJykuZHJhZ2dhYmxlKHsKICAgICAgc25hcDogewogICAgICAgIHRhcmdldHM6IHRyYW5zZm9ybWVkUG9pbnRzLAogICAgICAgIHJhbmdlOiAyMCAqIE1hdGgubWF4KHJvb3RNYXRyaXguYSwgcm9vdE1hdHJpeC5kKSwKICAgICAgfSwKICAgIH0pCiAgfQoKCiAgZnVuY3Rpb24gdXBkYXRlUG9pbnRzKCkgewogICAgY29uc3Qgc3RhciA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3IoIiNlZGl0LXN0YXIiKQoJY29uc3QgY29udHJvbFBvaW50cyA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoIi5wb2ludC1oYW5kbGUiKQoJY29uc3QgbnVtX2NwID0gNQoJY29uc3QgbnVtX3B0ID0gMTAxCglsZXQgeCA9IG5ldyBBcnJheShudW1fY3ApCglsZXQgeSA9IG5ldyBBcnJheShudW1fY3ApCiAgICBmb3IodmFyIGkgPSAwOyBpIDwgbnVtX2NwOyBpKyspIHsKICAgICB4W2ldID0gY29udHJvbFBvaW50c1tpXS54LmJhc2VWYWwudmFsdWUKICAgICB5W2ldID0gY29udHJvbFBvaW50c1tpXS55LmJhc2VWYWwudmFsdWUKICAgIH0KCgkvL2xldCB4aGF0ID0gbmV3IEFycmF5KDEwMSkKCS8vbGV0IHloYXQgPSBuZXcgQXJyYXkoMTAxKQoJY29uc3QgW3hoYXQsIHloYXRdID0gZm91cmllci5kZnQoeCAseSkKCQogICAgbGV0IHhsb25naGF0ID0gbmV3IEFycmF5KG51bV9wdCkuZmlsbCgwLjApCiAgICBsZXQgeWxvbmdoYXQgPSBuZXcgQXJyYXkobnVtX3B0KS5maWxsKDAuMCkKCgl4bG9uZ2hhdFswXSA9IHhoYXRbMF0gKiBudW1fcHQgLyBudW1fY3AKCXhsb25naGF0WzFdID0geGhhdFsxXSAqIG51bV9wdCAvIG51bV9jcAoJeGxvbmdoYXRbMl0gPSB4aGF0WzJdICogbnVtX3B0IC8gbnVtX2NwCgl4bG9uZ2hhdFs5OV0gPSB4aGF0WzNdICogbnVtX3B0IC8gbnVtX2NwCgl4bG9uZ2hhdFsxMDBdID0geGhhdFs0XSAqIG51bV9wdCAvIG51bV9jcAoKCXlsb25naGF0WzBdID0geWhhdFswXSAqIG51bV9wdCAvIG51bV9jcAoJeWxvbmdoYXRbMV0gPSB5aGF0WzFdICogbnVtX3B0IC8gbnVtX2NwCgl5bG9uZ2hhdFsyXSA9IHloYXRbMl0gKiBudW1fcHQgLyBudW1fY3AKCXlsb25naGF0Wzk5XSA9IHloYXRbM10gKiBudW1fcHQgLyBudW1fY3AKCXlsb25naGF0WzEwMF0gPSB5aGF0WzRdICogbnVtX3B0IC8gbnVtX2NwCgoJLy9sZXQgeGxvbmcgPSBuZXcgQXJyYXkoMTAxKQoJLy9sZXQgeWxvbmcgPSBuZXcgQXJyYXkoMTAxKQoJY29uc3QgW3hsb25nLCB5bG9uZ10gPSBmb3VyaWVyLmlkZnQoeGxvbmdoYXQsIHlsb25naGF0KQoJCiAgICBmb3IgKHZhciBpPTA7IGkgPCBudW1fcHQ7IGkrKykgewogICAgICBzdGFyLnBvaW50c1tpXS54ID0geGxvbmdbaV0KICAgICAgc3Rhci5wb2ludHNbaV0ueSA9IHlsb25nW2ldCiAgICB9CgkKICB9CgogIGZ1bmN0aW9uIHVwZGF0ZVBvaW50c1JhZGlhbCgpIHsKICAgIGNvbnN0IHN0YXIgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKCIjZWRpdC1zdGFyIikKCWNvbnN0IGNvbnRyb2xQb2ludHMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIucG9pbnQtaGFuZGxlIikKCWNvbnN0IG51bV9jcCA9IDUKCWNvbnN0IG51bV9wdCA9IDEwMAoJbGV0IHggPSBuZXcgQXJyYXkobnVtX2NwKQoJbGV0IHkgPSBuZXcgQXJyYXkobnVtX2NwKQoJbGV0IHIgPSBuZXcgQXJyYXkobnVtX2NwKQogICAgZm9yKHZhciBpID0gMDsgaSA8IG51bV9jcDsgaSsrKSB7CiAgICAgeFtpXSA9IGNvbnRyb2xQb2ludHNbaV0ueC5iYXNlVmFsLnZhbHVlCiAgICAgeVtpXSA9IGNvbnRyb2xQb2ludHNbaV0ueS5iYXNlVmFsLnZhbHVlCgkgcltpXSA9IE1hdGguaHlwb3QoeFtpXS0yMDAsIHlbaV0tMjAwKQoJIC8vY29uc29sZS5sb2cocltpXSkKICAgIH0KCQoKCWxldCB6ID0gbmV3IEFycmF5KG51bV9jcCkuZmlsbCgwLjApCgljb25zdCBbcmhhdCwgaWhhdF0gPSBmb3VyaWVyLmRmdChyLCB6KQoKCWxldCBybG9uZ2hhdCA9IG5ldyBBcnJheShudW1fcHQpLmZpbGwoMC4wKQoJbGV0IGlsb25naGF0ID0gbmV3IEFycmF5KG51bV9wdCkuZmlsbCgwLjApCgoJcmxvbmdoYXRbMF0gPSByaGF0WzBdICogbnVtX3B0IC8gbnVtX2NwCglybG9uZ2hhdFsxXSA9IHJoYXRbMV0gKiBudW1fcHQgLyBudW1fY3AKCXJsb25naGF0WzJdID0gcmhhdFsyXSAqIG51bV9wdCAvIG51bV9jcAoJcmxvbmdoYXRbOTldID0gcmhhdFszXSAqIG51bV9wdCAvIG51bV9jcAoJcmxvbmdoYXRbMTAwXSA9IHJoYXRbNF0gKiBudW1fcHQgLyBudW1fY3AKCglpbG9uZ2hhdFswXSA9IGloYXRbMF0gKiBudW1fcHQgLyBudW1fY3AKCWlsb25naGF0WzFdID0gaWhhdFsxXSAqIG51bV9wdCAvIG51bV9jcAoJaWxvbmdoYXRbMl0gPSBpaGF0WzJdICogbnVtX3B0IC8gbnVtX2NwCglpbG9uZ2hhdFs5OV0gPSBpaGF0WzNdICogbnVtX3B0IC8gbnVtX2NwCglpbG9uZ2hhdFsxMDBdID0gaWhhdFs0XSAqIG51bV9wdCAvIG51bV9jcAoKCWNvbnN0IFtybG9uZywgX10gPSBmb3VyaWVyLmlkZnQocmxvbmdoYXQsIGlsb25naGF0KQoJCgkvL2xldCB4bG9uZyA9IG5ldyBBcnJheShudW1fcHQpCgkvL2xldCB5bG9uZyA9IG5ldyBBcnJheShudW1fcHQpCgoJbGV0IHRoZXRhID0gLU1hdGguUEkKCglmb3IodmFyIGk9MDsgaSA8IG51bV9wdDsgaSsrKSB7CgkgICAvL2NvbnNvbGUubG9nKHJsb25nW2ldKQoJICAgc3Rhci5wb2ludHNbaV0ueCA9IHJsb25nW2ldICogTWF0aC5jb3ModGhldGEpICsgMjAwCgkgICBzdGFyLnBvaW50c1tpXS55ID0gcmxvbmdbaV0gKiBNYXRoLnNpbih0aGV0YSkgKyAyMDAKCSAgIHRoZXRhICs9IDIgKiBNYXRoLlBJIC8gKG51bV9wdCkKCX0KCglzdGFyLnBvaW50c1tudW1fcHRdLnggPSBzdGFyLnBvaW50c1swXS54CglzdGFyLnBvaW50c1tudW1fcHRdLnkgPSBzdGFyLnBvaW50c1swXS55CgoJCiAgfQoKICBpbnRlcmFjdChyb290KS5vbignbW91c2Vkb3duJywgYXBwbHlUcmFuc2Zvcm1zKS5vbigndG91Y2hzdGFydCcsIGFwcGx5VHJhbnNmb3JtcykKCiAgaW50ZXJhY3QoJy5wb2ludC1oYW5kbGUnKQogICAgLmRyYWdnYWJsZSh7CiAgICAgIG9uc3RhcnQ6IGZ1bmN0aW9uIChldmVudCkgewogICAgICAgIHJvb3Quc2V0QXR0cmlidXRlKCdjbGFzcycsICdkcmFnZ2luZycpCiAgICAgIH0sCiAgICAgIG9ubW92ZTogZnVuY3Rpb24gKGV2ZW50KSB7CiAgICAgICAgY29uc3QgaSA9IGV2ZW50LnRhcmdldC5nZXRBdHRyaWJ1dGUoJ2RhdGEtaW5kZXgnKSB8IDAKICAgICAgICBjb25zdCBwb2ludCA9IHN0YXIucG9pbnRzLmdldEl0ZW0oaSkKCgkJLy9HZXQgYW5nbGUgYW5kIHJhZGl1cwoJCWNvbnN0IHRoZXRhID0gTWF0aC5hdGFuMihwb2ludC55IC0gMjAwLCBwb2ludC54IC0gMjAwKQoKCSAgICAvL3VuY29uc3RyYWluZWQKICAgICAgICBwb2ludC54ICs9IGV2ZW50LmR4IC8gcm9vdE1hdHJpeC5hCiAgICAgICAgcG9pbnQueSArPSBldmVudC5keSAvIHJvb3RNYXRyaXguZAoKCSAgICAvL3JhZGlhbGx5IGNvbnN0cmFpbmVkCgkJY29uc3QgcmFkaWFsID0gTWF0aC5oeXBvdChwb2ludC55IC0gMjAwLCBwb2ludC54IC0gMjAwKQkgICAgCgkgICAgcG9pbnQueCA9IHJhZGlhbCAqIE1hdGguY29zKHRoZXRhKSArIDIwMAoJICAgIHBvaW50LnkgPSByYWRpYWwgKiBNYXRoLnNpbih0aGV0YSkgKyAyMDAKCiAgICAgICAgZXZlbnQudGFyZ2V0LnguYmFzZVZhbC52YWx1ZSA9IHBvaW50LngKICAgICAgICBldmVudC50YXJnZXQueS5iYXNlVmFsLnZhbHVlID0gcG9pbnQueQoKCQkvL3VwZGF0ZVBvaW50c1JhZGlhbCgpCgkgICAgLy91cGRhdGVQb2ludHMoKQogICAgICB9LAogICAgICBvbmVuZDogZnVuY3Rpb24gKGV2ZW50KSB7CiAgICAgICAgcm9vdC5zZXRBdHRyaWJ1dGUoJ2NsYXNzJywgJycpCgkgICAgdXBkYXRlUG9pbnRzUmFkaWFsKCkKICAgICAgfSwKICAgICAgc25hcDogewogICAgICAgIHRhcmdldHM6IG9yaWdpbmFsUG9pbnRzLAogICAgICAgIHJhbmdlOiAxMCwKICAgICAgICByZWxhdGl2ZVBvaW50czogW3sgeDogMC41LCB5OiAwLjUgfV0sCiAgICAgIH0sCiAgICAgIHJlc3RyaWN0OiB7IHJlc3RyaWN0aW9uOiBkb2N1bWVudC5yb290RWxlbWVudCB9LAogICAgfSkKICAgIC5zdHlsZUN1cnNvcihmYWxzZSkKCiAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcignZHJhZ3N0YXJ0JywgKGV2ZW50KSA9PiB7CiAgICBldmVudC5wcmV2ZW50RGVmYXVsdCgpCiAgfSkKPC9zY3JpcHQ+Cgk8L2JvbmQ+pG1pbWWpdGV4dC9odG1srHJvb3Rhc3NpZ25lZcCybGFzdF9ydW5fdGltZXN0YW1wy0HY4/83jwo9sHBlcnNpc3RfanNfc3RhdGXCt2hhc19wbHV0b19ob29rX2ZlYXR1cmVzwqdjZWxsX2lk2SRjODE1ZThlMS1mZmIxLTRiOTMtOGYzZS1hOGZkMGMxNDk1OTe5ZGVwZW5kc19vbl9kaXNhYmxlZF9jZWxsc8KncnVudGltZc4+nUaItXB1Ymxpc2hlZF9vYmplY3Rfa2V5c5C4ZGVwZW5kc19vbl9za2lwcGVkX2NlbGxzwqdlcnJvcmVkwrFjZWxsX2RlcGVuZGVuY2llc4LZJDM2MjcxNGIwLTc1ZDgtMTFlZC0yMmMwLTQ5Y2NlYjE2MWJiNYS0cHJlY2VkZW5jZV9oZXVyaXN0aWMHp2NlbGxfaWTZJDM2MjcxNGIwLTc1ZDgtMTFlZC0yMmMwLTQ5Y2NlYjE2MWJiNbRkb3duc3RyZWFtX2NlbGxzX21hcIOwSHlwZXJ0ZXh0TGl0ZXJhbJHZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N6LOuJHZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N6RGRlRXkLJ1cHN0cmVhbV9jZWxsc19tYXCDoS2QpXJhbmdlkKLPgJDZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N4S0cHJlY2VkZW5jZV9oZXVyaXN0aWMJp2NlbGxfaWTZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N7Rkb3duc3RyZWFtX2NlbGxzX21hcIWjYnVmkKNzdHKQpW15c3ZnkKVzdmdfeJClc3ZnX3mQsnVwc3RyZWFtX2NlbGxzX21hcN4AF6hJT0J1ZmZlcpCmU3RyaW5nkKN6aXCQpEBodGyQpE1haW6Q2SBIeXBlcnRleHRMaXRlcmFsLmF0dHJpYnV0ZV92YWx1ZZC3SHlwZXJ0ZXh0TGl0ZXJhbC5SZXN1bHSQ2SBNYWluLlBsdXRvUnVubmVyLkNvcmUuYXBwbGljYWJsZZChOpC3SHlwZXJ0ZXh0TGl0ZXJhbC5CeXBhc3OQuU1haW4uUGx1dG9SdW5uZXIuQmFzZS5nZXSQpUBiaW5kkNkoTWFpbi5QbHV0b1J1bm5lci5QbHV0b1J1bm5lci5jcmVhdGVfYm9uZJCjZW5kkKV0YWtlIZChLZClcHJpbnSQsEh5cGVydGV4dExpdGVyYWyR2SQzNjI3MTRiMC03NWQ4LTExZWQtMjJjMC00OWNjZWIxNjFiYjWhK5ChKpCjY29zkKLOuJHZJDM2MjcxNGIwLTc1ZDgtMTFlZC0yMmMwLTQ5Y2NlYjE2MWJiNaNzaW6QtGNlbGxfZXhlY3V0aW9uX29yZGVyktkkMzYyNzE0YjAtNzVkOC0xMWVkLTIyYzAtNDljY2ViMTYxYmI12SRjODE1ZThlMS1mZmIxLTRiOTMtOGYzZS1hOGZkMGMxNDk1OTe0bGFzdF9ob3RfcmVsb2FkX3RpbWXLAAAAAAAAAACpc2hvcnRwYXRouEludGVyZXN0aW5nIGJsdWVwcmludC5qbK5wcm9jZXNzX3N0YXR1c6VyZWFkeaRwYXRo2UZDOlxVc2Vyc1xraXR0aXNvcGlrdWxtXC5qdWxpYVxwbHV0b19ub3RlYm9va3NcSW50ZXJlc3RpbmcgYmx1ZXByaW50Lmpsrmxhc3Rfc2F2ZV90aW1ly0HY4/9DOl41qmNlbGxfb3JkZXKS2SQzNjI3MTRiMC03NWQ4LTExZWQtMjJjMC00OWNjZWIxNjFiYjXZJGM4MTVlOGUxLWZmYjEtNGI5My04ZjNlLWE4ZmQwYzE0OTU5N7FwdWJsaXNoZWRfb2JqZWN0c4ClbmJwa2eIsmluc3RhbGxlZF92ZXJzaW9uc4KwSHlwZXJ0ZXh0TGl0ZXJhbKUwLjkuNKRGRlRXpTEuNS4wsHRlcm1pbmFsX291dHB1dHOCsEh5cGVydGV4dExpdGVyYWzZJxtbMzJtG1sxbUxvYWRpbmcbWzIybRtbMzltIHBhY2thZ2VzLi4uCqRGRlRX2dkbWzMybRtbMW1Mb2FkaW5nG1syMm0bWzM5bSBwYWNrYWdlcy4uLgobWzMybRtbMW1QcmVjb21waWxpbmcbWzIybRtbMzltIHByb2plY3QuLi4KG1szMm0gIOKckyAbWzM5bRtbOTBtTUtMX2psbBtbMzltChtbMzJtICDinJMgG1szOW1GRlRXCiAgMiBkZXBlbmRlbmNpZXMgc3VjY2Vzc2Z1bGx5IHByZWNvbXBpbGVkIGluIDUgc2Vjb25kcy4gMTQgYWxyZWFkeSBwcmVjb21waWxlZC4Kp2VuYWJsZWTDt3Jlc3RhcnRfcmVjb21tZW5kZWRfbXNnwLRyZXN0YXJ0X3JlcXVpcmVkX21zZ8CtYnVzeV9wYWNrYWdlc5CvaW5zdGFsbF90aW1lX25zzwAAAAJ/rwHcrGluc3RhbnRpYXRlZMOrY2VsbF9pbnB1dHOC2SQzNjI3MTRiMC03NWQ4LTExZWQtMjJjMC00OWNjZWIxNjFiYjWEp2NlbGxfaWTZJDM2MjcxNGIwLTc1ZDgtMTFlZC0yMmMwLTQ5Y2NlYjE2MWJiNaRjb2Rl2VNiZWdpbgoJdXNpbmcgRkZUVwoJdXNpbmcgSHlwZXJ0ZXh0TGl0ZXJhbAoJzrggPSByYW5nZSgtz4AsIHN0b3A9z4AsIGxlbmd0aD0xMDEpCmVuZKhtZXRhZGF0YYOpc2hvd19sb2dzw6hkaXNhYmxlZMKuc2tpcF9hc19zY3JpcHTCq2NvZGVfZm9sZGVkwtkkYzgxNWU4ZTEtZmZiMS00YjkzLThmM2UtYThmZDBjMTQ5NTk3hKdjZWxsX2lk2SRjODE1ZThlMS1mZmIxLTRiOTMtOGYzZS1hOGZkMGMxNDk1OTekY29kZdoYW2JlZ2luCgkjIGZsYXR0ZW5lZAoJI3N2Z194ID0gMTAwKmNvcy4ozrgpIC4qIFJfY2lyY2xlX2Nvcy4ozrgpIC4rIDIwMAoJI3N2Z195ID0gMTAwKnNpbi4ozrgpIC4qIFJfY2lyY2xlX2Nvcy4ozrgpIC4rIDIwMAoKCSMgY2lyY2xlCglzdmdfeCA9IDEwMCpjb3MuKM64KSAuKyAyMDAKCXN2Z195ID0gMTAwKnNpbi4ozrgpIC4rIDIwMAoJYnVmID0gSU9CdWZmZXIoKQoJZm9yICh4LCB5KSBpbiB6aXAoc3ZnX3gsIHN2Z195KQoJCXByaW50KGJ1ZiwgeCAsIiwgIiwgeSwgIiwiKQoJZW5kCglzdHIgPSBTdHJpbmcodGFrZSEoYnVmKVsxOmVuZC0xXSkKCUBiaW5kIG15c3ZnIEBodGwoIiIiCgk8c3ZnIGlkPSJzdmctZWRpdC1kZW1vIiB2aWV3Qm94PSIwIDAgNDAwIDQwMCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmFsdWU9IjMiPgoJCgk8ZGVmcz4KCSAgICA8Y2lyY2xlIGlkPSJwb2ludC1oYW5kbGUiCgkgICAgICAgIHI9IjEwIiB4PSIwIiB5PSIwIgoJICAgICAgICBzdHJva2Utd2lkdGg9IjQiCgkgICAgICAgIGZpbGw9IiNmZmYiCgkgICAgICAgIGZpbGwtb3BhY2l0eT0iMC40IgoJICAgICAgICBzdHJva2U9IiNmZmYiLz4KCTwvZGVmcz4KCTxjaXJjbGUgY3g9IjIwMCIgY3k9IjIwMCIgcj0iMTAiIC8+Cgk8cG9seWdvbiBpZD0iZWRpdC1zdGFyIgoJICAgIHN0cm9rZT0iIzI5ZSIKCSAgICBzdHJva2Utd2lkdGg9IjIwIgoJICAgIHN0cm9rZS1saW5lam9pbj0icm91bmQiCgkgICAgZmlsbD0ibm9uZSIKCSAgICBwb2ludHM9IiQoc3RyKSIvPgoJPC9zdmc+Cgk8c2NyaXB0IHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9pbnRlcmFjdGpzQDEuMTAuMTcvZGlzdC9pbnRlcmFjdC5taW4uanMiPjwvc2NyaXB0Pgo8c2NyaXB0IHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9mb3VyaWVyL2ZvdXJpZXIubWluLmpzIj48L3NjcmlwdD4KCTxzY3JpcHQ+Cgljb25zb2xlLmxvZygiU1ZHIFNjcmlwdCIpCiAgY29uc3Qgc25zID0gJ2h0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnJwogIGNvbnN0IHhucyA9ICdodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rJwogIGNvbnN0IHJvb3QgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnc3ZnLWVkaXQtZGVtbycpCiAgY29uc3Qgc3RhciA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdlZGl0LXN0YXInKQogIGxldCByb290TWF0cml4CiAgY29uc3Qgb3JpZ2luYWxQb2ludHMgPSBbXQogIGxldCB0cmFuc2Zvcm1lZFBvaW50cyA9IFtdCgogIGZvciAobGV0IGkgPSAwLCBsZW4gPSBzdGFyLnBvaW50cy5udW1iZXJPZkl0ZW1zOyBpIDwgbGVuLTIwOyBpICs9IDIwKSB7CiAgICBjb25zdCBoYW5kbGUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc25zLCAndXNlJykKICAgIGNvbnN0IHBvaW50ID0gc3Rhci5wb2ludHMuZ2V0SXRlbShpKQogICAgY29uc3QgbmV3UG9pbnQgPSByb290LmNyZWF0ZVNWR1BvaW50KCkKCiAgICBoYW5kbGUuc2V0QXR0cmlidXRlTlMoeG5zLCAnaHJlZicsICcjcG9pbnQtaGFuZGxlJykKICAgIGhhbmRsZS5zZXRBdHRyaWJ1dGUoJ2NsYXNzJywgJ3BvaW50LWhhbmRsZScpCgogICAgaGFuZGxlLnguYmFzZVZhbC52YWx1ZSA9IG5ld1BvaW50LnggPSBwb2ludC54CiAgICBoYW5kbGUueS5iYXNlVmFsLnZhbHVlID0gbmV3UG9pbnQueSA9IHBvaW50LnkKCiAgICBoYW5kbGUuc2V0QXR0cmlidXRlKCdkYXRhLWluZGV4JywgaSkKCiAgICBvcmlnaW5hbFBvaW50cy5wdXNoKG5ld1BvaW50KQoKICAgIHJvb3QuYXBwZW5kQ2hpbGQoaGFuZGxlKQogIH0KCiAgZnVuY3Rpb24gYXBwbHlUcmFuc2Zvcm1zIChldmVudCkgewogICAgcm9vdE1hdHJpeCA9IHJvb3QuZ2V0U2NyZWVuQ1RNKCkKCiAgICB0cmFuc2Zvcm1lZFBvaW50cyA9IG9yaWdpbmFsUG9pbnRzLm1hcCgocG9pbnQpID0+IHsKICAgICAgcmV0dXJuIHBvaW50Lm1hdHJpeFRyYW5zZm9ybShyb290TWF0cml4KQogICAgfSkKCiAgICBpbnRlcmFjdCgnLnBvaW50LWhhbmRsZScpLmRyYWdnYWJsZSh7CiAgICAgIHNuYXA6IHsKICAgICAgICB0YXJnZXRzOiB0cmFuc2Zvcm1lZFBvaW50cywKICAgICAgICByYW5nZTogMjAgKiBNYXRoLm1heChyb290TWF0cml4LmEsIHJvb3RNYXRyaXguZCksCiAgICAgIH0sCiAgICB9KQogIH0KCgogIGZ1bmN0aW9uIHVwZGF0ZVBvaW50cygpIHsKICAgIGNvbnN0IHN0YXIgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKCIjZWRpdC1zdGFyIikKCWNvbnN0IGNvbnRyb2xQb2ludHMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIucG9pbnQtaGFuZGxlIikKCWNvbnN0IG51bV9jcCA9IDUKCWNvbnN0IG51bV9wdCA9IDEwMQoJbGV0IHggPSBuZXcgQXJyYXkobnVtX2NwKQoJbGV0IHkgPSBuZXcgQXJyYXkobnVtX2NwKQogICAgZm9yKHZhciBpID0gMDsgaSA8IG51bV9jcDsgaSsrKSB7CiAgICAgeFtpXSA9IGNvbnRyb2xQb2ludHNbaV0ueC5iYXNlVmFsLnZhbHVlCiAgICAgeVtpXSA9IGNvbnRyb2xQb2ludHNbaV0ueS5iYXNlVmFsLnZhbHVlCiAgICB9CgoJLy9sZXQgeGhhdCA9IG5ldyBBcnJheSgxMDEpCgkvL2xldCB5aGF0ID0gbmV3IEFycmF5KDEwMSkKCWNvbnN0IFt4aGF0LCB5aGF0XSA9IGZvdXJpZXIuZGZ0KHggLHkpCgkKICAgIGxldCB4bG9uZ2hhdCA9IG5ldyBBcnJheShudW1fcHQpLmZpbGwoMC4wKQogICAgbGV0IHlsb25naGF0ID0gbmV3IEFycmF5KG51bV9wdCkuZmlsbCgwLjApCgoJeGxvbmdoYXRbMF0gPSB4aGF0WzBdICogbnVtX3B0IC8gbnVtX2NwCgl4bG9uZ2hhdFsxXSA9IHhoYXRbMV0gKiBudW1fcHQgLyBudW1fY3AKCXhsb25naGF0WzJdID0geGhhdFsyXSAqIG51bV9wdCAvIG51bV9jcAoJeGxvbmdoYXRbOTldID0geGhhdFszXSAqIG51bV9wdCAvIG51bV9jcAoJeGxvbmdoYXRbMTAwXSA9IHhoYXRbNF0gKiBudW1fcHQgLyBudW1fY3AKCgl5bG9uZ2hhdFswXSA9IHloYXRbMF0gKiBudW1fcHQgLyBudW1fY3AKCXlsb25naGF0WzFdID0geWhhdFsxXSAqIG51bV9wdCAvIG51bV9jcAoJeWxvbmdoYXRbMl0gPSB5aGF0WzJdICogbnVtX3B0IC8gbnVtX2NwCgl5bG9uZ2hhdFs5OV0gPSB5aGF0WzNdICogbnVtX3B0IC8gbnVtX2NwCgl5bG9uZ2hhdFsxMDBdID0geWhhdFs0XSAqIG51bV9wdCAvIG51bV9jcAoKCS8vbGV0IHhsb25nID0gbmV3IEFycmF5KDEwMSkKCS8vbGV0IHlsb25nID0gbmV3IEFycmF5KDEwMSkKCWNvbnN0IFt4bG9uZywgeWxvbmddID0gZm91cmllci5pZGZ0KHhsb25naGF0LCB5bG9uZ2hhdCkKCQogICAgZm9yICh2YXIgaT0wOyBpIDwgbnVtX3B0OyBpKyspIHsKICAgICAgc3Rhci5wb2ludHNbaV0ueCA9IHhsb25nW2ldCiAgICAgIHN0YXIucG9pbnRzW2ldLnkgPSB5bG9uZ1tpXQogICAgfQoJCiAgfQoKICBmdW5jdGlvbiB1cGRhdGVQb2ludHNSYWRpYWwoKSB7CiAgICBjb25zdCBzdGFyID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvcigiI2VkaXQtc3RhciIpCgljb25zdCBjb250cm9sUG9pbnRzID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgiLnBvaW50LWhhbmRsZSIpCgljb25zdCBudW1fY3AgPSA1Cgljb25zdCBudW1fcHQgPSAxMDAKCWxldCB4ID0gbmV3IEFycmF5KG51bV9jcCkKCWxldCB5ID0gbmV3IEFycmF5KG51bV9jcCkKCWxldCByID0gbmV3IEFycmF5KG51bV9jcCkKICAgIGZvcih2YXIgaSA9IDA7IGkgPCBudW1fY3A7IGkrKykgewogICAgIHhbaV0gPSBjb250cm9sUG9pbnRzW2ldLnguYmFzZVZhbC52YWx1ZQogICAgIHlbaV0gPSBjb250cm9sUG9pbnRzW2ldLnkuYmFzZVZhbC52YWx1ZQoJIHJbaV0gPSBNYXRoLmh5cG90KHhbaV0tMjAwLCB5W2ldLTIwMCkKCSAvL2NvbnNvbGUubG9nKHJbaV0pCiAgICB9CgkKCglsZXQgeiA9IG5ldyBBcnJheShudW1fY3ApLmZpbGwoMC4wKQoJY29uc3QgW3JoYXQsIGloYXRdID0gZm91cmllci5kZnQociwgeikKCglsZXQgcmxvbmdoYXQgPSBuZXcgQXJyYXkobnVtX3B0KS5maWxsKDAuMCkKCWxldCBpbG9uZ2hhdCA9IG5ldyBBcnJheShudW1fcHQpLmZpbGwoMC4wKQoKCXJsb25naGF0WzBdID0gcmhhdFswXSAqIG51bV9wdCAvIG51bV9jcAoJcmxvbmdoYXRbMV0gPSByaGF0WzFdICogbnVtX3B0IC8gbnVtX2NwCglybG9uZ2hhdFsyXSA9IHJoYXRbMl0gKiBudW1fcHQgLyBudW1fY3AKCXJsb25naGF0Wzk5XSA9IHJoYXRbM10gKiBudW1fcHQgLyBudW1fY3AKCXJsb25naGF0WzEwMF0gPSByaGF0WzRdICogbnVtX3B0IC8gbnVtX2NwCgoJaWxvbmdoYXRbMF0gPSBpaGF0WzBdICogbnVtX3B0IC8gbnVtX2NwCglpbG9uZ2hhdFsxXSA9IGloYXRbMV0gKiBudW1fcHQgLyBudW1fY3AKCWlsb25naGF0WzJdID0gaWhhdFsyXSAqIG51bV9wdCAvIG51bV9jcAoJaWxvbmdoYXRbOTldID0gaWhhdFszXSAqIG51bV9wdCAvIG51bV9jcAoJaWxvbmdoYXRbMTAwXSA9IGloYXRbNF0gKiBudW1fcHQgLyBudW1fY3AKCgljb25zdCBbcmxvbmcsIF9dID0gZm91cmllci5pZGZ0KHJsb25naGF0LCBpbG9uZ2hhdCkKCQoJLy9sZXQgeGxvbmcgPSBuZXcgQXJyYXkobnVtX3B0KQoJLy9sZXQgeWxvbmcgPSBuZXcgQXJyYXkobnVtX3B0KQoKCWxldCB0aGV0YSA9IC1NYXRoLlBJCgoJZm9yKHZhciBpPTA7IGkgPCBudW1fcHQ7IGkrKykgewoJICAgLy9jb25zb2xlLmxvZyhybG9uZ1tpXSkKCSAgIHN0YXIucG9pbnRzW2ldLnggPSBybG9uZ1tpXSAqIE1hdGguY29zKHRoZXRhKSArIDIwMAoJICAgc3Rhci5wb2ludHNbaV0ueSA9IHJsb25nW2ldICogTWF0aC5zaW4odGhldGEpICsgMjAwCgkgICB0aGV0YSArPSAyICogTWF0aC5QSSAvIChudW1fcHQpCgl9CgoJc3Rhci5wb2ludHNbbnVtX3B0XS54ID0gc3Rhci5wb2ludHNbMF0ueAoJc3Rhci5wb2ludHNbbnVtX3B0XS55ID0gc3Rhci5wb2ludHNbMF0ueQoKCQogIH0KCiAgaW50ZXJhY3Qocm9vdCkub24oJ21vdXNlZG93bicsIGFwcGx5VHJhbnNmb3Jtcykub24oJ3RvdWNoc3RhcnQnLCBhcHBseVRyYW5zZm9ybXMpCgogIGludGVyYWN0KCcucG9pbnQtaGFuZGxlJykKICAgIC5kcmFnZ2FibGUoewogICAgICBvbnN0YXJ0OiBmdW5jdGlvbiAoZXZlbnQpIHsKICAgICAgICByb290LnNldEF0dHJpYnV0ZSgnY2xhc3MnLCAnZHJhZ2dpbmcnKQogICAgICB9LAogICAgICBvbm1vdmU6IGZ1bmN0aW9uIChldmVudCkgewogICAgICAgIGNvbnN0IGkgPSBldmVudC50YXJnZXQuZ2V0QXR0cmlidXRlKCdkYXRhLWluZGV4JykgfCAwCiAgICAgICAgY29uc3QgcG9pbnQgPSBzdGFyLnBvaW50cy5nZXRJdGVtKGkpCgoJCS8vR2V0IGFuZ2xlIGFuZCByYWRpdXMKCQljb25zdCB0aGV0YSA9IE1hdGguYXRhbjIocG9pbnQueSAtIDIwMCwgcG9pbnQueCAtIDIwMCkKCgkgICAgLy91bmNvbnN0cmFpbmVkCiAgICAgICAgcG9pbnQueCArPSBldmVudC5keCAvIHJvb3RNYXRyaXguYQogICAgICAgIHBvaW50LnkgKz0gZXZlbnQuZHkgLyByb290TWF0cml4LmQKCgkgICAgLy9yYWRpYWxseSBjb25zdHJhaW5lZAoJCWNvbnN0IHJhZGlhbCA9IE1hdGguaHlwb3QocG9pbnQueSAtIDIwMCwgcG9pbnQueCAtIDIwMCkJICAgIAoJICAgIHBvaW50LnggPSByYWRpYWwgKiBNYXRoLmNvcyh0aGV0YSkgKyAyMDAKCSAgICBwb2ludC55ID0gcmFkaWFsICogTWF0aC5zaW4odGhldGEpICsgMjAwCgogICAgICAgIGV2ZW50LnRhcmdldC54LmJhc2VWYWwudmFsdWUgPSBwb2ludC54CiAgICAgICAgZXZlbnQudGFyZ2V0LnkuYmFzZVZhbC52YWx1ZSA9IHBvaW50LnkKCgkJLy91cGRhdGVQb2ludHNSYWRpYWwoKQoJICAgIC8vdXBkYXRlUG9pbnRzKCkKICAgICAgfSwKICAgICAgb25lbmQ6IGZ1bmN0aW9uIChldmVudCkgewogICAgICAgIHJvb3Quc2V0QXR0cmlidXRlKCdjbGFzcycsICcnKQoJICAgIHVwZGF0ZVBvaW50c1JhZGlhbCgpCiAgICAgIH0sCiAgICAgIHNuYXA6IHsKICAgICAgICB0YXJnZXRzOiBvcmlnaW5hbFBvaW50cywKICAgICAgICByYW5nZTogMTAsCiAgICAgICAgcmVsYXRpdmVQb2ludHM6IFt7IHg6IDAuNSwgeTogMC41IH1dLAogICAgICB9LAogICAgICByZXN0cmljdDogeyByZXN0cmljdGlvbjogZG9jdW1lbnQucm9vdEVsZW1lbnQgfSwKICAgIH0pCiAgICAuc3R5bGVDdXJzb3IoZmFsc2UpCgogIGRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoJ2RyYWdzdGFydCcsIChldmVudCkgPT4gewogICAgZXZlbnQucHJldmVudERlZmF1bHQoKQogIH0pCjwvc2NyaXB0PgoJIiIiKQplbmSobWV0YWRhdGGDqXNob3dfbG9nc8OoZGlzYWJsZWTCrnNraXBfYXNfc2NyaXB0wqtjb2RlX2ZvbGRlZMKrbm90ZWJvb2tfaWTZJDM2MjcxNGIwLTc1ZDgtMTFlZC0xNDk0LWE3ODVhYmU0MzEzY6tpbl90ZW1wX2RpcsOobWV0YWRhdGGA";
window.pluto_preamble_html = undefined;
</script>
<meta name="pluto-insertion-spot-parameters">
<script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.505b982d.js" type="module" defer="" integrity="sha384-RAKZ8kuxvc3ilICBZgvEJ57Z4OfXYz37Eoglh4v+smYrAwpEH6pkrwdz+rKsXJLg" crossorigin="anonymous"></script><script src="https://cdn.jsdelivr.net/gh/fonsp/Pluto.jl@0.19.16/frontend-dist/editor.b9f0ac7b.js" integrity="sha384-bgokkaJz2F/u6BCZYRcl36Jkfwy7x9SeobwBF3EWarMEaCfg5aSU3Z7nSuLQdETn" crossorigin="anonymous"></script><script type="text/javascript" id="MathJax-script" integrity="sha384-4kE/rQ11E8xT9QgrCBTyvenkuPfQo8rXYQvJZuMgxyPOoUfpatjQPlgdv6V5yhUK" crossorigin="" not-the-src-yet="https://cdn.jsdelivr.net/npm/mathjax@3.2.2/es5/tex-svg-full.js" async=""></script></head><body class="loading no-MαθJax"> <div style="display:flex;min-height:100vh;"> <pluto-editor class="fullscreen"></pluto-editor> </div> </body></html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment