2024-06-15 14:55:33 +02:00
|
|
|
"use client";
|
|
|
|
import React, { useState } from "react";
|
|
|
|
import HeroCards from "./HeroCards";
|
2024-11-24 19:33:52 +01:00
|
|
|
import { Button, buttonVariants } from "../ui/button";
|
2024-06-15 14:55:33 +02:00
|
|
|
import Link from "next/link";
|
|
|
|
import { Check, Clipboard } from "lucide-react";
|
|
|
|
import GridPattern from "../ui/grid-pattern";
|
|
|
|
import AnimatedGradientText from "../ui/animated-gradient-text";
|
|
|
|
import { Happy_Monkey } from "next/font/google";
|
|
|
|
import { cn } from "@/lib/utils";
|
|
|
|
import Image from "next/image";
|
|
|
|
|
|
|
|
const happyMonkey = Happy_Monkey({
|
2024-09-07 09:12:48 +02:00
|
|
|
preload: true,
|
|
|
|
weight: ["400"],
|
|
|
|
subsets: ["latin"]
|
2024-06-15 14:55:33 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
const Hero = () => {
|
2024-09-07 09:12:48 +02:00
|
|
|
const [isCopied, setIsCopied] = useState(false);
|
|
|
|
const [command, setCommand] = useState(
|
2025-01-05 10:19:00 +01:00
|
|
|
'sudo bash -c "$(curl -fsSL https://downloads.svrjs.org/installer/svr.js.installer.linux.20250105.sh)"'
|
2024-09-07 09:12:48 +02:00
|
|
|
);
|
2024-12-04 21:21:19 +01:00
|
|
|
const [selectedButton, setSelectedButton] = useState<
|
|
|
|
"linux" | "windows" | "docker"
|
|
|
|
>("linux");
|
2024-06-15 14:55:33 +02:00
|
|
|
|
2024-09-07 09:12:48 +02:00
|
|
|
const commands = {
|
|
|
|
linux:
|
2025-01-05 10:19:00 +01:00
|
|
|
'sudo bash -c "$(curl -fsSL https://downloads.svrjs.org/installer/svr.js.installer.linux.20250105.sh)"',
|
2024-12-04 21:21:19 +01:00
|
|
|
windows:
|
2025-01-05 10:19:00 +01:00
|
|
|
'powershell -c "irm https://downloads.svrjs.org/installer/svr.js.installer.windows.20250105.ps1 | iex"',
|
2024-09-07 09:12:48 +02:00
|
|
|
docker:
|
|
|
|
"docker pull svrjs/svrjs && docker run --name mysvrjs -d -p 80:80 --restart=always svrjs/svrjs"
|
|
|
|
};
|
2024-06-15 14:55:33 +02:00
|
|
|
|
2024-09-07 09:12:48 +02:00
|
|
|
const copyToClipboard = () => {
|
|
|
|
navigator.clipboard.writeText(command);
|
|
|
|
setIsCopied(true);
|
|
|
|
setTimeout(() => setIsCopied(false), 2000);
|
|
|
|
};
|
2024-06-15 14:55:33 +02:00
|
|
|
|
2024-12-04 21:21:19 +01:00
|
|
|
const handleButtonClick = (type: "linux" | "windows" | "docker") => {
|
2024-09-07 09:12:48 +02:00
|
|
|
setCommand(commands[type]);
|
|
|
|
setSelectedButton(type);
|
|
|
|
};
|
2024-06-15 14:55:33 +02:00
|
|
|
|
2024-09-07 09:12:48 +02:00
|
|
|
return (
|
|
|
|
<section className="relative sm:container grid lg:grid-cols-2 place-items-center py-20 md:py-24 gap-10">
|
|
|
|
<GridPattern
|
|
|
|
className={cn(
|
|
|
|
"[mask-image:radial-gradient(300px_circle_at_center,white,transparent)]",
|
|
|
|
"inset-x-0 inset-y-[-50%] h-[200%] opacity-30"
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
<div className="text-center lg:text-start space-y-6">
|
|
|
|
<AnimatedGradientText className="mx-auto lg:mx-0">
|
|
|
|
🎉{" "}
|
|
|
|
<hr className="mx-2 h-4 w-[1px] shrink-0 bg-black dark:bg-gray-300" />
|
|
|
|
<span
|
|
|
|
className={cn(
|
|
|
|
`inline animate-gradient bg-gradient-to-r from-[#235b1a] to-[#315620] dark:bg-gradient-to-r dark:from-[#6df458] dark:to-[#4c932a] bg-[length:var(--bg-size)_100%] bg-clip-text text-transparent`
|
|
|
|
)}
|
|
|
|
>
|
|
|
|
Expanding server functionality
|
|
|
|
</span>
|
|
|
|
</AnimatedGradientText>
|
2024-11-24 19:33:52 +01:00
|
|
|
<div className="text-5xl md:text-6xl font-bold">
|
2024-09-07 09:12:48 +02:00
|
|
|
<h1 className="inline custom-title">
|
2024-09-19 19:38:23 +02:00
|
|
|
Host with{" "}
|
2024-09-07 09:12:48 +02:00
|
|
|
<span className="text-transparent bg-gradient-to-r from-green-300 to-primary bg-clip-text">
|
2024-09-19 19:38:23 +02:00
|
|
|
reliability
|
|
|
|
</span>
|
|
|
|
, scale with{" "}
|
|
|
|
<span className="text-transparent bg-gradient-to-r from-green-300 to-primary bg-clip-text">
|
|
|
|
ease
|
|
|
|
</span>
|
2024-09-07 09:12:48 +02:00
|
|
|
</h1>
|
2024-11-24 19:33:52 +01:00
|
|
|
</div>
|
2024-09-07 09:12:48 +02:00
|
|
|
<p className="text-lg text-muted-foreground md:w-10/12 mx-auto lg:mx-0">
|
2024-09-19 19:38:23 +02:00
|
|
|
Host reliably and securely, and scale effortlessly with SVR.JS, an
|
2025-01-07 16:27:52 +01:00
|
|
|
open-source and configurable web server running on Node.js that's
|
2024-09-19 19:38:23 +02:00
|
|
|
designed to handle high loads.
|
2024-09-07 09:12:48 +02:00
|
|
|
</p>
|
|
|
|
<div className="relative mx-auto lg:mx-0 flex gap-2 flex-col-reverse lg:flex-row justify-start items-center w-fit">
|
|
|
|
<Button
|
|
|
|
className="w-fit"
|
|
|
|
onClick={copyToClipboard}
|
|
|
|
variant={!isCopied ? "secondary" : "secondary"}
|
|
|
|
>
|
|
|
|
{!isCopied ? (
|
|
|
|
<Clipboard className="w-4 h-4 mr-2" />
|
|
|
|
) : (
|
|
|
|
<Check className="w-4 h-4 mr-2" />
|
|
|
|
)}
|
2024-12-02 19:02:26 +01:00
|
|
|
{command.slice(0, 37)}...
|
2024-09-07 09:12:48 +02:00
|
|
|
</Button>
|
|
|
|
<p className="hidden lg:block">|</p>
|
|
|
|
<p className="block lg:hidden">or</p>
|
2024-11-24 19:33:52 +01:00
|
|
|
<Link className={`${buttonVariants()} w-full`} href="/downloads">
|
|
|
|
Download
|
2024-09-07 09:12:48 +02:00
|
|
|
</Link>
|
|
|
|
<div className="pointer-events-none dark:invert -scale-x-100 absolute -bottom-14 max-lg:left-0 lg:right-20 inline-flex justify-center items-center gap-1">
|
|
|
|
<Image
|
|
|
|
src="/curly-arrow.png"
|
|
|
|
width={35}
|
|
|
|
height={35}
|
|
|
|
alt="Curly arrow"
|
|
|
|
/>
|
|
|
|
<span
|
|
|
|
className={cn(
|
|
|
|
`mt-10 font-bold text-black -scale-x-100 text-sm ${happyMonkey.className}`
|
|
|
|
)}
|
|
|
|
>
|
|
|
|
Try Now!
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="flex items-center lg:justify-start justify-center gap-3 w-full">
|
|
|
|
<Button
|
|
|
|
className={`rounded-full w-12 h-12 lg:w-16 lg:h-16 ${
|
|
|
|
selectedButton === "linux"
|
|
|
|
? "bg-accent"
|
|
|
|
: "bg-primary-foreground/20"
|
|
|
|
}`}
|
|
|
|
variant={"ghost"}
|
|
|
|
onClick={() => handleButtonClick("linux")}
|
|
|
|
>
|
|
|
|
<span className="sr-only">Linux</span>
|
|
|
|
<svg
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
2024-09-08 18:23:25 +02:00
|
|
|
width={64}
|
|
|
|
height={64}
|
2024-09-07 09:12:48 +02:00
|
|
|
viewBox="0 0 448 512"
|
|
|
|
fill="none"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
fill="currentColor"
|
|
|
|
d="M220.8 123.3c1 .5 1.8 1.7 3 1.7 1.1 0 2.8-.4 2.9-1.5.2-1.4-1.9-2.3-3.2-2.9-1.7-.7-3.9-1-5.5-.1-.4.2-.8.7-.6 1.1.3 1.3 2.3 1.1 3.4 1.7zm-21.9 1.7c1.2 0 2-1.2 3-1.7 1.1-.6 3.1-.4 3.5-1.6.2-.4-.2-.9-.6-1.1-1.6-.9-3.8-.6-5.5.1-1.3.6-3.4 1.5-3.2 2.9.1 1 1.8 1.5 2.8 1.4zM420 403.8c-3.6-4-5.3-11.6-7.2-19.7-1.8-8.1-3.9-16.8-10.5-22.4-1.3-1.1-2.6-2.1-4-2.9-1.3-.8-2.7-1.5-4.1-2 9.2-27.3 5.6-54.5-3.7-79.1-11.4-30.1-31.3-56.4-46.5-74.4-17.1-21.5-33.7-41.9-33.4-72C311.1 85.4 315.7.1 234.8 0 132.4-.2 158 103.4 156.9 135.2c-1.7 23.4-6.4 41.8-22.5 64.7-18.9 22.5-45.5 58.8-58.1 96.7-6 17.9-8.8 36.1-6.2 53.3-6.5 5.8-11.4 14.7-16.6 20.2-4.2 4.3-10.3 5.9-17 8.3s-14 6-18.5 14.5c-2.1 3.9-2.8 8.1-2.8 12.4 0 3.9.6 7.9 1.2 11.8 1.2 8.1 2.5 15.7.8 20.8-5.2 14.4-5.9 24.4-2.2 31.7 3.8 7.3 11.4 10.5 20.1 12.3 17.3 3.6 40.8 2.7 59.3 12.5 19.8 10.4 39.9 14.1 55.9 10.4 11.6-2.6 21.1-9.6 25.9-20.2 12.5-.1 26.3-5.4 48.3-6.6 14.9-1.2 33.6 5.3 55.1 4.1.6 2.3 1.4 4.6 2.5 6.7v.1c8.3 16.7 23.8 24.3 40.3 23 16.6-1.3 34.1-11 48.3-27.9 13.6-16.4 36-23.2 50.9-32.2 7.4-4.5 13.4-10.1 13.9-18.3.4-8.2-4.4-17.3-15.5-29.7zM223.7 87.3c9.8-22.2 34.2-21.8 44-.4 6.5 14.2 3.6 30.9-4.3 40.4-1.6-.8-5.9-2.6-12.6-4.9 1.1-1.2 3.1-2.7 3.9-4.6 4.8-11.8-.2-27-9.1-27.3-7.3-.5-13.9 10.8-11.8 23-4.1-2-9.4-3.5-13-4.4-1-6.9-.3-14.6 2.9-21.8zM183 75.8c10.1 0 20.8 14.2 19.1 33.5-3.5 1-7.1 2.5-10.2 4.6 1.2-8.9-3.3-20.1-9.6-19.6-8.4.7-9.8 21.2-1.8 28.1 1 .8 1.9-.2-5.9 5.5-15.6-14.6-10.5-52.1 8.4-52.1zm-13.6 60.7c6.2-4.6 13.6-10 14.1-10.5 4.7-4.4 13.5-14.2 27.9-14.2 7.1 0 15.6 2.3 25.9 8.9 6.3 4.1 11.3 4.4 22.6 9.3 8.4 3.5 13.7 9.7 10.5 18.2-2.6 7.1-11 14.4-22.7 18.1-11.1 3.6-19.8 16-38.2 14.9-3.9-.2-7-1-9.6-2.1-8-3.5-12.2-10.4-20-15-8.6-4.8-13.2-10.4-14.7-15.3-1.4-4.9 0-9 4.2-12.3zm3.3 334c-2.7 35.1-43.9 34.4-75.3 18-29.9-15.8-68.6-6.5-76.5-21.9-2.4-4.7-2.4-12.7 2.6-26.4v-.2c2.4-7.6.6-16-.6-23.9-1.2-7.8-1.8-15 .9-20 3.5-6.7 8.5-9.1 14.8-11.3 10.3-3.7 11.8-3.4 19.6-9.9 5.5-5.7 9.5-12.9 14.3-18 5.1-5.5 10-8.1 17.7-6.9 8.1 1.2 15.1 6.8 21.9 16l19.6 35.6c9.5 19.9 43.1 48.4 41 68.9zm-1.4-25.9c-4.1-6.6-9.6-13.6-14.4-19.6 7.1 0 14.2-2.2 16.7-8.9 2.3-6.2 0-14.9-7.4-24.9-13.5-18.2-38.3-32.5-38.3-32.5-13.5-8.4-21.1-18.7-24.6-29.9s-3-23.3-.3-35.2c5.2-22.9 18.6-45.2 27.2-59.2 2.3-1.7.8 3.2-8.7 20.8-8.5 16.1-24.4 53.3-2.6 82.4.6-20.7 5.5-41.8 13.8-61.5 12-27.4 37.3-74.9 39.3-112.7 1.1.8 4.6 3.2 6.2 4.1 4.6 2.7 8.1 6.7 12.6 10.3 12.4 10 28.5 9.2 42.4 1.2 6.2-3.5 11.2-7.5 15.9-9 9.9-3.1 17.8-8.6 22.3-15 7.7 30.4 25.7 74.3 37.2 95.7 6.1 11.4 18.3 35.5 23.6 64.6 3.3-.1 7 .4 10.9 1.4 13.8-35.7-11.7-74.2-23.3-84.9-4.7-4.6-4.9-6.6-2.6-6.5 12.6 11.2 29.2 33.7 35.2 59 2.8 11.6 3.3 23.7.4 35.7 16.4 6.8 35.9 17.9 30.7 34.8-2.2-.1-3.2 0-4.2 0 3.2-10.1-3.9-17.6-22.8-26.1-19.6-8.6-36-8.6-38.3 12.5-12.1 4.2-18.3 14.7-21.4 27.3-2.8 11.2-3.6 24.7-4.4 39.9-.5 7.7-3.6 18-6.8 29-32.1 22.9-76.7 32.9-114.3 7.2zm257.4-11.5c-.9 16.8-41.2 19.9-63.2 46.5-13.2 15.7-29.4 24.4-43.6 25.5s-26.5-4.8-33.7-19.3c-4.7-11.1-2.4-23.1 1.1-36.3 3.7-14.2 9.2-28.8 9.9-40.6.8-15.2 1.7-28.5 4.2-38.7 2.6-10.3 6.6-17.2 13.7-21.1.3-.2.7-.3 1-.5.8 13.2 7.3 26.6 18.8 29.5 12.6 3.3 30.7-7.5 38.4-16.3 9-.3 15.7-.9 22.6 5.1 9.9 8.5 7.1 30.3 17.1 41.6 10.6 11.6 14 19.5 13.7 24.6zM173.3 148.7c2 1.9 4.7 4.5 8 7.1 6.6 5.2 15.8 10.6 27.3 10.6 11.6 0 22.5-5.9 31.8-10.8 4.9-2.6 10.9-7 14.8-10.4s5.9-6.3 3.1-6.6-2.6 2.6-6 5.1c-4.4 3.2-9.7 7.4-13.9 9.8-7.4 4.2-19.5 10.2-29.9 10.2s-18.7-4.8-24.9-9.7c-3.1-2.5-5.7-5-7.7-6.9-1.5-1.4-1.9-4.6-4.3-4.9-1.4-.1-1.8 3.7 1.7 6.5z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</Button>
|
2024-12-04 21:21:19 +01:00
|
|
|
<Button
|
|
|
|
className={`rounded-full w-12 h-12 lg:w-16 lg:h-16 ${
|
|
|
|
selectedButton === "windows"
|
|
|
|
? "bg-accent"
|
|
|
|
: "bg-primary-foreground/20"
|
|
|
|
}`}
|
|
|
|
variant={"ghost"}
|
|
|
|
onClick={() => handleButtonClick("windows")}
|
|
|
|
>
|
|
|
|
<span className="sr-only">Windows Server</span>
|
|
|
|
<svg
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
|
|
width={64}
|
|
|
|
height={64}
|
|
|
|
viewBox="0 0 4875 4875"
|
|
|
|
fill="none"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
fill="currentColor"
|
|
|
|
d="M0 0h2311v2310H0zm2564 0h2311v2310H2564zM0 2564h2311v2311H0zm2564 0h2311v2311H2564"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</Button>
|
2024-09-07 09:12:48 +02:00
|
|
|
<Button
|
|
|
|
className={`rounded-full w-12 h-12 lg:w-16 lg:h-16 ${
|
|
|
|
selectedButton === "docker"
|
|
|
|
? "bg-accent"
|
|
|
|
: "bg-primary-foreground/20"
|
|
|
|
}`}
|
|
|
|
variant={"ghost"}
|
|
|
|
onClick={() => handleButtonClick("docker")}
|
|
|
|
>
|
|
|
|
<span className="sr-only">Docker</span>
|
|
|
|
<svg
|
|
|
|
xmlns="http://www.w3.org/2000/svg"
|
2024-09-08 18:23:25 +02:00
|
|
|
width={64}
|
|
|
|
height={64}
|
2024-09-07 09:12:48 +02:00
|
|
|
viewBox="0 0 50 50"
|
|
|
|
fill="none"
|
|
|
|
>
|
|
|
|
<path
|
|
|
|
fill="currentColor"
|
|
|
|
d="M27.336 23.076h-5.164v-5.8h5.164v5.8Zm0-19.951h-5.164v5.928h5.164V3.125Zm6.11 14.14H28.28v5.801h5.164v-5.8Zm-12.212-7.04H16.07v5.869h5.164v-5.87Zm6.102 0h-5.164v5.869h5.164v-5.87ZM48.96 19.99c-1.125-.947-3.719-1.289-5.711-.82-.258-2.344-1.305-4.385-3.211-6.22l-1.094-.909-.726 1.367c-1.438 2.715-1.828 7.188-.29 10.137-.679.459-2.015 1.084-3.78 1.045H.186c-.68 4.96.454 11.406 3.438 15.83 2.898 4.287 7.242 6.465 12.922 6.465 12.297 0 21.398-7.08 25.656-19.942 1.672.04 5.281.01 7.133-4.414.117-.244.516-1.289.664-1.67l-1.04-.869ZM9.03 17.266H3.875v5.8h5.164v-5.8h-.008Zm6.102 0H9.969v5.8h5.164v-5.8Zm6.101 0H16.07v5.8h5.164v-5.8Zm-6.101-7.041H9.969v5.869h5.164v-5.87Z"
|
|
|
|
/>
|
|
|
|
</svg>
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="z-10">
|
|
|
|
<HeroCards />
|
|
|
|
</div>
|
|
|
|
<div className="shadow"></div>
|
|
|
|
</section>
|
|
|
|
);
|
2024-06-15 14:55:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
export default Hero;
|