@home-cinema/app 0.0.2

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.
@@ -0,0 +1 @@
1
+ *,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}:before,:after{--tw-content: ""}html,:host{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji",Segoe UI Symbol,"Noto Color Emoji";font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}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;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dl,dd,h1,h2,h3,h4,h5,h6,hr,figure,p,pre{margin:0}fieldset{margin:0;padding:0}legend{padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}.pointer-events-none{pointer-events:none}.static{position:static}.fixed{position:fixed}.\!absolute{position:absolute!important}.absolute{position:absolute}.relative{position:relative}.left-0{left:0}.left-2{left:.5rem}.left-\[-40px\]{left:-40px}.left-\[50\%\]{left:50%}.right-\[-40px\]{right:-40px}.top-0{top:0}.top-2{top:.5rem}.top-20{top:5rem}.top-5{top:1.25rem}.top-\[-50px\]{top:-50px}.top-\[13px\]{top:13px}.top-\[20\%\]{top:20%}.top-\[30\%\]{top:30%}.top-\[50\%\]{top:50%}.\!z-10{z-index:10!important}.-z-10{z-index:-10}.z-0{z-index:0}.z-10{z-index:10}.z-20{z-index:20}.z-30{z-index:30}.z-40{z-index:40}.z-50{z-index:50}.z-\[-10\]{z-index:-10}.z-\[100\]{z-index:100}.col-span-1{grid-column:span 1 / span 1}.col-span-3{grid-column:span 3 / span 3}.col-span-4{grid-column:span 4 / span 4}.col-span-6{grid-column:span 6 / span 6}.col-span-7{grid-column:span 7 / span 7}.col-span-9{grid-column:span 9 / span 9}.\!ms-4{margin-inline-start:1rem!important}.\!ms-\[50\%\]{margin-inline-start:50%!important}.\!mt-10{margin-top:2.5rem!important}.\!mt-3{margin-top:.75rem!important}.\!mt-5{margin-top:1.25rem!important}.mb-10{margin-bottom:2.5rem}.mb-2{margin-bottom:.5rem}.mb-20{margin-bottom:5rem}.mb-3{margin-bottom:.75rem}.mb-5{margin-bottom:1.25rem}.ml-2{margin-left:.5rem}.mr-2{margin-right:.5rem}.mr-3{margin-right:.75rem}.mr-5{margin-right:1.25rem}.ms-3{margin-inline-start:.75rem}.ms-5{margin-inline-start:1.25rem}.ms-\[100\%\]{margin-inline-start:100%}.ms-\[50\%\]{margin-inline-start:50%}.ms-\[90\%\]{margin-inline-start:90%}.mt-1{margin-top:.25rem}.mt-10{margin-top:2.5rem}.mt-16{margin-top:4rem}.mt-2{margin-top:.5rem}.mt-20{margin-top:5rem}.mt-24{margin-top:6rem}.mt-28{margin-top:7rem}.mt-3{margin-top:.75rem}.mt-5{margin-top:1.25rem}.mt-7{margin-top:1.75rem}.mt-\[30\%\]{margin-top:30%}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.grid{display:grid}.\!hidden{display:none!important}.hidden{display:none}.h-10{height:2.5rem}.h-2{height:.5rem}.h-3{height:.75rem}.h-5{height:1.25rem}.h-7{height:1.75rem}.h-8{height:2rem}.h-\[172px\]{height:172px}.h-\[200px\]{height:200px}.h-\[2px\]{height:2px}.h-\[300px\]{height:300px}.h-\[345px\]{height:345px}.h-\[450px\]{height:450px}.h-\[588px\]{height:588px}.h-\[80vh\]{height:80vh}.h-fit{height:-moz-fit-content;height:fit-content}.h-full{height:100%}.h-screen{height:100vh}.max-h-\[300px\]{max-height:300px}.min-h-\[100px\]{min-height:100px}.min-h-\[450px\]{min-height:450px}.\!w-\[150px\]{width:150px!important}.\!w-\[225px\]{width:225px!important}.\!w-fit{width:-moz-fit-content!important;width:fit-content!important}.w-0{width:0px}.w-10{width:2.5rem}.w-20{width:5rem}.w-5{width:1.25rem}.w-7{width:1.75rem}.w-8{width:2rem}.w-\[115px\]{width:115px}.w-\[200px\]{width:200px}.w-\[230px\]{width:230px}.w-\[300px\]{width:300px}.w-\[700px\]{width:700px}.w-\[90\%\]{width:90%}.w-fit{width:-moz-fit-content;width:fit-content}.w-full{width:100%}.min-w-\[300px\]{min-width:300px}.max-w-\[500px\]{max-width:500px}.shrink-0{flex-shrink:0}.translate-x-\[-100\%\]{--tw-translate-x: -100%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-\[-50\%\]{--tw-translate-x: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-\[-90\%\]{--tw-translate-x: -90%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-y-\[-30\%\]{--tw-translate-y: -30%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-y-\[-50\%\]{--tw-translate-y: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.rotate-\[30deg\]{--tw-rotate: 30deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-75{--tw-scale-x: .75;--tw-scale-y: .75;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.cursor-pointer{cursor:pointer}.list-none{list-style-type:none}.grid-cols-10{grid-template-columns:repeat(10,minmax(0,1fr))}.grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-col-reverse{flex-direction:column-reverse}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.\!justify-center{justify-content:center!important}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.gap-1{gap:.25rem}.gap-10{gap:2.5rem}.gap-2{gap:.5rem}.gap-20{gap:5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.gap-5{gap:1.25rem}.gap-6{gap:1.5rem}.gap-7{gap:1.75rem}.gap-8{gap:2rem}.overflow-hidden{overflow:hidden}.overflow-visible{overflow:visible}.overflow-y-auto{overflow-y:auto}.overflow-x-scroll{overflow-x:scroll}.overflow-y-scroll{overflow-y:scroll}.whitespace-nowrap{white-space:nowrap}.text-nowrap{text-wrap:nowrap}.break-words{overflow-wrap:break-word}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-md{border-radius:.375rem}.rounded-sm{border-radius:.125rem}.rounded-xl{border-radius:.75rem}.\!border-0{border-width:0px!important}.border{border-width:1px}.border-2{border-width:2px}.border-4{border-width:4px}.border-\[1px\]{border-width:1px}.border-\[2px\]{border-width:2px}.border-\[\#ffffff4f\]{border-color:#ffffff4f}.border-green-600{--tw-border-opacity: 1;border-color:rgb(22 163 74 / var(--tw-border-opacity, 1))}.border-red-600{--tw-border-opacity: 1;border-color:rgb(220 38 38 / var(--tw-border-opacity, 1))}.border-red-700{--tw-border-opacity: 1;border-color:rgb(185 28 28 / var(--tw-border-opacity, 1))}.border-white{--tw-border-opacity: 1;border-color:rgb(255 255 255 / var(--tw-border-opacity, 1))}.border-yellow-500{--tw-border-opacity: 1;border-color:rgb(234 179 8 / var(--tw-border-opacity, 1))}.\!bg-\[\#000000\]{--tw-bg-opacity: 1 !important;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))!important}.\!bg-\[\#151515\]{--tw-bg-opacity: 1 !important;background-color:rgb(21 21 21 / var(--tw-bg-opacity, 1))!important}.\!bg-transparent{background-color:transparent!important}.bg-\[\#00000050\]{background-color:#00000050}.bg-\[\#0000005c\]{background-color:#0000005c}.bg-\[\#0000007b\]{background-color:#0000007b}.bg-\[\#0000009c\]{background-color:#0000009c}.bg-\[\#000000\]{--tw-bg-opacity: 1;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))}.bg-\[\#000000a3\]{background-color:#000000a3}.bg-\[\#000000c0\]{background-color:#000000c0}.bg-\[\#080808\]{--tw-bg-opacity: 1;background-color:rgb(8 8 8 / var(--tw-bg-opacity, 1))}.bg-\[\#0f0f0f\]{--tw-bg-opacity: 1;background-color:rgb(15 15 15 / var(--tw-bg-opacity, 1))}.bg-\[\#141414cb\]{background-color:#141414cb}.bg-\[\#202020\]{--tw-bg-opacity: 1;background-color:rgb(32 32 32 / var(--tw-bg-opacity, 1))}.bg-\[\#212121e4\]{background-color:#212121e4}.bg-\[\#444\]{--tw-bg-opacity: 1;background-color:rgb(68 68 68 / var(--tw-bg-opacity, 1))}.bg-\[\#b4b4b43e\]{background-color:#b4b4b43e}.bg-\[\#ff38384f\]{background-color:#ff38384f}.bg-\[\#ffffff0d\]{background-color:#ffffff0d}.bg-black{--tw-bg-opacity: 1;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))}.bg-green-600{--tw-bg-opacity: 1;background-color:rgb(22 163 74 / var(--tw-bg-opacity, 1))}.bg-red-600{--tw-bg-opacity: 1;background-color:rgb(220 38 38 / var(--tw-bg-opacity, 1))}.bg-transparent{background-color:transparent}.bg-white{--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.bg-yellow-500{--tw-bg-opacity: 1;background-color:rgb(234 179 8 / var(--tw-bg-opacity, 1))}.bg-gradient-to-t{background-image:linear-gradient(to top,var(--tw-gradient-stops))}.from-\[\#000000\]{--tw-gradient-from: #000000 var(--tw-gradient-from-position);--tw-gradient-to: rgb(0 0 0 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.to-\[\#00000086\]{--tw-gradient-to: #00000086 var(--tw-gradient-to-position)}.to-\[\#00000091\]{--tw-gradient-to: #00000091 var(--tw-gradient-to-position)}.p-0{padding:0}.p-1{padding:.25rem}.p-10{padding:2.5rem}.p-2{padding:.5rem}.p-3{padding:.75rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.\!pb-0{padding-bottom:0!important}.\!pb-2{padding-bottom:.5rem!important}.\!pr-2{padding-right:.5rem!important}.\!pr-5{padding-right:1.25rem!important}.\!pr-7{padding-right:1.75rem!important}.\!ps-2{padding-inline-start:.5rem!important}.\!ps-5{padding-inline-start:1.25rem!important}.\!ps-7{padding-inline-start:1.75rem!important}.\!pt-0{padding-top:0!important}.\!pt-2{padding-top:.5rem!important}.pb-2{padding-bottom:.5rem}.pr-10{padding-right:2.5rem}.pr-2{padding-right:.5rem}.pr-3{padding-right:.75rem}.pr-5{padding-right:1.25rem}.pr-8{padding-right:2rem}.pr-\[4px\]{padding-right:4px}.ps-10{padding-inline-start:2.5rem}.ps-2{padding-inline-start:.5rem}.ps-3{padding-inline-start:.75rem}.ps-5{padding-inline-start:1.25rem}.ps-8{padding-inline-start:2rem}.ps-\[4px\]{padding-inline-start:4px}.pt-10{padding-top:2.5rem}.pt-2{padding-top:.5rem}.text-center{text-align:center}.\!text-\[14px\]{font-size:14px!important}.\!text-base{font-size:1rem!important;line-height:1.5rem!important}.\!text-sm{font-size:.875rem!important;line-height:1.25rem!important}.text-2xl{font-size:1.5rem;line-height:2rem}.text-3xl{font-size:1.875rem;line-height:2.25rem}.text-4xl{font-size:2.25rem;line-height:2.5rem}.text-5xl{font-size:3rem;line-height:1}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.text-base{font-size:1rem;line-height:1.5rem}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.font-black{font-weight:900}.font-bold{font-weight:700}.font-extrabold{font-weight:800}.font-normal{font-weight:400}.font-semibold{font-weight:600}.\!text-red-500{--tw-text-opacity: 1 !important;color:rgb(239 68 68 / var(--tw-text-opacity, 1))!important}.\!text-white{--tw-text-opacity: 1 !important;color:rgb(255 255 255 / var(--tw-text-opacity, 1))!important}.text-black{--tw-text-opacity: 1;color:rgb(0 0 0 / var(--tw-text-opacity, 1))}.text-blue-500{--tw-text-opacity: 1;color:rgb(59 130 246 / var(--tw-text-opacity, 1))}.text-blue-600{--tw-text-opacity: 1;color:rgb(37 99 235 / var(--tw-text-opacity, 1))}.text-green-600{--tw-text-opacity: 1;color:rgb(22 163 74 / var(--tw-text-opacity, 1))}.text-red-500{--tw-text-opacity: 1;color:rgb(239 68 68 / var(--tw-text-opacity, 1))}.text-red-600{--tw-text-opacity: 1;color:rgb(220 38 38 / var(--tw-text-opacity, 1))}.text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.text-yellow-500{--tw-text-opacity: 1;color:rgb(234 179 8 / var(--tw-text-opacity, 1))}.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}.outline-none{outline:2px solid transparent;outline-offset:2px}.blur-3xl{--tw-blur: blur(64px);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)}.blur-\[100px\]{--tw-blur: blur(100px);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)}.blur-\[253px\]{--tw-blur: blur(253px);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)}.drop-shadow-md{--tw-drop-shadow: drop-shadow(0 4px 3px rgb(0 0 0 / .07)) drop-shadow(0 2px 2px rgb(0 0 0 / .06));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)}.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(4px);-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{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.duration-100{transition-duration:.1s}.duration-150{transition-duration:.15s}.duration-200{transition-duration:.2s}.duration-300{transition-duration:.3s}.duration-500{transition-duration:.5s}.duration-\[2000ms\]{transition-duration:2s}.ease-out{transition-timing-function:cubic-bezier(0,0,.2,1)}:root{font-family:Inter,system-ui,Avenir,Helvetica,Arial,sans-serif;line-height:1.5;font-weight:400;color-scheme:light dark;color:#ffffffde;background-color:#000;font-synthesis:none;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}@keyframes shimmer{0%{background-position:-100% 0}to{background-position:100% 0}}.loading-background{background:linear-gradient(90deg,#5c5c5c 25%,#707070,#5d5d5d 75%);background-size:200% 100%;animation:shimmer 1.5s infinite}.tv_show{background-size:100%;background-position:center}.tv_show:hover{background-size:110%}@keyframes pop{0%{scale:1}10%{scale:1.25}80%{scale:1.25}to{scale:1}}.pop-animation{animation:pop .7s ease-in-out alternate}.placeholder\:font-bold::-moz-placeholder{font-weight:700}.placeholder\:font-bold::placeholder{font-weight:700}.after\:bg-black:after{content:var(--tw-content);--tw-bg-opacity: 1;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))}.hover\:\!bg-\[\#242424\]:hover{--tw-bg-opacity: 1 !important;background-color:rgb(36 36 36 / var(--tw-bg-opacity, 1))!important}.hover\:bg-\[\#50505059\]:hover{background-color:#50505059}.hover\:bg-\[\#b4b4b43e\]:hover{background-color:#b4b4b43e}.hover\:underline:hover{text-decoration-line:underline}.hover\:opacity-100:hover{opacity:1}.hover\:opacity-15:hover{opacity:.15}@media (min-width: 640px){.sm\:left-auto{left:auto}.sm\:top-auto{top:auto}.sm\:block{display:block}.sm\:hidden{display:none}.sm\:h-\[500px\]{height:500px}.sm\:w-\[500px\]{width:500px}.sm\:pr-32{padding-right:8rem}.sm\:ps-32{padding-inline-start:8rem}.sm\:text-lg{font-size:1.125rem;line-height:1.75rem}}@media (min-width: 768px){.md\:relative{position:relative}.md\:left-2{left:.5rem}.md\:top-2{top:.5rem}.md\:mb-20{margin-bottom:5rem}.md\:mr-0{margin-right:0}.md\:mr-10{margin-right:2.5rem}.md\:mt-20{margin-top:5rem}.md\:mt-5{margin-top:1.25rem}.md\:mt-\[15\%\]{margin-top:15%}.md\:\!block{display:block!important}.md\:block{display:block}.md\:flex{display:flex}.md\:grid{display:grid}.md\:hidden{display:none}.md\:h-10{height:2.5rem}.md\:h-20{height:5rem}.md\:w-20{width:5rem}.md\:\!scale-100{--tw-scale-x: 1 !important;--tw-scale-y: 1 !important;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))!important}.md\:grid-cols-11{grid-template-columns:repeat(11,minmax(0,1fr))}.md\:flex-row{flex-direction:row}.md\:flex-nowrap{flex-wrap:nowrap}.md\:items-start{align-items:flex-start}.md\:justify-start{justify-content:flex-start}.md\:gap-16{gap:4rem}.md\:p-10{padding:2.5rem}.md\:p-5{padding:1.25rem}.md\:pb-3{padding-bottom:.75rem}.md\:pe-28{padding-inline-end:7rem}.md\:pr-16{padding-right:4rem}.md\:pr-20{padding-right:5rem}.md\:pr-32{padding-right:8rem}.md\:ps-10{padding-inline-start:2.5rem}.md\:ps-16{padding-inline-start:4rem}.md\:ps-20{padding-inline-start:5rem}.md\:ps-28{padding-inline-start:7rem}.md\:ps-32{padding-inline-start:8rem}.md\:pt-3{padding-top:.75rem}.md\:text-2xl{font-size:1.5rem;line-height:2rem}.md\:text-3xl{font-size:1.875rem;line-height:2.25rem}.md\:text-4xl{font-size:2.25rem;line-height:2.5rem}.md\:text-7xl{font-size:4.5rem;line-height:1}.md\:text-base{font-size:1rem;line-height:1.5rem}.md\:text-lg{font-size:1.125rem;line-height:1.75rem}.md\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width: 1024px){.lg\:relative{position:relative}.lg\:left-0{left:0}.lg\:left-\[300px\]{left:300px}.lg\:top-0{top:0}.lg\:top-\[75px\]{top:75px}.lg\:z-10{z-index:10}.lg\:col-span-3{grid-column:span 3 / span 3}.lg\:ms-32{margin-inline-start:8rem}.lg\:mt-20{margin-top:5rem}.lg\:block{display:block}.lg\:inline-block{display:inline-block}.lg\:flex{display:flex}.lg\:hidden{display:none}.lg\:h-10{height:2.5rem}.lg\:h-5{height:1.25rem}.lg\:h-\[345px\]{height:345px}.lg\:w-\[230px\]{width:230px}.lg\:translate-x-0{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.lg\:grid-cols-12{grid-template-columns:repeat(12,minmax(0,1fr))}.lg\:gap-16{gap:4rem}.lg\:overflow-auto{overflow:auto}.lg\:text-wrap{text-wrap:wrap}.lg\:p-5{padding:1.25rem}.lg\:pr-28{padding-right:7rem}.lg\:ps-28{padding-inline-start:7rem}.lg\:text-base{font-size:1rem;line-height:1.5rem}.lg\:text-lg{font-size:1.125rem;line-height:1.75rem}.lg\:text-xl{font-size:1.25rem;line-height:1.75rem}}@media (min-width: 1280px){.xl\:relative{position:relative}.xl\:z-30{z-index:30}.xl\:col-span-1{grid-column:span 1 / span 1}.xl\:col-span-2{grid-column:span 2 / span 2}.xl\:block{display:block}.xl\:flex{display:flex}.xl\:hidden{display:none}.xl\:w-auto{width:auto}.xl\:justify-between{justify-content:space-between}.xl\:gap-20{gap:5rem}.xl\:bg-\[\#ffffffa9\]{background-color:#ffffffa9}.xl\:opacity-20{opacity:.2}}.horz-list{display:flex;overflow-x:auto;scrollbar-width:thin;scrollbar-color:#888 #f0f0f0;scroll-behavior:smooth}.scroll-container::-webkit-scrollbar{height:8px}.scroll-container::-webkit-scrollbar-thumb{background:#888;border-radius:4px}.scroll-container::-webkit-scrollbar-thumb:hover{background:#555}
@@ -0,0 +1,15 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <link rel="icon" type="image/svg+xml" href="/assets/favicon-RZMYYI_a.svg" />
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
+ <title>Home Cinema</title>
8
+ <!--app-head-->
9
+ <script type="module" crossorigin src="/assets/index-CeKR2k9Q.js"></script>
10
+ <link rel="stylesheet" crossorigin href="/assets/index-DWHHH3Bn.css">
11
+ </head>
12
+ <body>
13
+ <div id="root"><!--app-html--></div>
14
+ </body>
15
+ </html>
@@ -0,0 +1 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
@@ -0,0 +1,44 @@
1
+ import { openDB } from "idb";
2
+ var ObjectStores;
3
+ (function (ObjectStores) {
4
+ ObjectStores["MOVIES"] = "saved_movies";
5
+ ObjectStores["TVSHOWS"] = "saved_tvshows";
6
+ })(ObjectStores || (ObjectStores = {}));
7
+ const dbPromise = openDB("user-data", 1, {
8
+ upgrade(db) {
9
+ db.createObjectStore(ObjectStores.MOVIES, { keyPath: "id" }); // use movie.id as key
10
+ db.createObjectStore(ObjectStores.TVSHOWS, { keyPath: "id" }); // use tv.id as key
11
+ },
12
+ });
13
+ export async function addMovie(movie) {
14
+ const db = await dbPromise;
15
+ db.put(ObjectStores.MOVIES, movie);
16
+ }
17
+ export async function getMovies() {
18
+ const db = await dbPromise;
19
+ return await db.getAll(ObjectStores.MOVIES);
20
+ }
21
+ export async function getMovieById(id) {
22
+ const db = await dbPromise;
23
+ return await db.get(ObjectStores.MOVIES, id);
24
+ }
25
+ export async function removeMovie(id) {
26
+ const db = await dbPromise;
27
+ await db.delete(ObjectStores.MOVIES, id);
28
+ }
29
+ export async function addTVShow(tv) {
30
+ const db = await dbPromise;
31
+ db.put(ObjectStores.TVSHOWS, tv);
32
+ }
33
+ export async function getTVShows() {
34
+ const db = await dbPromise;
35
+ return await db.getAll(ObjectStores.TVSHOWS);
36
+ }
37
+ export async function getTVShowById(id) {
38
+ const db = await dbPromise;
39
+ return await db.get(ObjectStores.TVSHOWS, id);
40
+ }
41
+ export async function removeTVShow(id) {
42
+ const db = await dbPromise;
43
+ await db.delete(ObjectStores.TVSHOWS, id);
44
+ }
@@ -0,0 +1,116 @@
1
+ import axios, { AxiosError } from "axios";
2
+ export var TMDBErrorStatusCode;
3
+ (function (TMDBErrorStatusCode) {
4
+ TMDBErrorStatusCode[TMDBErrorStatusCode["GET_TVSHOWS_ERR"] = 0] = "GET_TVSHOWS_ERR";
5
+ TMDBErrorStatusCode[TMDBErrorStatusCode["BAD_REQUEST"] = 1] = "BAD_REQUEST";
6
+ })(TMDBErrorStatusCode || (TMDBErrorStatusCode = {}));
7
+ export class TMDBError extends Error {
8
+ code = 0;
9
+ statusCode;
10
+ constructor(message, statusCode, code) {
11
+ super(message);
12
+ this.name = this.constructor.name;
13
+ this.statusCode = statusCode;
14
+ this.code = code;
15
+ if (Error.captureStackTrace) {
16
+ Error.captureStackTrace(this, this.constructor);
17
+ }
18
+ }
19
+ static format(e) {
20
+ return `[${new Date().toLocaleTimeString()}] STATUS CODE : ${e.statusCode} | ${e.message}`;
21
+ }
22
+ }
23
+ export class TMDBApi {
24
+ static tvShowsendPoint = "https://api.themoviedb.org/3/trending/tv/week";
25
+ static tvShowDetailsEndPoint = "https://api.themoviedb.org/3/tv/:id";
26
+ static tvShowSeasonDetails = "https://api.themoviedb.org/3/tv/:series_id/season/:season_number";
27
+ static tvShowsSearchEndPoint = "https://api.themoviedb.org/3/search/tv";
28
+ api_key;
29
+ constructor(api_key) {
30
+ this.api_key = api_key;
31
+ }
32
+ authHeader() {
33
+ return `Bearer ${this.api_key}`;
34
+ }
35
+ handleAxiosErr(e) {
36
+ if (e.status && e.status < 500) {
37
+ throw new TMDBError(`${(e.response?.data).status_message}`, e.status, TMDBErrorStatusCode.BAD_REQUEST);
38
+ }
39
+ throw new TMDBError(`${e.message}`, e.status, TMDBErrorStatusCode.GET_TVSHOWS_ERR);
40
+ }
41
+ async getTVShows(page) {
42
+ const url = new URL(TMDBApi.tvShowsendPoint);
43
+ url.searchParams.set("page", page || "1");
44
+ try {
45
+ const resp = await axios.get(url.href, {
46
+ headers: {
47
+ Authorization: this.authHeader(),
48
+ },
49
+ });
50
+ return resp.data;
51
+ }
52
+ catch (e) {
53
+ if (e instanceof AxiosError) {
54
+ if (e.status && e.status < 500) {
55
+ throw new TMDBError(`Cannot get tv shows : ${e.response?.data.status_message}`, e.status, TMDBErrorStatusCode.BAD_REQUEST);
56
+ }
57
+ throw new TMDBError(`unable to get tv shows : ${e.message}`, e.status, TMDBErrorStatusCode.GET_TVSHOWS_ERR);
58
+ }
59
+ throw e;
60
+ }
61
+ }
62
+ async getTVShowDetails(id) {
63
+ const url = TMDBApi.tvShowDetailsEndPoint.replace(":id", id);
64
+ try {
65
+ const resp = await axios.get(url, {
66
+ headers: {
67
+ Authorization: this.authHeader(),
68
+ },
69
+ });
70
+ return resp.data;
71
+ }
72
+ catch (err) {
73
+ if (err instanceof AxiosError) {
74
+ this.handleAxiosErr(err);
75
+ }
76
+ throw err;
77
+ }
78
+ }
79
+ async getSeasonDetails(series_id, season_number) {
80
+ let url = TMDBApi.tvShowSeasonDetails.replace(":series_id", series_id);
81
+ url = url.replace(":season_number", season_number);
82
+ try {
83
+ const resp = await axios.get(url, {
84
+ headers: {
85
+ Authorization: this.authHeader(),
86
+ },
87
+ });
88
+ return resp.data;
89
+ }
90
+ catch (err) {
91
+ if (err instanceof AxiosError) {
92
+ this.handleAxiosErr(err);
93
+ }
94
+ throw err;
95
+ }
96
+ }
97
+ async searchTvShows(query, page) {
98
+ let url = new URL(TMDBApi.tvShowsSearchEndPoint);
99
+ url.searchParams.set("query", query);
100
+ url.searchParams.set("page", page);
101
+ try {
102
+ const resp = await axios.get(url.href, {
103
+ headers: {
104
+ Authorization: this.authHeader(),
105
+ },
106
+ });
107
+ return resp.data;
108
+ }
109
+ catch (err) {
110
+ if (err instanceof AxiosError) {
111
+ this.handleAxiosErr(err);
112
+ }
113
+ throw err;
114
+ }
115
+ }
116
+ }
@@ -0,0 +1,10 @@
1
+ import { jsx } from "react/jsx-runtime";
2
+ import { StrictMode } from "react";
3
+ import { renderToString } from "react-dom/server";
4
+ function render(_url) {
5
+ const html = renderToString(/* @__PURE__ */ jsx(StrictMode, {}));
6
+ return { html };
7
+ }
8
+ export {
9
+ render
10
+ };
@@ -0,0 +1 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
package/dist/server.js ADDED
@@ -0,0 +1,8 @@
1
+ import { bootServer } from "./serverBoot.js";
2
+ import env from "dotenv";
3
+ env.config();
4
+ let PORT = 5173;
5
+ if (process.env.NODE_ENV === "production") {
6
+ PORT = 4173;
7
+ }
8
+ bootServer(PORT);
@@ -0,0 +1,77 @@
1
+ import env from "dotenv";
2
+ import fs from "node:fs/promises";
3
+ import express from "express";
4
+ import path from "node:path";
5
+ import { fileURLToPath } from "node:url";
6
+ import routes from "./api/index.js";
7
+ env.config();
8
+ const isProduction = process.env.NODE_ENV === "production";
9
+ const base = process.env.BASE || "/";
10
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
11
+ const configs = JSON.parse(await fs.readFile(path.join(__dirname, "../home_cinema_config.json"), "utf-8"));
12
+ export async function bootServer(port) {
13
+ const templateHtml = isProduction
14
+ ? await fs.readFile(path.join(__dirname, "../dist/client/index.html"), "utf-8")
15
+ : "";
16
+ const app = express();
17
+ app.use(express.json());
18
+ app.use(express.urlencoded({ extended: true }));
19
+ let vite;
20
+ if (!isProduction) {
21
+ const { createServer } = await import("vite");
22
+ vite = await createServer({
23
+ server: { middlewareMode: true },
24
+ appType: "custom",
25
+ base,
26
+ });
27
+ app.use(vite.middlewares);
28
+ }
29
+ else {
30
+ const compression = (await import("compression")).default;
31
+ const sirv = (await import("sirv")).default;
32
+ app.use(compression());
33
+ app.use(base, sirv(path.join(__dirname, "../dist/client"), { extensions: [] }));
34
+ }
35
+ if (configs["torrent-streamer-api"].external === false) {
36
+ const streamerRouter = (await import("torrent-streamer-api")).default;
37
+ app.use(streamerRouter({
38
+ torrentFilesTimeout: 1000 * 30,
39
+ }));
40
+ }
41
+ Object.values(routes).forEach((router) => {
42
+ app.use(router);
43
+ });
44
+ app.use("*", async (req, res) => {
45
+ try {
46
+ const url = req.originalUrl.replace(base, "");
47
+ let template;
48
+ let render;
49
+ if (!isProduction && vite) {
50
+ template = await fs.readFile(path.join(__dirname, "../index.html"), "utf-8");
51
+ template = await vite.transformIndexHtml(url, template);
52
+ render = (await vite.ssrLoadModule(path.join(__dirname, "./entry-server.jsx"))).render;
53
+ }
54
+ else {
55
+ template = templateHtml;
56
+ //@ts-ignore
57
+ render = (await import("./server/entry-server.js")).render;
58
+ }
59
+ const rendered = await render(url);
60
+ const html = template
61
+ .replace("<!--app-head-->", rendered.head ?? "")
62
+ .replace("<!--app-html-->", rendered.html ?? "");
63
+ res.status(200).set({ "Content-Type": "text/html" }).send(html);
64
+ }
65
+ catch (e) {
66
+ vite?.ssrFixStacktrace(e);
67
+ console.error(e);
68
+ res.status(500).end(e.stack);
69
+ }
70
+ });
71
+ return new Promise((resolve) => {
72
+ const server = app.listen(port, () => {
73
+ console.log(`Server started at http://localhost:${port}`);
74
+ resolve(server);
75
+ });
76
+ });
77
+ }
@@ -0,0 +1,6 @@
1
+ {
2
+ "torrent-streamer-api": {
3
+ "external": false,
4
+ "origin": "http://192.168.1.26:8080"
5
+ }
6
+ }
package/package.json ADDED
@@ -0,0 +1,63 @@
1
+ {
2
+ "name": "@home-cinema/app",
3
+ "version": "0.0.2",
4
+ "type": "module",
5
+ "main": "dist/serverBoot.js",
6
+ "files": [
7
+ "dist",
8
+ "home_cinema_config.json"
9
+ ],
10
+ "scripts": {
11
+ "dev": "nodemon",
12
+ "build": "tsc -p tsconfig.server.json && npm run build:client && npm run build:server",
13
+ "lint": "eslint .",
14
+ "preview": "NODE_ENV=production node --experimental-json-modules dist/server",
15
+ "build:client": "vite build --outDir dist/client",
16
+ "build:server": "vite build --outDir dist/server --ssr src/entry-server.jsx"
17
+ },
18
+ "dependencies": {
19
+ "@emotion/react": "^11.14.0",
20
+ "@emotion/styled": "^11.14.0",
21
+ "@fortawesome/fontawesome-svg-core": "^6.7.2",
22
+ "@fortawesome/free-regular-svg-icons": "^7.1.0",
23
+ "@fortawesome/free-solid-svg-icons": "^6.7.2",
24
+ "@fortawesome/react-fontawesome": "^0.2.2",
25
+ "@mui/icons-material": "^7.1.0",
26
+ "@mui/joy": "^5.0.0-beta.52",
27
+ "@mui/material": "^7.1.0",
28
+ "@vitejs/plugin-react": "^4.3.4",
29
+ "axios": "^1.7.9",
30
+ "compression": "^1.7.5",
31
+ "dotenv": "^16.4.7",
32
+ "express": "^4.21.2",
33
+ "idb": "^8.0.3",
34
+ "pretty-bytes": "^6.1.1",
35
+ "react": "^18.3.1",
36
+ "react-dom": "^18.3.1",
37
+ "react-player": "^2.16.0",
38
+ "react-router": "^7.1.1",
39
+ "sirv": "^3.0.0",
40
+ "torrent-streamer-api": "^2.0.0"
41
+ },
42
+ "devDependencies": {
43
+ "@eslint/js": "^9.17.0",
44
+ "@types/compression": "^1.7.5",
45
+ "@types/express": "^5.0.0",
46
+ "@types/node": "^22.10.5",
47
+ "@types/react": "^18.3.18",
48
+ "@types/react-dom": "^18.3.5",
49
+ "@vitejs/plugin-react-swc": "^3.5.0",
50
+ "autoprefixer": "^10.4.20",
51
+ "eslint": "^9.17.0",
52
+ "eslint-plugin-react-hooks": "^5.0.0",
53
+ "eslint-plugin-react-refresh": "^0.4.16",
54
+ "globals": "^15.14.0",
55
+ "nodemon": "^3.1.9",
56
+ "postcss": "^8.4.49",
57
+ "tailwindcss": "^3.4.17",
58
+ "ts-node": "^10.9.2",
59
+ "typescript": "~5.6.2",
60
+ "typescript-eslint": "^8.18.2",
61
+ "vite": "^6.0.5"
62
+ }
63
+ }