lenslet 0.2.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- lenslet/__init__.py +7 -0
- lenslet/api.py +157 -0
- lenslet/cli.py +121 -0
- lenslet/frontend/assets/index-B-0lZ7yu.js +44 -0
- lenslet/frontend/assets/index-c56aKxHZ.css +1 -0
- lenslet/frontend/favicon.ico +0 -0
- lenslet/frontend/index.html +14 -0
- lenslet/metadata.py +151 -0
- lenslet/server.py +520 -0
- lenslet/storage/__init__.py +6 -0
- lenslet/storage/base.py +35 -0
- lenslet/storage/dataset.py +591 -0
- lenslet/storage/local.py +69 -0
- lenslet/storage/memory.py +472 -0
- lenslet/storage/parquet.py +483 -0
- lenslet/workspace.py +60 -0
- lenslet-0.2.1.dist-info/METADATA +134 -0
- lenslet-0.2.1.dist-info/RECORD +20 -0
- lenslet-0.2.1.dist-info/WHEEL +4 -0
- lenslet-0.2.1.dist-info/entry_points.txt +2 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
/*! tailwindcss v4.1.18 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:system-ui,-apple-system,Segoe UI,Roboto,sans-serif;--font-mono:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;--color-black:#000;--color-white:#fff;--spacing:.25rem;--text-xs:12px;--text-xs--line-height:16px;--text-sm:13px;--text-sm--line-height:18px;--text-lg:1.125rem;--text-lg--line-height:calc(1.75/1.125);--font-weight-medium:500;--tracking-wide:.025em;--radius-md:.375rem;--radius-lg:10px;--radius-xl:.75rem;--shadow-lg:0 10px 15px -3px #0000001a,0 4px 6px -4px #0000001a;--ease-out:cubic-bezier(0,0,.2,1);--blur-sm:8px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4,0,.2,1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--radius:8px;--color-panel:var(--panel);--color-hover:var(--hover);--color-text:var(--text);--color-muted:var(--muted);--color-accent:var(--accent);--color-accent-muted:var(--accent-muted);--color-accent-strong:var(--accent-strong);--color-border:var(--border);--color-surface:var(--surface);--color-surface-hover:var(--surface-hover);--color-surface-overlay:var(--surface-overlay);--color-surface-inset:var(--surface-inset);--color-star-active:var(--star-active);--color-star-inactive:var(--star-inactive);--color-star-hover:var(--star-hover);--color-danger:var(--danger);--z-toolbar:50;--z-menu:1000}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}:root{--bg:#0f0f10;--panel:#18181b;--hover:#27272a;--border:#27272a;--thumb-bg:transparent;--surface:#1b1b1b;--surface-hover:#252525;--surface-active:#2f2f2f;--surface-overlay:#141414;--surface-inset:#121212;--text:#e4e4e7;--text-secondary:#cfd1d4;--muted:#71717a;--accent:#3b82f6;--accent-hover:#4a8fff;--accent-muted:#3b82f626;--accent-strong:#3b82f64d;--star-active:#ffd166;--star-inactive:#6f6f6f;--star-hover:#e7c46a;--danger:#ff6b6b;--success:#9ad4b5;--border-subtle:#ffffff0d;--border-strong:#3a3a3a;--border-hover:#4a4a4a;--shadow-sm:0 1px 2px #0000004d;--shadow-md:0 6px 16px #00000059;--shadow-lg:0 10px 26px #0006;--shadow-xl:0 10px 30px #00000073}*{box-sizing:border-box}html,body,#root{background:var(--bg);height:100%;color:var(--text)}body{-webkit-font-smoothing:antialiased;margin:0;font:13px/1.4 -apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica,Arial,sans-serif;overflow:hidden}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.visible{visibility:visible}.sr-only{clip-path:inset(50%);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.sticky{position:sticky}.inset-0{inset:calc(var(--spacing)*0)}.inset-y-0{inset-block:calc(var(--spacing)*0)}.top-0{top:calc(var(--spacing)*0)}.top-1\.5{top:calc(var(--spacing)*1.5)}.top-3{top:calc(var(--spacing)*3)}.top-12{top:calc(var(--spacing)*12)}.top-\[48px\]{top:48px}.right-0{right:calc(var(--spacing)*0)}.right-3{right:calc(var(--spacing)*3)}.right-\[calc\(var\(--right\)-3px\)\]{right:calc(var(--right) - 3px)}.right-\[var\(--right\)\]{right:var(--right)}.bottom-0{bottom:calc(var(--spacing)*0)}.left-0{left:calc(var(--spacing)*0)}.left-1\.5{left:calc(var(--spacing)*1.5)}.left-\[calc\(var\(--left\)-3px\)\]{left:calc(var(--left) - 3px)}.left-\[var\(--left\)\]{left:var(--left)}.z-10{z-index:10}.z-\[999\]{z-index:999}.z-\[var\(--z-toolbar\)\]{z-index:var(--z-toolbar)}.col-span-full{grid-column:1/-1}.col-start-1{grid-column-start:1}.col-start-2{grid-column-start:2}.col-start-3{grid-column-start:3}.row-start-1{grid-row-start:1}.row-start-2{grid-row-start:2}.container{width:100%}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.mx-1\.5{margin-inline:calc(var(--spacing)*1.5)}.mt-1{margin-top:calc(var(--spacing)*1)}.mt-2{margin-top:calc(var(--spacing)*2)}.mt-3{margin-top:calc(var(--spacing)*3)}.mr-1{margin-right:calc(var(--spacing)*1)}.mb-1{margin-bottom:calc(var(--spacing)*1)}.mb-1\.5{margin-bottom:calc(var(--spacing)*1.5)}.mb-2{margin-bottom:calc(var(--spacing)*2)}.mb-3{margin-bottom:calc(var(--spacing)*3)}.ml-2{margin-left:calc(var(--spacing)*2)}.ml-10{margin-left:calc(var(--spacing)*10)}.ml-auto{margin-left:auto}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.contents{display:contents}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.aspect-\[4\/3\]{aspect-ratio:4/3}.h-1\.5{height:calc(var(--spacing)*1.5)}.h-5{height:calc(var(--spacing)*5)}.h-6{height:calc(var(--spacing)*6)}.h-7{height:calc(var(--spacing)*7)}.h-8{height:calc(var(--spacing)*8)}.h-9{height:calc(var(--spacing)*9)}.h-12{height:calc(var(--spacing)*12)}.h-24{height:calc(var(--spacing)*24)}.h-28{height:calc(var(--spacing)*28)}.h-48{height:calc(var(--spacing)*48)}.h-\[18px\]{height:18px}.h-\[160px\]{height:160px}.h-full{height:100%}.max-h-\[80vh\]{max-height:80vh}.max-h-none{max-height:none}.min-h-0{min-height:calc(var(--spacing)*0)}.min-h-\[28px\]{min-height:28px}.min-h-\[32px\]{min-height:32px}.w-1\.5{width:calc(var(--spacing)*1.5)}.w-4{width:calc(var(--spacing)*4)}.w-6{width:calc(var(--spacing)*6)}.w-7{width:calc(var(--spacing)*7)}.w-10{width:calc(var(--spacing)*10)}.w-16{width:calc(var(--spacing)*16)}.w-20{width:calc(var(--spacing)*20)}.w-24{width:calc(var(--spacing)*24)}.w-32{width:calc(var(--spacing)*32)}.w-\[18px\]{width:18px}.w-\[200px\]{width:200px}.w-\[220px\]{width:220px}.w-\[240px\]{width:240px}.w-fit{width:fit-content}.w-full{width:100%}.w-px{width:1px}.max-w-\[70\%\]{max-width:70%}.max-w-\[80vw\]{max-width:80vw}.max-w-none{max-width:none}.min-w-0{min-width:calc(var(--spacing)*0)}.min-w-\[24px\]{min-width:24px}.min-w-\[32px\]{min-width:32px}.min-w-\[42px\]{min-width:42px}.min-w-\[48px\]{min-width:48px}.min-w-\[78px\]{min-width:78px}.min-w-\[80px\]{min-width:80px}.min-w-\[110px\]{min-width:110px}.flex-1{flex:1}.shrink-0{flex-shrink:0}.rotate-180{rotate:180deg}.transform{transform:var(--tw-rotate-x,)var(--tw-rotate-y,)var(--tw-rotate-z,)var(--tw-skew-x,)var(--tw-skew-y,)}.cursor-col-resize{cursor:col-resize}.cursor-crosshair{cursor:crosshair}.cursor-grab{cursor:grab}.cursor-grabbing{cursor:grabbing}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.cursor-zoom-in{cursor:zoom-in}.resize{resize:both}.resize-y{resize:vertical}.appearance-none{-webkit-appearance:none;-moz-appearance:none;appearance:none}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-\[auto_1fr_auto\]{grid-template-columns:auto 1fr auto}.grid-cols-\[var\(--left\)_1fr_var\(--right\)\]{grid-template-columns:var(--left)1fr var(--right)}.grid-rows-\[48px_1fr\]{grid-template-rows:48px 1fr}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.justify-start{justify-content:flex-start}.gap-0{gap:calc(var(--spacing)*0)}.gap-1{gap:calc(var(--spacing)*1)}.gap-1\.5{gap:calc(var(--spacing)*1.5)}.gap-2{gap:calc(var(--spacing)*2)}.gap-3{gap:calc(var(--spacing)*3)}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*1)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*1)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-1\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*1.5)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*1.5)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*2)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*2)*calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing)*4)*var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing)*4)*calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.rounded{border-radius:var(--radius)}.rounded-\[10px\]{border-radius:10px}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-xl{border-radius:var(--radius-xl)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-accent{border-color:var(--color-accent)}.border-border,.border-border\/60{border-color:var(--color-border)}@supports (color:color-mix(in lab,red,red)){.border-border\/60{border-color:color-mix(in oklab,var(--color-border)60%,transparent)}}.border-border\/70{border-color:var(--color-border)}@supports (color:color-mix(in lab,red,red)){.border-border\/70{border-color:color-mix(in oklab,var(--color-border)70%,transparent)}}.border-white\/5{border-color:#ffffff0d}@supports (color:color-mix(in lab,red,red)){.border-white\/5{border-color:color-mix(in oklab,var(--color-white)5%,transparent)}}.bg-\[var\(--thumb-bg\,\#121212\)\]{background-color:var(--thumb-bg,#121212)}.bg-accent-muted{background-color:var(--color-accent-muted)}.bg-accent-strong{background-color:var(--color-accent-strong)}.bg-accent\/10{background-color:var(--color-accent)}@supports (color:color-mix(in lab,red,red)){.bg-accent\/10{background-color:color-mix(in oklab,var(--color-accent)10%,transparent)}}.bg-accent\/15{background-color:var(--color-accent)}@supports (color:color-mix(in lab,red,red)){.bg-accent\/15{background-color:color-mix(in oklab,var(--color-accent)15%,transparent)}}.bg-accent\/20{background-color:var(--color-accent)}@supports (color:color-mix(in lab,red,red)){.bg-accent\/20{background-color:color-mix(in oklab,var(--color-accent)20%,transparent)}}.bg-black\/20{background-color:#0003}@supports (color:color-mix(in lab,red,red)){.bg-black\/20{background-color:color-mix(in oklab,var(--color-black)20%,transparent)}}.bg-black\/25{background-color:#00000040}@supports (color:color-mix(in lab,red,red)){.bg-black\/25{background-color:color-mix(in oklab,var(--color-black)25%,transparent)}}.bg-border{background-color:var(--color-border)}.bg-hover{background-color:var(--color-hover)}.bg-panel,.bg-panel\/80{background-color:var(--color-panel)}@supports (color:color-mix(in lab,red,red)){.bg-panel\/80{background-color:color-mix(in oklab,var(--color-panel)80%,transparent)}}.bg-surface{background-color:var(--color-surface)}.bg-surface-inset{background-color:var(--color-surface-inset)}.bg-surface-overlay{background-color:var(--color-surface-overlay)}.bg-transparent{background-color:#0000}.bg-white\/5{background-color:#ffffff0d}@supports (color:color-mix(in lab,red,red)){.bg-white\/5{background-color:color-mix(in oklab,var(--color-white)5%,transparent)}}.object-contain{object-fit:contain}.object-cover{object-fit:cover}.p-0{padding:calc(var(--spacing)*0)}.p-1{padding:calc(var(--spacing)*1)}.p-2{padding:calc(var(--spacing)*2)}.p-3{padding:calc(var(--spacing)*3)}.p-4{padding:calc(var(--spacing)*4)}.px-0\.5{padding-inline:calc(var(--spacing)*.5)}.px-1{padding-inline:calc(var(--spacing)*1)}.px-1\.5{padding-inline:calc(var(--spacing)*1.5)}.px-2{padding-inline:calc(var(--spacing)*2)}.px-2\.5{padding-inline:calc(var(--spacing)*2.5)}.px-3{padding-inline:calc(var(--spacing)*3)}.py-0\.5{padding-block:calc(var(--spacing)*.5)}.py-1{padding-block:calc(var(--spacing)*1)}.py-1\.5{padding-block:calc(var(--spacing)*1.5)}.py-2{padding-block:calc(var(--spacing)*2)}.py-2\.5{padding-block:calc(var(--spacing)*2.5)}.py-3{padding-block:calc(var(--spacing)*3)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.font-mono{font-family:var(--font-mono)}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.leading-\[1\.3\]{--tw-leading:1.3;line-height:1.3}.leading-\[15px\]{--tw-leading:15px;line-height:15px}.leading-\[18px\]{--tw-leading:18px;line-height:18px}.leading-none{--tw-leading:1;line-height:1}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.break-words{overflow-wrap:break-word}.break-all{word-break:break-all}.hyphens-auto{-webkit-hyphens:auto;hyphens:auto}.whitespace-pre-wrap{white-space:pre-wrap}.text-\[\#d9dce2\]{color:#d9dce2}.text-accent{color:var(--color-accent)}.text-danger{color:var(--color-danger)}.text-inherit{color:inherit}.text-muted{color:var(--color-muted)}.text-star-active{color:var(--color-star-active)}.text-star-inactive{color:var(--color-star-inactive)}.text-text{color:var(--color-text)}.text-white\/90{color:#ffffffe6}@supports (color:color-mix(in lab,red,red)){.text-white\/90{color:color-mix(in oklab,var(--color-white)90%,transparent)}}.uppercase{text-transform:uppercase}.no-underline{text-decoration-line:none}.underline{text-decoration-line:underline}.underline-offset-2{text-underline-offset:2px}.opacity-0{opacity:0}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-60{opacity:.6}.opacity-70{opacity:.7}.opacity-85{opacity:.85}.opacity-100{opacity:1}.opacity-\[0\.98\]{opacity:.98}.shadow-\[0_1px_0_rgba\(255\,255\,255\,\.04\)\,0_6px_8px_-6px_rgba\(0\,0\,0\,\.5\)\]{--tw-shadow:0 1px 0 var(--tw-shadow-color,#ffffff0a),0 6px 8px -6px var(--tw-shadow-color,#00000080);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.outline-2{outline-style:var(--tw-outline-style);outline-width:2px}.outline-offset-2{outline-offset:2px}.outline-accent{outline-color:var(--color-accent)}.filter{filter:var(--tw-blur,)var(--tw-brightness,)var(--tw-contrast,)var(--tw-grayscale,)var(--tw-hue-rotate,)var(--tw-invert,)var(--tw-saturate,)var(--tw-sepia,)var(--tw-drop-shadow,)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,)}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-75{--tw-duration:75ms;transition-duration:75ms}.duration-200{--tw-duration:.2s;transition-duration:.2s}.duration-\[110ms\]{--tw-duration:.11s;transition-duration:.11s}.duration-\[140ms\]{--tw-duration:.14s;transition-duration:.14s}.duration-\[160ms\]{--tw-duration:.16s;transition-duration:.16s}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}.will-change-transform{will-change:transform}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}@media (hover:hover){.group-hover\:opacity-50:is(:where(.group):hover *){opacity:.5}}.placeholder\:text-muted::placeholder{color:var(--color-muted)}@media (hover:hover){.hover\:border-border:hover{border-color:var(--color-border)}.hover\:bg-accent\/20:hover{background-color:var(--color-accent)}@supports (color:color-mix(in lab,red,red)){.hover\:bg-accent\/20:hover{background-color:color-mix(in oklab,var(--color-accent)20%,transparent)}}.hover\:bg-black\/35:hover{background-color:#00000059}@supports (color:color-mix(in lab,red,red)){.hover\:bg-black\/35:hover{background-color:color-mix(in oklab,var(--color-black)35%,transparent)}}.hover\:bg-hover:hover{background-color:var(--color-hover)}.hover\:bg-surface-hover:hover{background-color:var(--color-surface-hover)}.hover\:bg-white\/5:hover{background-color:#ffffff0d}@supports (color:color-mix(in lab,red,red)){.hover\:bg-white\/5:hover{background-color:color-mix(in oklab,var(--color-white)5%,transparent)}}.hover\:text-danger:hover{color:var(--color-danger)}.hover\:text-star-hover:hover{color:var(--color-star-hover)}.hover\:text-text:hover{color:var(--color-text)}.hover\:underline:hover{text-decoration-line:underline}.hover\:opacity-100:hover{opacity:1}.hover\:outline:hover,.hover\:outline-1:hover{outline-style:var(--tw-outline-style);outline-width:1px}.hover\:outline-accent:hover{outline-color:var(--color-accent)}}.focus\:w-\[260px\]:focus{width:260px}.focus\:border-border:focus{border-color:var(--color-border)}.focus\:outline-none:focus,.focus-visible\:outline-none:focus-visible{--tw-outline-style:none;outline-style:none}.disabled\:opacity-60:disabled{opacity:.6}.\[\&\:\:-moz-range-thumb\]\:h-3::-moz-range-thumb{height:calc(var(--spacing)*3)}.\[\&\:\:-moz-range-thumb\]\:w-3::-moz-range-thumb{width:calc(var(--spacing)*3)}.\[\&\:\:-moz-range-thumb\]\:rounded-full::-moz-range-thumb{border-radius:3.40282e38px}.\[\&\:\:-moz-range-thumb\]\:border-0::-moz-range-thumb{border-style:var(--tw-border-style);border-width:0}.\[\&\:\:-moz-range-thumb\]\:bg-text::-moz-range-thumb{background-color:var(--color-text)}.\[\&\:\:-webkit-slider-thumb\]\:h-3::-webkit-slider-thumb{height:calc(var(--spacing)*3)}.\[\&\:\:-webkit-slider-thumb\]\:w-3::-webkit-slider-thumb{width:calc(var(--spacing)*3)}.\[\&\:\:-webkit-slider-thumb\]\:appearance-none::-webkit-slider-thumb{-webkit-appearance:none;-moz-appearance:none;appearance:none}.\[\&\:\:-webkit-slider-thumb\]\:rounded-full::-webkit-slider-thumb{border-radius:3.40282e38px}.\[\&\:\:-webkit-slider-thumb\]\:bg-text::-webkit-slider-thumb{background-color:var(--color-text)}}.scrollbar-thin{scrollbar-width:thin;scrollbar-color:#fff3 transparent}.scrollbar-thin::-webkit-scrollbar{width:10px;height:10px}.scrollbar-thin::-webkit-scrollbar-track{background:0 0}.scrollbar-thin::-webkit-scrollbar-thumb{border:2px solid var(--bg);background:#fff3;border-radius:10px}.scrollbar-thin::-webkit-scrollbar-thumb:hover{background:#ffffff4d}.zoom-slider{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:0 0;width:140px;height:18px}.zoom-slider:focus{outline:none}.zoom-slider::-webkit-slider-runnable-track{background:var(--border);border-radius:2px;height:3px}.zoom-slider::-webkit-slider-thumb{-webkit-appearance:none;background:var(--text);border-radius:50%;width:12px;height:12px;margin-top:-4.5px}.zoom-slider::-moz-range-track{background:var(--border);border-radius:2px;height:3px}.zoom-slider::-moz-range-thumb{background:var(--text);border:none;border-radius:50%;width:12px;height:12px}:focus-visible{outline:2px solid var(--accent);outline-offset:2px;border-radius:6px}.drag-ghost{opacity:.8;background:var(--hover);pointer-events:none;border-radius:8px;width:160px;height:120px;position:fixed;top:-1000px;left:-1000px;overflow:hidden;box-shadow:0 6px 18px #00000073}.drag-ghost>img{object-fit:cover;width:100%;height:100%;display:block}.thumb-filename{-webkit-line-clamp:2;text-overflow:ellipsis;word-break:break-word;-webkit-box-orient:vertical;max-width:100%;display:-webkit-box;overflow:hidden}.drop-target{box-shadow:inset 0 0 0 2px var(--accent)!important;background:#3a8fff26!important}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}@keyframes zoomIn{0%{opacity:0;transform:scale(.95)}to{opacity:1;transform:scale(1)}}@keyframes slideDown{0%{opacity:0;transform:translateY(-4px)}to{opacity:1;transform:translateY(0)}}.sr-only{width:1px;height:1px;position:absolute;top:auto;left:-9999px;overflow:hidden}.btn{border:1px solid var(--border);background:var(--surface);height:32px;color:var(--text);cursor:pointer;white-space:nowrap;border-radius:8px;justify-content:center;align-items:center;gap:6px;padding:0 10px;font-size:13px;font-weight:400;transition:background .1s,border-color .1s;display:inline-flex}.btn:hover{background:var(--surface-hover);border-color:var(--border-strong)}.btn:active{background:var(--surface-active)}.btn:disabled{opacity:.5;cursor:not-allowed}.btn-icon{width:32px;padding:0}.btn-sm{height:28px;padding:0 8px;font-size:12px}.btn-active{background:var(--accent-muted);border-color:var(--accent-strong)}.btn-ghost{background:0 0;border-color:#0000}.btn-ghost:hover{background:var(--hover);border-color:#0000}.dropdown-trigger{border:1px solid var(--border);background:var(--surface);height:32px;color:var(--text);cursor:pointer;border-radius:8px;align-items:center;gap:6px;padding:0 10px;font-size:13px;transition:background .1s,border-color .1s;display:inline-flex}.dropdown-trigger:hover{background:var(--surface-hover);border-color:var(--border-strong)}.dropdown-panel{z-index:var(--z-menu);background:var(--surface);border:1px solid var(--border);min-width:160px;box-shadow:var(--shadow-lg);border-radius:10px;padding:6px;animation:.12s ease-out slideDown;position:absolute}.dropdown-item{width:100%;color:var(--text);cursor:pointer;text-align:left;background:0 0;border:none;border-radius:6px;align-items:center;gap:8px;padding:8px 10px;font-size:13px;transition:background .1s;display:flex}.dropdown-item:hover{background:var(--hover)}.dropdown-item[data-active=true]{background:var(--accent-muted);color:var(--accent)}.dropdown-item:disabled{color:var(--muted);cursor:not-allowed}.dropdown-divider{background:var(--border);height:1px;margin:6px 0}.dropdown-label{text-transform:uppercase;letter-spacing:.5px;color:var(--muted);padding:6px 10px 4px;font-size:11px;font-weight:500}.filter-chip{background:var(--accent-muted);border:1px solid var(--border);height:26px;color:var(--text);border-radius:13px;align-items:center;gap:6px;padding:0 8px 0 10px;font-size:12px;display:inline-flex}.filter-chip-remove{border:1px solid var(--border);width:18px;height:18px;color:var(--text);cursor:pointer;background:#00000040;border-radius:50%;justify-content:center;align-items:center;font-size:14px;line-height:1;transition:background .1s;display:flex}.filter-chip-remove:hover{background:#0006}@media (prefers-reduced-motion:reduce){*{transition:none!important;animation:none!important}}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}
|
|
Binary file
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<link rel="icon" type="image/x-icon" href="/favicon.ico" />
|
|
7
|
+
<title>Lenslet</title>
|
|
8
|
+
<script type="module" crossorigin src="/assets/index-B-0lZ7yu.js"></script>
|
|
9
|
+
<link rel="stylesheet" crossorigin href="/assets/index-c56aKxHZ.css">
|
|
10
|
+
</head>
|
|
11
|
+
<body>
|
|
12
|
+
<div id="root"></div>
|
|
13
|
+
</body>
|
|
14
|
+
</html>
|
lenslet/metadata.py
ADDED
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
import io
|
|
3
|
+
import struct
|
|
4
|
+
import zlib
|
|
5
|
+
from typing import Any, Dict, List
|
|
6
|
+
|
|
7
|
+
from PIL import Image, PngImagePlugin
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
PNG_SIGNATURE = b"\x89PNG\r\n\x1a\n"
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def _parse_png_text_chunks(data: bytes) -> List[Dict[str, Any]]:
|
|
14
|
+
"""
|
|
15
|
+
Parse PNG chunks and extract tEXt, zTXt, and iTXt entries.
|
|
16
|
+
"""
|
|
17
|
+
if not data.startswith(PNG_SIGNATURE):
|
|
18
|
+
raise ValueError("Not a PNG file.")
|
|
19
|
+
|
|
20
|
+
pos = len(PNG_SIGNATURE)
|
|
21
|
+
out: List[Dict[str, Any]] = []
|
|
22
|
+
|
|
23
|
+
while pos + 8 <= len(data):
|
|
24
|
+
# Read chunk length and type
|
|
25
|
+
length = struct.unpack(">I", data[pos:pos+4])[0]
|
|
26
|
+
ctype = data[pos+4:pos+8]
|
|
27
|
+
pos += 8
|
|
28
|
+
|
|
29
|
+
if pos + length + 4 > len(data):
|
|
30
|
+
break
|
|
31
|
+
|
|
32
|
+
cdata = data[pos:pos+length]
|
|
33
|
+
pos += length
|
|
34
|
+
|
|
35
|
+
# Skip CRC (4 bytes)
|
|
36
|
+
pos += 4
|
|
37
|
+
|
|
38
|
+
if ctype == b"tEXt":
|
|
39
|
+
# Latin-1: key\0value
|
|
40
|
+
try:
|
|
41
|
+
null_idx = cdata.index(b"\x00")
|
|
42
|
+
key = cdata[:null_idx].decode("latin-1", "replace")
|
|
43
|
+
text = cdata[null_idx+1:].decode("latin-1", "replace")
|
|
44
|
+
out.append({"type": "tEXt", "keyword": key, "text": text})
|
|
45
|
+
except Exception:
|
|
46
|
+
pass
|
|
47
|
+
|
|
48
|
+
elif ctype == b"zTXt":
|
|
49
|
+
# key\0compression_method(1) + compressed data
|
|
50
|
+
try:
|
|
51
|
+
null_idx = cdata.index(b"\x00")
|
|
52
|
+
key = cdata[:null_idx].decode("latin-1", "replace")
|
|
53
|
+
method = cdata[null_idx+1:null_idx+2]
|
|
54
|
+
comp = cdata[null_idx+2:]
|
|
55
|
+
if method == b"\x00": # zlib/deflate
|
|
56
|
+
text = zlib.decompress(comp).decode("latin-1", "replace")
|
|
57
|
+
out.append({"type": "zTXt", "keyword": key, "text": text})
|
|
58
|
+
except Exception:
|
|
59
|
+
pass
|
|
60
|
+
|
|
61
|
+
elif ctype == b"iTXt":
|
|
62
|
+
# UTF-8: key\0flag(1)\0method(1)\0lang\0translated\0text
|
|
63
|
+
try:
|
|
64
|
+
i0 = cdata.index(b"\x00")
|
|
65
|
+
key = cdata[:i0].decode("latin-1", "replace")
|
|
66
|
+
comp_flag = cdata[i0+1:i0+2]
|
|
67
|
+
comp_method = cdata[i0+2:i0+3]
|
|
68
|
+
rest = cdata[i0+3:]
|
|
69
|
+
|
|
70
|
+
i1 = rest.index(b"\x00")
|
|
71
|
+
language_tag = rest[:i1].decode("ascii", "replace")
|
|
72
|
+
rest2 = rest[i1+1:]
|
|
73
|
+
|
|
74
|
+
i2 = rest2.index(b"\x00")
|
|
75
|
+
translated_keyword = rest2[:i2].decode("utf-8", "replace")
|
|
76
|
+
text_bytes = rest2[i2+1:]
|
|
77
|
+
|
|
78
|
+
if comp_flag == b"\x01" and comp_method == b"\x00":
|
|
79
|
+
text = zlib.decompress(text_bytes).decode("utf-8", "replace")
|
|
80
|
+
else:
|
|
81
|
+
text = text_bytes.decode("utf-8", "replace")
|
|
82
|
+
|
|
83
|
+
out.append({
|
|
84
|
+
"type": "iTXt",
|
|
85
|
+
"keyword": key,
|
|
86
|
+
"language_tag": language_tag,
|
|
87
|
+
"translated_keyword": translated_keyword,
|
|
88
|
+
"text": text,
|
|
89
|
+
})
|
|
90
|
+
except Exception:
|
|
91
|
+
pass
|
|
92
|
+
|
|
93
|
+
if ctype == b"IEND":
|
|
94
|
+
break
|
|
95
|
+
|
|
96
|
+
return out
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
def read_png_info(file_obj) -> Dict[str, Any]:
|
|
100
|
+
"""
|
|
101
|
+
Given an uploaded file (path or file-like), return structured PNG text info.
|
|
102
|
+
|
|
103
|
+
Returns a dict with:
|
|
104
|
+
- found_text_chunks: list of parsed PNG text chunks (tEXt, zTXt, iTXt)
|
|
105
|
+
- pil_info: raw PIL .info dict
|
|
106
|
+
- quick_fields: commonly-used fields extracted for convenience
|
|
107
|
+
"""
|
|
108
|
+
if hasattr(file_obj, "read"):
|
|
109
|
+
data = file_obj.read()
|
|
110
|
+
else:
|
|
111
|
+
with open(file_obj, "rb") as f:
|
|
112
|
+
data = f.read()
|
|
113
|
+
|
|
114
|
+
# Parse PNG text chunks manually for full extraction
|
|
115
|
+
found_text_chunks = _parse_png_text_chunks(data)
|
|
116
|
+
|
|
117
|
+
# Also get PIL's parsed info (may overlap but provides additional fields)
|
|
118
|
+
pil_info: Dict[str, Any] = {}
|
|
119
|
+
try:
|
|
120
|
+
img = Image.open(io.BytesIO(data))
|
|
121
|
+
pil_info = dict(img.info)
|
|
122
|
+
for k, v in list(pil_info.items()):
|
|
123
|
+
if isinstance(v, (bytes, bytearray)):
|
|
124
|
+
try:
|
|
125
|
+
pil_info[k] = v.decode("utf-8", "replace")
|
|
126
|
+
except Exception:
|
|
127
|
+
pil_info[k] = repr(v)
|
|
128
|
+
elif isinstance(v, PngImagePlugin.PngInfo):
|
|
129
|
+
pil_info[k] = "PngInfo(...)"
|
|
130
|
+
except Exception as e:
|
|
131
|
+
pil_info = {"_error": f"Pillow failed to open PNG: {e}"}
|
|
132
|
+
|
|
133
|
+
# Build quick_fields from commonly-used metadata keys
|
|
134
|
+
quick_fields: Dict[str, Any] = {}
|
|
135
|
+
# First check PIL info for these fields
|
|
136
|
+
for field in ("parameters", "Software", "prompt", "Description"):
|
|
137
|
+
if field in pil_info:
|
|
138
|
+
quick_fields[field] = pil_info[field]
|
|
139
|
+
# Also check extracted chunks as fallback
|
|
140
|
+
for chunk in found_text_chunks:
|
|
141
|
+
keyword = chunk.get("keyword", "")
|
|
142
|
+
if keyword in ("parameters", "Software", "prompt", "Description"):
|
|
143
|
+
if keyword not in quick_fields:
|
|
144
|
+
quick_fields[keyword] = chunk.get("text", "")
|
|
145
|
+
|
|
146
|
+
return {
|
|
147
|
+
"found_text_chunks": found_text_chunks,
|
|
148
|
+
"pil_info": pil_info,
|
|
149
|
+
"quick_fields": quick_fields,
|
|
150
|
+
}
|
|
151
|
+
|