Compare commits
3 Commits
proproposa
...
proposal-4
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5415741940 | ||
|
|
68797053ac | ||
|
|
c0e53ad652 |
41
.gitignore
vendored
Normal file
41
.gitignore
vendored
Normal file
@@ -0,0 +1,41 @@
|
||||
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
|
||||
|
||||
# dependencies
|
||||
/node_modules
|
||||
/.pnp
|
||||
.pnp.*
|
||||
.yarn/*
|
||||
!.yarn/patches
|
||||
!.yarn/plugins
|
||||
!.yarn/releases
|
||||
!.yarn/versions
|
||||
|
||||
# testing
|
||||
/coverage
|
||||
|
||||
# next.js
|
||||
/.next/
|
||||
/out/
|
||||
|
||||
# production
|
||||
/build
|
||||
|
||||
# misc
|
||||
.DS_Store
|
||||
*.pem
|
||||
|
||||
# debug
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
.pnpm-debug.log*
|
||||
|
||||
# env files (can opt-in for committing if needed)
|
||||
.env*
|
||||
|
||||
# vercel
|
||||
.vercel
|
||||
|
||||
# typescript
|
||||
*.tsbuildinfo
|
||||
next-env.d.ts
|
||||
39
README.md
39
README.md
@@ -1,39 +1,2 @@
|
||||
# P2P Poll App
|
||||
|
||||
There are various issues of Trust:
|
||||
The possiblity to generate lots of users that do a lot of things (at a rather low cost)
|
||||
The possibility to put out wrong data, maby not even contradicting but additional to existing data.
|
||||
The possibility to do all kinds of shenenigans like spam other users with some requests
|
||||
|
||||
Due to low programming knowledge, the starting point of this proposal was to mirror how normal groups of people solve issues of trust to then automate and possibly improve the process. There are already some systems out there like Trust flow or random walk. As far as i understand it, the Flexible Trust Web also already does something like this, also maby RWOT and GNUweb but i didn't read into them too much since i discovered them rather late and want to look for feedback anyway. After all, a system with a clearer consensus might be preferable to some.
|
||||
|
||||
If random new people should be able to use the system as equals to previous users, but the system never has real identities as an input, then there is no way to fully prevent the creation of new users to manipulate or sabotage the poll. But it can be assumed, that your friends are rather trustworthy and most likely also their friends and so on. And if someone makes huge ammounts or just one second account, they will probably only have the creator or maby some other people as friends, and even they might already be less socially connected than a normal user.
|
||||
So the social distance to another user should be evaluated to see, whether you should count their vote.
|
||||
This is evaluated for and by every user individually, based on the information they were sent. The ammount of contacts you won't count are displayed to you, such that you get a hint at how many people you are missing but also how many people are not counting you. This encourages people to try to prove others/vise versa and make social connections to officially tie the network closer together such that the voting system works and confirms itself. It would be great, if there was some chat attached to the poll. If people want to prove their (or others) trusworhiness within this system, they are then also encouraged to have productive discussions, probably about the matter of the poll.
|
||||
Everyone in a poll with you is a "contact" of yours.
|
||||
"users" can have "friends".
|
||||
You can also manually mark users as suspicious or trustworthy or normal again.
|
||||
The system for evaluating the trustworthyness of users is somehow a mix between the concepts "weighted path score" and "trust flow" with 5 steps.
|
||||
That means for 5 steps starting with you, all friends and trusted people of people looked at in this step get some trust from the people we look at: 0.8 * The trust of the looked at person (if trusted) + 0.8 * The trust of the looked at person / friends the looked at person has (if friend). Then the trust of the person that received trust may maximally be 100. The Trust you have to yourself is 100.
|
||||
You can also mark someone as trustworthy or untrustworthy. That is then also sent around to everyone if you want(should be the standard, but maby a user wants to just see how the trustworthyness will look like after the change).
|
||||
If you receive such an information, you can make the following calculations immidiately and after every assesment of everyones trustworthyness:
|
||||
If the accused is less trustworthy then the accusing person, decrease the accused trustworthyness to 0 and the accused friends and trustees trustworthyness by the trustworthyness of the accusing person.
|
||||
If the trustworhyness of the accusing person is less than the trustworthyness of the accused, then reduce the trustworthyness of the accusing person to 0 and the accusing persons friends and trustees by the trustworthyness of the accused * 0,2.
|
||||
If you mark someone as trustworthy:
|
||||
The Trust flowing to the trusted person from you will also be 0.8 of your trust.
|
||||
Maby this should also be the effect of beeing "friends" since "trust" might be something you could more intuitively casually deal out after a short chat. If that change were to occur, then the effect would have to be switched around.
|
||||
All contacts can maximally have the Trust 100.
|
||||
|
||||
|
||||
Future matters:
|
||||
If there can be any discrepancy of sent information, depending on what sender you trust most, you will mark one of the senders as untrustworthy and neglect all future information from this user. Since everything can be signed and such, that shouldnˋt be an issue tho, but if it was, the ammount of "useless" messages to already informed people might have to increase to validate received data.
|
||||
A system to showcase the social connections in a 2D - format would be neat.
|
||||
(most likely something like this exists already)
|
||||
Obviously the user would also have to see other context like the total of all votes (trusted or not)
|
||||
|
||||
Anonymous polls:
|
||||
A system of individually assigned trust poses a challenge for a system where you can decide not to trust some voters.
|
||||
If there is no other option some compromises might be makable, such as:
|
||||
-Your Friends can know what you voted for
|
||||
-The Person initiating a poll just decides on the validity of participants according to an own judgement of trust at the moment of poll-creation
|
||||
-A System with clear Consensus of who to trust
|
||||
Remove-Item -Recurse -Force node_modules, .next
|
||||
BIN
app/favicon.ico
Normal file
BIN
app/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 25 KiB |
26
app/globals.css
Normal file
26
app/globals.css
Normal file
@@ -0,0 +1,26 @@
|
||||
@import "tailwindcss";
|
||||
|
||||
:root {
|
||||
--background: #ffffff;
|
||||
--foreground: #171717;
|
||||
}
|
||||
|
||||
@theme inline {
|
||||
--color-background: var(--background);
|
||||
--color-foreground: var(--foreground);
|
||||
--font-sans: var(--font-geist-sans);
|
||||
--font-mono: var(--font-geist-mono);
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
:root {
|
||||
--background: #0a0a0a;
|
||||
--foreground: #ededed;
|
||||
}
|
||||
}
|
||||
|
||||
body {
|
||||
background: var(--background);
|
||||
color: var(--foreground);
|
||||
font-family: Arial, Helvetica, sans-serif;
|
||||
}
|
||||
25
app/layout.tsx
Normal file
25
app/layout.tsx
Normal file
@@ -0,0 +1,25 @@
|
||||
import type { Metadata } from "next";
|
||||
|
||||
import "./globals.css";
|
||||
|
||||
|
||||
|
||||
export const metadata: Metadata = {
|
||||
title: "p2p polling",
|
||||
description: "creating a p2p-polling app",
|
||||
};
|
||||
|
||||
export default function RootLayout({
|
||||
children,
|
||||
}: Readonly<{
|
||||
children: React.ReactNode;
|
||||
}>) {
|
||||
return (
|
||||
<html
|
||||
lang="en"
|
||||
className={` h-full antialiased`}
|
||||
>
|
||||
<body className="min-h-full flex flex-col">{children}</body>
|
||||
</html>
|
||||
);
|
||||
}
|
||||
47
app/page.tsx
Normal file
47
app/page.tsx
Normal file
@@ -0,0 +1,47 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
import usePeerManager from "../hooks/usePeerManager";
|
||||
import usePollManager from "../hooks/usePollManager";
|
||||
import PollCreation from "../components/PollCreation";
|
||||
import PollActive from "../components/PollActive";
|
||||
import PeersList from "../components/PeersList";
|
||||
|
||||
export default function Page() {
|
||||
const peerManager = usePeerManager();
|
||||
const pollManager = usePollManager(peerManager);
|
||||
|
||||
const [connectId, setConnectId] = useState("");
|
||||
|
||||
return (
|
||||
<div className="p-6 space-y-4">
|
||||
<h1 className="text-2xl font-bold">P2P Poll App</h1>
|
||||
|
||||
{/* Connect */}
|
||||
<div className="flex gap-2">
|
||||
<input
|
||||
className="border p-2"
|
||||
placeholder="Peer ID"
|
||||
value={connectId}
|
||||
onChange={(e) => setConnectId(e.target.value)}
|
||||
/>
|
||||
<button
|
||||
className="bg-blue-500 text-white px-4"
|
||||
onClick={() => peerManager.connectToPeer(connectId)}
|
||||
>
|
||||
Connect
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<p>Your ID: {peerManager.peerId}</p>
|
||||
|
||||
{pollManager.poll ? (
|
||||
<PollActive pollManager={pollManager} peerId={peerManager.peerId} />
|
||||
) : (
|
||||
<PollCreation pollManager={pollManager} />
|
||||
)}
|
||||
|
||||
<PeersList peers={peerManager.peers} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
0
components/Modal.tsx
Normal file
0
components/Modal.tsx
Normal file
22
components/Notification.tsx
Normal file
22
components/Notification.tsx
Normal file
@@ -0,0 +1,22 @@
|
||||
"use client";
|
||||
|
||||
type NotificationType = "info" | "success" | "error";
|
||||
|
||||
interface NotificationProps {
|
||||
message: string;
|
||||
type?: NotificationType;
|
||||
}
|
||||
|
||||
export default function Notification({ message, type = "info" }: NotificationProps) {
|
||||
const colors: Record<NotificationType, string> = {
|
||||
info: "bg-blue-100 text-blue-700",
|
||||
success: "bg-green-100 text-green-700",
|
||||
error: "bg-red-100 text-red-700",
|
||||
};
|
||||
|
||||
return (
|
||||
<div className={`px-4 py-2 rounded shadow ${colors[type]}`}>
|
||||
{message}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
14
components/PeersList.tsx
Normal file
14
components/PeersList.tsx
Normal file
@@ -0,0 +1,14 @@
|
||||
"use client";
|
||||
|
||||
export default function PeersList({ peers }: { peers: string[] }) {
|
||||
return (
|
||||
<div>
|
||||
<h3>Peers</h3>
|
||||
{peers.length === 0 ? (
|
||||
<p>No peers</p>
|
||||
) : (
|
||||
peers.map((p) => <div key={p}>{p}</div>)
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
22
components/PollActive.tsx
Normal file
22
components/PollActive.tsx
Normal file
@@ -0,0 +1,22 @@
|
||||
"use client";
|
||||
|
||||
import PollOption from "./PollOption";
|
||||
|
||||
export default function PollActive({ pollManager, peerId }: any) {
|
||||
const poll = pollManager.poll;
|
||||
|
||||
return (
|
||||
<div>
|
||||
<h2 className="text-xl font-bold">{poll.question}</h2>
|
||||
|
||||
{poll.options.map((opt: any) => (
|
||||
<PollOption
|
||||
key={opt.id}
|
||||
option={opt}
|
||||
pollManager={pollManager}
|
||||
peerId={peerId}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
40
components/PollCreation.tsx
Normal file
40
components/PollCreation.tsx
Normal file
@@ -0,0 +1,40 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
|
||||
export default function PollCreation({ pollManager }: any) {
|
||||
const [question, setQuestion] = useState("");
|
||||
const [options, setOptions] = useState<string[]>(["", ""]);
|
||||
|
||||
return (
|
||||
<div>
|
||||
<input
|
||||
className="border p-2 w-full"
|
||||
placeholder="Question"
|
||||
value={question}
|
||||
onChange={(e) => setQuestion(e.target.value)}
|
||||
/>
|
||||
|
||||
{options.map((opt, i) => (
|
||||
<input
|
||||
key={i}
|
||||
className="border p-2 w-full mt-2"
|
||||
placeholder={`Option ${i + 1}`}
|
||||
value={opt}
|
||||
onChange={(e) => {
|
||||
const newOpts = [...options];
|
||||
newOpts[i] = e.target.value;
|
||||
setOptions(newOpts);
|
||||
}}
|
||||
/>
|
||||
))}
|
||||
|
||||
<button
|
||||
className="bg-green-500 text-white px-4 mt-2"
|
||||
onClick={() => pollManager.createPoll(question, options)}
|
||||
>
|
||||
Create Poll
|
||||
</button>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
12
components/PollOption.tsx
Normal file
12
components/PollOption.tsx
Normal file
@@ -0,0 +1,12 @@
|
||||
"use client";
|
||||
|
||||
export default function PollOption({ option, pollManager, peerId }: any) {
|
||||
return (
|
||||
<div
|
||||
className="border p-2 mt-2 cursor-pointer"
|
||||
onClick={() => pollManager.vote(option.id, peerId)}
|
||||
>
|
||||
{option.text} - {option.votes} votes
|
||||
</div>
|
||||
);
|
||||
}
|
||||
18
eslint.config.mjs
Normal file
18
eslint.config.mjs
Normal file
@@ -0,0 +1,18 @@
|
||||
import { defineConfig, globalIgnores } from "eslint/config";
|
||||
import nextVitals from "eslint-config-next/core-web-vitals";
|
||||
import nextTs from "eslint-config-next/typescript";
|
||||
|
||||
const eslintConfig = defineConfig([
|
||||
...nextVitals,
|
||||
...nextTs,
|
||||
// Override default ignores of eslint-config-next.
|
||||
globalIgnores([
|
||||
// Default ignores of eslint-config-next:
|
||||
".next/**",
|
||||
"out/**",
|
||||
"build/**",
|
||||
"next-env.d.ts",
|
||||
]),
|
||||
]);
|
||||
|
||||
export default eslintConfig;
|
||||
58
hooks/usePeerManager.ts
Normal file
58
hooks/usePeerManager.ts
Normal file
@@ -0,0 +1,58 @@
|
||||
"use client";
|
||||
|
||||
import { useEffect, useRef, useState } from "react";
|
||||
import Peer from "peerjs";
|
||||
|
||||
export default function usePeerManager() {
|
||||
const [peerId, setPeerId] = useState<string | null>(null);
|
||||
const [peers, setPeers] = useState<string[]>([]);
|
||||
const peerRef = useRef<Peer | null>(null);
|
||||
const connectionsRef = useRef<Map<string, any>>(new Map());
|
||||
|
||||
useEffect(() => {
|
||||
const peer = new Peer();
|
||||
peerRef.current = peer;
|
||||
|
||||
peer.on("open", (id) => {
|
||||
setPeerId(id);
|
||||
});
|
||||
|
||||
peer.on("connection", (conn) => {
|
||||
conn.on("open", () => {
|
||||
connectionsRef.current.set(conn.peer, conn);
|
||||
setPeers(Array.from(connectionsRef.current.keys()));
|
||||
});
|
||||
|
||||
conn.on("data", (data) => {
|
||||
console.log("Received:", data);
|
||||
});
|
||||
|
||||
conn.on("close", () => {
|
||||
connectionsRef.current.delete(conn.peer);
|
||||
setPeers(Array.from(connectionsRef.current.keys()));
|
||||
});
|
||||
});
|
||||
|
||||
return () => {
|
||||
peer.destroy();
|
||||
};
|
||||
}, []);
|
||||
|
||||
const connectToPeer = (id: string) => {
|
||||
if (!peerRef.current) return;
|
||||
const conn = peerRef.current.connect(id);
|
||||
|
||||
conn.on("open", () => {
|
||||
connectionsRef.current.set(conn.peer, conn);
|
||||
setPeers(Array.from(connectionsRef.current.keys()));
|
||||
});
|
||||
};
|
||||
|
||||
const broadcast = (data: any) => {
|
||||
connectionsRef.current.forEach((conn) => {
|
||||
if (conn.open) conn.send(data);
|
||||
});
|
||||
};
|
||||
|
||||
return { peerId, peers, connectToPeer, broadcast };
|
||||
}
|
||||
68
hooks/usePollManager.ts
Normal file
68
hooks/usePollManager.ts
Normal file
@@ -0,0 +1,68 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
|
||||
type Option = {
|
||||
id: string;
|
||||
text: string;
|
||||
votes: number;
|
||||
voters: string[];
|
||||
};
|
||||
|
||||
type Poll = {
|
||||
id: string;
|
||||
question: string;
|
||||
options: Option[];
|
||||
};
|
||||
|
||||
export default function usePollManager(peerManager: any) {
|
||||
const [poll, setPoll] = useState<Poll | null>(null);
|
||||
const [myVote, setMyVote] = useState<string | null>(null);
|
||||
|
||||
const createPoll = (question: string, options: string[]) => {
|
||||
const newPoll: Poll = {
|
||||
id: Date.now().toString(),
|
||||
question,
|
||||
options: options.map((opt, i) => ({
|
||||
id: `opt-${i}`,
|
||||
text: opt,
|
||||
votes: 0,
|
||||
voters: [],
|
||||
})),
|
||||
};
|
||||
|
||||
setPoll(newPoll);
|
||||
peerManager.broadcast({ type: "poll", poll: newPoll });
|
||||
};
|
||||
|
||||
const vote = (optionId: string, peerId: string) => {
|
||||
if (!poll) return;
|
||||
|
||||
const updated = { ...poll };
|
||||
|
||||
updated.options.forEach((opt) => {
|
||||
const index = opt.voters.indexOf(peerId);
|
||||
if (index !== -1) {
|
||||
opt.voters.splice(index, 1);
|
||||
opt.votes--;
|
||||
}
|
||||
});
|
||||
|
||||
const option = updated.options.find((o) => o.id === optionId);
|
||||
if (!option) return;
|
||||
|
||||
option.votes++;
|
||||
option.voters.push(peerId);
|
||||
|
||||
setPoll(updated);
|
||||
setMyVote(optionId);
|
||||
|
||||
peerManager.broadcast({
|
||||
type: "vote",
|
||||
optionId,
|
||||
voterId: peerId,
|
||||
});
|
||||
};
|
||||
|
||||
return { poll, createPoll, vote, myVote };
|
||||
}
|
||||
9
next.config.ts
Normal file
9
next.config.ts
Normal file
@@ -0,0 +1,9 @@
|
||||
import type { NextConfig } from "next";
|
||||
|
||||
const nextConfig: NextConfig = {
|
||||
experimental: {
|
||||
turbopackFileSystemCacheForDev: true,
|
||||
}
|
||||
};
|
||||
|
||||
export default nextConfig;
|
||||
6660
package-lock.json
generated
Normal file
6660
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
27
package.json
Normal file
27
package.json
Normal file
@@ -0,0 +1,27 @@
|
||||
{
|
||||
"name": "p2p-polling",
|
||||
"version": "0.1.0",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"dev": "next dev",
|
||||
"build": "next build",
|
||||
"start": "next start",
|
||||
"lint": "eslint"
|
||||
},
|
||||
"dependencies": {
|
||||
"next": "16.2.0",
|
||||
"peerjs": "^1.5.5",
|
||||
"react": "19.2.4",
|
||||
"react-dom": "19.2.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@tailwindcss/postcss": "^4",
|
||||
"@types/node": "^20",
|
||||
"@types/react": "^19",
|
||||
"@types/react-dom": "^19",
|
||||
"eslint": "^9",
|
||||
"eslint-config-next": "16.2.0",
|
||||
"tailwindcss": "^4",
|
||||
"typescript": "^5"
|
||||
}
|
||||
}
|
||||
7
postcss.config.mjs
Normal file
7
postcss.config.mjs
Normal file
@@ -0,0 +1,7 @@
|
||||
const config = {
|
||||
plugins: {
|
||||
"@tailwindcss/postcss": {},
|
||||
},
|
||||
};
|
||||
|
||||
export default config;
|
||||
1
public/file.svg
Normal file
1
public/file.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg fill="none" viewBox="0 0 16 16" xmlns="http://www.w3.org/2000/svg"><path d="M14.5 13.5V5.41a1 1 0 0 0-.3-.7L9.8.29A1 1 0 0 0 9.08 0H1.5v13.5A2.5 2.5 0 0 0 4 16h8a2.5 2.5 0 0 0 2.5-2.5m-1.5 0v-7H8v-5H3v12a1 1 0 0 0 1 1h8a1 1 0 0 0 1-1M9.5 5V2.12L12.38 5zM5.13 5h-.62v1.25h2.12V5zm-.62 3h7.12v1.25H4.5zm.62 3h-.62v1.25h7.12V11z" clip-rule="evenodd" fill="#666" fill-rule="evenodd"/></svg>
|
||||
|
After Width: | Height: | Size: 391 B |
1
public/globe.svg
Normal file
1
public/globe.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg fill="none" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"><g clip-path="url(#a)"><path fill-rule="evenodd" clip-rule="evenodd" d="M10.27 14.1a6.5 6.5 0 0 0 3.67-3.45q-1.24.21-2.7.34-.31 1.83-.97 3.1M8 16A8 8 0 1 0 8 0a8 8 0 0 0 0 16m.48-1.52a7 7 0 0 1-.96 0H7.5a4 4 0 0 1-.84-1.32q-.38-.89-.63-2.08a40 40 0 0 0 3.92 0q-.25 1.2-.63 2.08a4 4 0 0 1-.84 1.31zm2.94-4.76q1.66-.15 2.95-.43a7 7 0 0 0 0-2.58q-1.3-.27-2.95-.43a18 18 0 0 1 0 3.44m-1.27-3.54a17 17 0 0 1 0 3.64 39 39 0 0 1-4.3 0 17 17 0 0 1 0-3.64 39 39 0 0 1 4.3 0m1.1-1.17q1.45.13 2.69.34a6.5 6.5 0 0 0-3.67-3.44q.65 1.26.98 3.1M8.48 1.5l.01.02q.41.37.84 1.31.38.89.63 2.08a40 40 0 0 0-3.92 0q.25-1.2.63-2.08a4 4 0 0 1 .85-1.32 7 7 0 0 1 .96 0m-2.75.4a6.5 6.5 0 0 0-3.67 3.44 29 29 0 0 1 2.7-.34q.31-1.83.97-3.1M4.58 6.28q-1.66.16-2.95.43a7 7 0 0 0 0 2.58q1.3.27 2.95.43a18 18 0 0 1 0-3.44m.17 4.71q-1.45-.12-2.69-.34a6.5 6.5 0 0 0 3.67 3.44q-.65-1.27-.98-3.1" fill="#666"/></g><defs><clipPath id="a"><path fill="#fff" d="M0 0h16v16H0z"/></clipPath></defs></svg>
|
||||
|
After Width: | Height: | Size: 1.0 KiB |
1
public/next.svg
Normal file
1
public/next.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 394 80"><path fill="#000" d="M262 0h68.5v12.7h-27.2v66.6h-13.6V12.7H262V0ZM149 0v12.7H94v20.4h44.3v12.6H94v21h55v12.6H80.5V0h68.7zm34.3 0h-17.8l63.8 79.4h17.9l-32-39.7 32-39.6h-17.9l-23 28.6-23-28.6zm18.3 56.7-9-11-27.1 33.7h17.8l18.3-22.7z"/><path fill="#000" d="M81 79.3 17 0H0v79.3h13.6V17l50.2 62.3H81Zm252.6-.4c-1 0-1.8-.4-2.5-1s-1.1-1.6-1.1-2.6.3-1.8 1-2.5 1.6-1 2.6-1 1.8.3 2.5 1a3.4 3.4 0 0 1 .6 4.3 3.7 3.7 0 0 1-3 1.8zm23.2-33.5h6v23.3c0 2.1-.4 4-1.3 5.5a9.1 9.1 0 0 1-3.8 3.5c-1.6.8-3.5 1.3-5.7 1.3-2 0-3.7-.4-5.3-1s-2.8-1.8-3.7-3.2c-.9-1.3-1.4-3-1.4-5h6c.1.8.3 1.6.7 2.2s1 1.2 1.6 1.5c.7.4 1.5.5 2.4.5 1 0 1.8-.2 2.4-.6a4 4 0 0 0 1.6-1.8c.3-.8.5-1.8.5-3V45.5zm30.9 9.1a4.4 4.4 0 0 0-2-3.3 7.5 7.5 0 0 0-4.3-1.1c-1.3 0-2.4.2-3.3.5-.9.4-1.6 1-2 1.6a3.5 3.5 0 0 0-.3 4c.3.5.7.9 1.3 1.2l1.8 1 2 .5 3.2.8c1.3.3 2.5.7 3.7 1.2a13 13 0 0 1 3.2 1.8 8.1 8.1 0 0 1 3 6.5c0 2-.5 3.7-1.5 5.1a10 10 0 0 1-4.4 3.5c-1.8.8-4.1 1.2-6.8 1.2-2.6 0-4.9-.4-6.8-1.2-2-.8-3.4-2-4.5-3.5a10 10 0 0 1-1.7-5.6h6a5 5 0 0 0 3.5 4.6c1 .4 2.2.6 3.4.6 1.3 0 2.5-.2 3.5-.6 1-.4 1.8-1 2.4-1.7a4 4 0 0 0 .8-2.4c0-.9-.2-1.6-.7-2.2a11 11 0 0 0-2.1-1.4l-3.2-1-3.8-1c-2.8-.7-5-1.7-6.6-3.2a7.2 7.2 0 0 1-2.4-5.7 8 8 0 0 1 1.7-5 10 10 0 0 1 4.3-3.5c2-.8 4-1.2 6.4-1.2 2.3 0 4.4.4 6.2 1.2 1.8.8 3.2 2 4.3 3.4 1 1.4 1.5 3 1.5 5h-5.8z"/></svg>
|
||||
|
After Width: | Height: | Size: 1.3 KiB |
1
public/vercel.svg
Normal file
1
public/vercel.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg fill="none" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1155 1000"><path d="m577.3 0 577.4 1000H0z" fill="#fff"/></svg>
|
||||
|
After Width: | Height: | Size: 128 B |
1
public/window.svg
Normal file
1
public/window.svg
Normal file
@@ -0,0 +1 @@
|
||||
<svg fill="none" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"><path fill-rule="evenodd" clip-rule="evenodd" d="M1.5 2.5h13v10a1 1 0 0 1-1 1h-11a1 1 0 0 1-1-1zM0 1h16v11.5a2.5 2.5 0 0 1-2.5 2.5h-11A2.5 2.5 0 0 1 0 12.5zm3.75 4.5a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5M7 4.75a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0m1.75.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5" fill="#666"/></svg>
|
||||
|
After Width: | Height: | Size: 385 B |
34
tsconfig.json
Normal file
34
tsconfig.json
Normal file
@@ -0,0 +1,34 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"target": "ES2017",
|
||||
"lib": ["dom", "dom.iterable", "esnext"],
|
||||
"allowJs": true,
|
||||
"skipLibCheck": true,
|
||||
"strict": true,
|
||||
"noEmit": true,
|
||||
"esModuleInterop": true,
|
||||
"module": "esnext",
|
||||
"moduleResolution": "bundler",
|
||||
"resolveJsonModule": true,
|
||||
"isolatedModules": true,
|
||||
"jsx": "react-jsx",
|
||||
"incremental": true,
|
||||
"plugins": [
|
||||
{
|
||||
"name": "next"
|
||||
}
|
||||
],
|
||||
"paths": {
|
||||
"@/*": ["./*"]
|
||||
}
|
||||
},
|
||||
"include": [
|
||||
"next-env.d.ts",
|
||||
"**/*.ts",
|
||||
"**/*.tsx",
|
||||
".next/types/**/*.ts",
|
||||
".next/dev/types/**/*.ts",
|
||||
"**/*.mts"
|
||||
],
|
||||
"exclude": ["node_modules"]
|
||||
}
|
||||
Reference in New Issue
Block a user