Skip to content

Commit 49ee36a

Browse files
authored
feat: Js-to-Ts Converter Added
Js-to-Ts Converter Added
2 parents 39a7ef6 + a171099 commit 49ee36a

File tree

4 files changed

+297
-5
lines changed

4 files changed

+297
-5
lines changed

package-lock.json

+65-5
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+3
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
"@mui/material": "^5.15.19",
2525
"@radix-ui/react-select": "^2.1.1",
2626
"@react-pdf/renderer": "^3.4.4",
27+
"@types/typescript": "^2.0.0",
2728
"@vercel/analytics": "^1.3.1",
2829
"asciidoctor": "^3.0.4",
2930
"autoprefixer": "10.4.20",
@@ -63,6 +64,8 @@
6364
"tailwind-merge": "^2.5.2",
6465
"tailwindcss": "3.4.4",
6566
"tailwindcss-animate": "^1.0.7",
67+
"typeify": "^0.1.1",
68+
"typescript": "^5.6.2",
6669
"uuid": "^10.0.0"
6770
},
6871
"devDependencies": {

src/app/Js-to-Ts-Converter/page.jsx

+223
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,223 @@
1+
"use client";
2+
import React, { useState } from "react";
3+
import { NavBar } from "@/components/navbar";
4+
5+
const Converter = () => {
6+
const [inputCode, setInputCode] = useState("");
7+
const [outputCode, setOutputCode] = useState("");
8+
const [isDarkMode, setIsDarkMode] = useState(false);
9+
const [copySuccess, setCopySuccess] = useState(false);
10+
11+
const handleCopy = () => {
12+
if (outputCode) {
13+
navigator.clipboard.writeText(outputCode);
14+
setCopySuccess(true);
15+
setTimeout(() => setCopySuccess(false), 3000);
16+
}
17+
};
18+
19+
const toggleTheme = () => {
20+
setIsDarkMode(!isDarkMode);
21+
};
22+
23+
const handleConvert = () => {
24+
const convertedCode = convertJsToTs(inputCode);
25+
setOutputCode(convertedCode);
26+
};
27+
28+
const convertJsToTs = (jsCode) => {
29+
let tsCode = jsCode
30+
// Convert object declarations with inferred types for common data types
31+
.replace(/const\s+(\w+)\s*=\s*{([^}]+)}/g, (match, varName, props) => {
32+
const propLines = props
33+
.split(",")
34+
.map((prop) => {
35+
const [key, value] = prop.split(":").map((s) => s.trim());
36+
37+
// Infer types based on the value format
38+
const inferredType = inferType(value);
39+
40+
// Ensure both key and value are present to avoid extra ': any'
41+
return key && value ? `${key}: ${inferredType}` : "";
42+
})
43+
.filter(Boolean)
44+
.join(", "); // Filter out any empty properties
45+
46+
return `const ${varName}: { ${propLines} } = {${props}};`;
47+
})
48+
// Convert array declarations with inferred types
49+
.replace(
50+
/const\s+(\w+)\s*=\s*\[([^;]+)\]/g,
51+
(match, varName, elements) => {
52+
const inferredType = inferArrayType(elements);
53+
return `const ${varName}: ${inferredType}[] = [${elements}]`;
54+
},
55+
)
56+
// Convert let, var variables with inferred types
57+
.replace(
58+
/(const|let|var)\s+(\w+)\s*=\s*([^;]+)/g,
59+
(match, declarationType, varName, value) => {
60+
const inferredType = inferType(value);
61+
return `${declarationType} ${varName}: ${inferredType} = ${value}`;
62+
},
63+
)
64+
// Convert function declarations with inferred parameter types
65+
.replace(/function\s+(\w+)\s*\(([^)]*)\)\s*{/g, (match, name, params) => {
66+
const typedParams = params
67+
.split(",")
68+
.map((param) => {
69+
const [paramName] = param.trim().split(" ");
70+
return `${paramName}: any`; // Default to `any` for parameters
71+
})
72+
.join(", ");
73+
74+
return `function ${name}(${typedParams}): void {`;
75+
})
76+
// Clean up extra semicolons
77+
.replace(/;\s*;/g, ";") // Remove double semicolons
78+
.replace(/;\s*}/g, "}"); // Remove semicolon before closing braces
79+
80+
return tsCode;
81+
};
82+
83+
// Type inference function for simple values
84+
const inferType = (value) => {
85+
if (!isNaN(Number(value))) return "number";
86+
if (/^".*"$/.test(value) || /^'.*'$/.test(value)) return "string";
87+
if (/true|false/.test(value)) return "boolean";
88+
if (/^\[.*\]$/.test(value)) return "any[]"; // Default for arrays
89+
if (/^\{.*\}$/.test(value)) return "object"; // Default for objects
90+
return "any"; // Fallback for other types
91+
};
92+
93+
// Type inference for array elements
94+
const inferArrayType = (elements) => {
95+
const elementArray = elements.split(",").map((el) => el.trim());
96+
const inferredType = elementArray.every((el) => !isNaN(Number(el)))
97+
? "number"
98+
: elementArray.every((el) => /^".*"$/.test(el))
99+
? "string"
100+
: "any"; // Fallback for mixed or unknown types
101+
102+
return inferredType;
103+
};
104+
105+
return (
106+
<div
107+
className={`${
108+
isDarkMode ? "bg-gray-900 text-gray-400" : "bg-gray-100 text-gray-500"
109+
} min-h-screen w-full pb-2`}
110+
>
111+
<NavBar
112+
title={"JS to TS Converter"}
113+
isDarkMode={isDarkMode}
114+
toggleTheme={toggleTheme}
115+
/>
116+
<h1 className="relative z-10 font-sans text-5xl font-bold text-center text-transparent md:text-7xl bg-clip-text bg-gradient-to-b from-neutral-200 to-neutral-600 mb-7 mt-4">
117+
JS to TS Converter
118+
</h1>
119+
120+
<div className="flex flex-col md:flex-row gap-2">
121+
<textarea
122+
className={`
123+
w-full
124+
ml-7
125+
mr-4
126+
p-2
127+
rounded-lg
128+
border border-gray-300
129+
focus:ring-2 focus:ring-blue-500
130+
focus:outline-none
131+
transition duration-200 ease-in-out
132+
placeholder-gray-400
133+
shadow-sm
134+
hover:shadow-md
135+
resize-none
136+
${
137+
isDarkMode
138+
? "bg-gray-800 text-gray-200"
139+
: "bg-gray-200 text-gray-500"
140+
} `}
141+
rows="10"
142+
placeholder="Paste JavaScript code here..."
143+
value={inputCode}
144+
onChange={(e) => setInputCode(e.target.value)}
145+
></textarea>
146+
147+
<div className="relative w-full mr-7">
148+
<textarea
149+
className={`
150+
w-full
151+
p-2
152+
rounded-lg
153+
border border-gray-300
154+
focus:ring-2 focus:ring-blue-500
155+
focus:outline-none
156+
transition duration-200 ease-in-out
157+
placeholder-gray-400
158+
shadow-sm
159+
hover:shadow-md
160+
resize-none
161+
${
162+
isDarkMode
163+
? "bg-gray-800 text-gray-200"
164+
: "bg-gray-200 text-gray-500"
165+
} `}
166+
rows="10"
167+
readOnly
168+
value={outputCode}
169+
></textarea>
170+
171+
{copySuccess ? (
172+
<button
173+
className="absolute top-2 right-2 bg-blue-400 text-white rounded px-4 py-2 flex items-center"
174+
onClick={handleCopy}
175+
>
176+
<svg
177+
xmlns="http://www.w3.org/2000/svg"
178+
viewBox="0 0 24 24"
179+
fill="white"
180+
className="w-6 h-6 mr-1"
181+
>
182+
<path
183+
fillRule="evenodd"
184+
d="M16.707 7.707a1 1 0 00-1.414-1.414L9 12.586l-2.293-2.293a1 1 0 10-1.414 1.414l3 3a1 1 0 001.414 0l7-7z"
185+
clipRule="evenodd"
186+
/>
187+
</svg>
188+
</button>
189+
) : (
190+
<button
191+
className="absolute top-2 right-2 bg-gray-200 hover:bg-gray-300 rounded p-2 focus:outline-none"
192+
onClick={handleCopy}
193+
>
194+
<svg
195+
xmlns="http://www.w3.org/2000/svg"
196+
fill="none"
197+
viewBox="0 0 24 24"
198+
stroke="currentColor"
199+
className="w-6 h-6 text-gray-700"
200+
>
201+
<path
202+
strokeLinecap="round"
203+
strokeLinejoin="round"
204+
strokeWidth={2}
205+
d="M8 16h8m0 0l-3 3m3-3l-3-3m5-2V6a2 2 0 00-2-2H7a2 2 0 00-2 2v6a2 2 0 002 2h3"
206+
/>
207+
</svg>
208+
</button>
209+
)}
210+
</div>
211+
</div>
212+
213+
<button
214+
className="bg-gradient-to-r from-blue-500 to-indigo-600 text-white px-6 py-3 mt-6 rounded-lg shadow-lg hover:from-blue-600 hover:to-indigo-700 hover:shadow-xl transform hover:scale-105 transition duration-300 ease-in-out focus:outline-none focus:ring-4 focus:ring-blue-300 ml-7"
215+
onClick={handleConvert}
216+
>
217+
Convert to TypeScript
218+
</button>
219+
</div>
220+
);
221+
};
222+
223+
export default Converter;

src/db/tools.json

+6
Original file line numberDiff line numberDiff line change
@@ -142,5 +142,11 @@
142142
"name": "Opensource Collection",
143143
"link": "/opensource",
144144
"ctg": "other"
145+
},
146+
{
147+
"id": 25,
148+
"name": "Js To Ts Converter",
149+
"link": "/Js-to-Ts-Converter",
150+
"ctg": "other"
145151
}
146152
]

0 commit comments

Comments
 (0)