Compare commits
5 Commits
Author | SHA1 | Date |
---|---|---|
Tuan-Dat Tran | bec4c608f1 | |
Tuan-Dat Tran | f76a7a8c4c | |
Tuan-Dat Tran | 0ff5419c23 | |
Tuan-Dat Tran | b7a47ca903 | |
Tuan-Dat Tran | b3eb962024 |
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "athome"
|
name = "athome"
|
||||||
version = "0.2.1"
|
version = "0.2.2"
|
||||||
authors = ["Tuan-Dat Tran <tuan-dat.tran@tudattr.dev>"]
|
authors = ["Tuan-Dat Tran <tuan-dat.tran@tudattr.dev>"]
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
|
@ -8,7 +8,6 @@ edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
serde = { version = "1.0.197", features = ["derive"] }
|
serde = { version = "1.0.197", features = ["derive"] }
|
||||||
|
|
||||||
dioxus = { version = "0.5", features = ["fullstack", "router"] }
|
dioxus = { version = "0.5", features = ["fullstack", "router"] }
|
||||||
|
|
||||||
# Debug
|
# Debug
|
||||||
|
|
|
@ -11,7 +11,8 @@ COPY ./Cargo.toml ./Cargo.toml
|
||||||
COPY ./input.css ./input.css
|
COPY ./input.css ./input.css
|
||||||
COPY ./Dioxus.toml ./Dioxus.toml
|
COPY ./Dioxus.toml ./Dioxus.toml
|
||||||
COPY ./tailwind.config.js ./tailwind.config.js
|
COPY ./tailwind.config.js ./tailwind.config.js
|
||||||
RUN npx tailwindcss -i ./input.css -o ./assets/tailwind.css && dx build --platform fullstack --release
|
RUN npx tailwindcss -i ./input.css -o ./assets/tailwind.css
|
||||||
|
RUN dx build --platform fullstack --release
|
||||||
|
|
||||||
FROM dioxus AS runner
|
FROM dioxus AS runner
|
||||||
WORKDIR /app/
|
WORKDIR /app/
|
||||||
|
|
10
README.md
|
@ -2,6 +2,16 @@
|
||||||
|
|
||||||
My personal website.
|
My personal website.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npx tailwindcss -i ./input.css -o ./assets/tailwind.css --watch
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh
|
||||||
|
dx serve --platform fullstack
|
||||||
|
```
|
||||||
|
|
||||||
## Screenshot
|
## Screenshot
|
||||||
|
|
||||||
[[./resources/screenshot.webp]]
|
[[./resources/screenshot.webp]]
|
||||||
|
|
After Width: | Height: | Size: 27 KiB |
After Width: | Height: | Size: 110 KiB |
After Width: | Height: | Size: 24 KiB |
After Width: | Height: | Size: 742 B |
After Width: | Height: | Size: 1.9 KiB |
Before Width: | Height: | Size: 130 KiB After Width: | Height: | Size: 15 KiB |
|
@ -0,0 +1 @@
|
||||||
|
<svg version="1.1" id="main_outline" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" style="enable-background:new 0 0 640 640;" xml:space="preserve" viewBox="5.67 143.05 628.65 387.55"> <g> <path id="teabag" style="fill:#FFFFFF" d="M395.9,484.2l-126.9-61c-12.5-6-17.9-21.2-11.8-33.8l61-126.9c6-12.5,21.2-17.9,33.8-11.8 c17.2,8.3,27.1,13,27.1,13l-0.1-109.2l16.7-0.1l0.1,117.1c0,0,57.4,24.2,83.1,40.1c3.7,2.3,10.2,6.8,12.9,14.4 c2.1,6.1,2,13.1-1,19.3l-61,126.9C423.6,484.9,408.4,490.3,395.9,484.2z"></path> <g> <g> <path style="fill:#609926" d="M622.7,149.8c-4.1-4.1-9.6-4-9.6-4s-117.2,6.6-177.9,8c-13.3,0.3-26.5,0.6-39.6,0.7c0,39.1,0,78.2,0,117.2 c-5.5-2.6-11.1-5.3-16.6-7.9c0-36.4-0.1-109.2-0.1-109.2c-29,0.4-89.2-2.2-89.2-2.2s-141.4-7.1-156.8-8.5 c-9.8-0.6-22.5-2.1-39,1.5c-8.7,1.8-33.5,7.4-53.8,26.9C-4.9,212.4,6.6,276.2,8,285.8c1.7,11.7,6.9,44.2,31.7,72.5 c45.8,56.1,144.4,54.8,144.4,54.8s12.1,28.9,30.6,55.5c25,33.1,50.7,58.9,75.7,62c63,0,188.9-0.1,188.9-0.1s12,0.1,28.3-10.3 c14-8.5,26.5-23.4,26.5-23.4s12.9-13.8,30.9-45.3c5.5-9.7,10.1-19.1,14.1-28c0,0,55.2-117.1,55.2-231.1 C633.2,157.9,624.7,151.8,622.7,149.8z M125.6,353.9c-25.9-8.5-36.9-18.7-36.9-18.7S69.6,321.8,60,295.4 c-16.5-44.2-1.4-71.2-1.4-71.2s8.4-22.5,38.5-30c13.8-3.7,31-3.1,31-3.1s7.1,59.4,15.7,94.2c7.2,29.2,24.8,77.7,24.8,77.7 S142.5,359.9,125.6,353.9z M425.9,461.5c0,0-6.1,14.5-19.6,15.4c-5.8,0.4-10.3-1.2-10.3-1.2s-0.3-0.1-5.3-2.1l-112.9-55 c0,0-10.9-5.7-12.8-15.6c-2.2-8.1,2.7-18.1,2.7-18.1L322,273c0,0,4.8-9.7,12.2-13c0.6-0.3,2.3-1,4.5-1.5c8.1-2.1,18,2.8,18,2.8 l110.7,53.7c0,0,12.6,5.7,15.3,16.2c1.9,7.4-0.5,14-1.8,17.2C474.6,363.8,425.9,461.5,425.9,461.5z"></path> <path style="fill:#609926" d="M326.8,380.1c-8.2,0.1-15.4,5.8-17.3,13.8c-1.9,8,2,16.3,9.1,20c7.7,4,17.5,1.8,22.7-5.4 c5.1-7.1,4.3-16.9-1.8-23.1l24-49.1c1.5,0.1,3.7,0.2,6.2-0.5c4.1-0.9,7.1-3.6,7.1-3.6c4.2,1.8,8.6,3.8,13.2,6.1 c4.8,2.4,9.3,4.9,13.4,7.3c0.9,0.5,1.8,1.1,2.8,1.9c1.6,1.3,3.4,3.1,4.7,5.5c1.9,5.5-1.9,14.9-1.9,14.9 c-2.3,7.6-18.4,40.6-18.4,40.6c-8.1-0.2-15.3,5-17.7,12.5c-2.6,8.1,1.1,17.3,8.9,21.3c7.8,4,17.4,1.7,22.5-5.3 c5-6.8,4.6-16.3-1.1-22.6c1.9-3.7,3.7-7.4,5.6-11.3c5-10.4,13.5-30.4,13.5-30.4c0.9-1.7,5.7-10.3,2.7-21.3 c-2.5-11.4-12.6-16.7-12.6-16.7c-12.2-7.9-29.2-15.2-29.2-15.2s0-4.1-1.1-7.1c-1.1-3.1-2.8-5.1-3.9-6.3c4.7-9.7,9.4-19.3,14.1-29 c-4.1-2-8.1-4-12.2-6.1c-4.8,9.8-9.7,19.7-14.5,29.5c-6.7-0.1-12.9,3.5-16.1,9.4c-3.4,6.3-2.7,14.1,1.9,19.8 C343.2,346.5,335,363.3,326.8,380.1z"></path> </g> </g> </g> </svg>
|
After Width: | Height: | Size: 2.5 KiB |
|
@ -0,0 +1,5 @@
|
||||||
|
<svg id="Group_1282" data-name="Group 1282" xmlns="http://www.w3.org/2000/svg" width="76.624" height="65.326" viewBox="0 0 76.624 65.326">
|
||||||
|
<path id="Path_2525" data-name="Path 2525" d="M1165,274.515a1.2,1.2,0,0,0,1.21-1.269c0-.9-.543-1.33-1.657-1.33h-1.8v4.712h.677v-2.054h.832l.019.025,1.291,2.029h.724l-1.389-2.1Zm-.783-.472h-.785V272.45h.995c.514,0,1.1.084,1.1.757,0,.774-.593.836-1.314.836" transform="translate(-1092.136 -213.406)" fill="#0a66c2"/>
|
||||||
|
<path id="Path_2520" data-name="Path 2520" d="M958.98,112.559h-9.6V97.525c0-3.585-.064-8.2-4.993-8.2-5,0-5.765,3.906-5.765,7.939v15.294h-9.6V81.642h9.216v4.225h.129a10.1,10.1,0,0,1,9.093-4.994c9.73,0,11.524,6.4,11.524,14.726ZM918.19,77.416a5.571,5.571,0,1,1,5.57-5.572,5.571,5.571,0,0,1-5.57,5.572m4.8,35.143h-9.61V81.642h9.61Zm40.776-55.2h-55.21a4.728,4.728,0,0,0-4.781,4.67v55.439a4.731,4.731,0,0,0,4.781,4.675h55.21a4.741,4.741,0,0,0,4.8-4.675V62.025a4.738,4.738,0,0,0-4.8-4.67" transform="translate(-903.776 -57.355)" fill="#0a66c2"/>
|
||||||
|
<path id="Path_2526" data-name="Path 2526" d="M1156.525,264.22a4.418,4.418,0,1,0,.085,0h-.085m0,8.33a3.874,3.874,0,1,1,3.809-3.938c0,.022,0,.043,0,.065a3.791,3.791,0,0,1-3.708,3.871h-.1" transform="translate(-1084.362 -207.809)" fill="#0a66c2"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 1.2 KiB |
Before Width: | Height: | Size: 34 KiB |
Before Width: | Height: | Size: 11 MiB |
Before Width: | Height: | Size: 2.2 MiB After Width: | Height: | Size: 190 KiB |
Before Width: | Height: | Size: 75 KiB |
|
@ -32,7 +32,7 @@ User-Agent: GPTBot
|
||||||
Disallow: /
|
Disallow: /
|
||||||
|
|
||||||
User-agent: *
|
User-agent: *
|
||||||
Disallow: /*
|
Disallow: /impressum
|
||||||
Allow: /
|
Allow: /
|
||||||
Allow: /resume
|
Allow: /resume
|
||||||
Allow: /publications
|
Allow: /publications
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
{"name":"","short_name":"","icons":[{"src":"/android-chrome-192x192.png","sizes":"192x192","type":"image/png"},{"src":"/android-chrome-512x512.png","sizes":"512x512","type":"image/png"}],"theme_color":"#ffffff","background_color":"#ffffff","display":"standalone"}
|
|
@ -608,6 +608,10 @@ video {
|
||||||
inset-inline-start: -0.375rem;
|
inset-inline-start: -0.375rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.m-16 {
|
||||||
|
margin: 4rem;
|
||||||
|
}
|
||||||
|
|
||||||
.mx-16 {
|
.mx-16 {
|
||||||
margin-left: 4rem;
|
margin-left: 4rem;
|
||||||
margin-right: 4rem;
|
margin-right: 4rem;
|
||||||
|
@ -628,6 +632,16 @@ video {
|
||||||
margin-bottom: 2rem;
|
margin-bottom: 2rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.my-16 {
|
||||||
|
margin-top: 4rem;
|
||||||
|
margin-bottom: 4rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx-8 {
|
||||||
|
margin-left: 2rem;
|
||||||
|
margin-right: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
.mb-1 {
|
.mb-1 {
|
||||||
margin-bottom: 0.25rem;
|
margin-bottom: 0.25rem;
|
||||||
}
|
}
|
||||||
|
@ -676,11 +690,6 @@ video {
|
||||||
display: flex;
|
display: flex;
|
||||||
}
|
}
|
||||||
|
|
||||||
.size-auto {
|
|
||||||
width: auto;
|
|
||||||
height: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.h-16 {
|
.h-16 {
|
||||||
height: 4rem;
|
height: 4rem;
|
||||||
}
|
}
|
||||||
|
@ -705,10 +714,6 @@ video {
|
||||||
height: 1px;
|
height: 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.h-screen {
|
|
||||||
height: 100vh;
|
|
||||||
}
|
|
||||||
|
|
||||||
.min-h-screen {
|
.min-h-screen {
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
}
|
}
|
||||||
|
@ -790,6 +795,16 @@ video {
|
||||||
gap: 1rem;
|
gap: 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.gap-2 {
|
||||||
|
gap: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.space-x-4 > :not([hidden]) ~ :not([hidden]) {
|
||||||
|
--tw-space-x-reverse: 0;
|
||||||
|
margin-right: calc(1rem * var(--tw-space-x-reverse));
|
||||||
|
margin-left: calc(1rem * calc(1 - var(--tw-space-x-reverse)));
|
||||||
|
}
|
||||||
|
|
||||||
.space-x-8 > :not([hidden]) ~ :not([hidden]) {
|
.space-x-8 > :not([hidden]) ~ :not([hidden]) {
|
||||||
--tw-space-x-reverse: 0;
|
--tw-space-x-reverse: 0;
|
||||||
margin-right: calc(2rem * var(--tw-space-x-reverse));
|
margin-right: calc(2rem * var(--tw-space-x-reverse));
|
||||||
|
@ -802,6 +817,18 @@ video {
|
||||||
margin-bottom: calc(0.5rem * var(--tw-space-y-reverse));
|
margin-bottom: calc(0.5rem * var(--tw-space-y-reverse));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.space-y-4 > :not([hidden]) ~ :not([hidden]) {
|
||||||
|
--tw-space-y-reverse: 0;
|
||||||
|
margin-top: calc(1rem * calc(1 - var(--tw-space-y-reverse)));
|
||||||
|
margin-bottom: calc(1rem * var(--tw-space-y-reverse));
|
||||||
|
}
|
||||||
|
|
||||||
|
.space-y-8 > :not([hidden]) ~ :not([hidden]) {
|
||||||
|
--tw-space-y-reverse: 0;
|
||||||
|
margin-top: calc(2rem * calc(1 - var(--tw-space-y-reverse)));
|
||||||
|
margin-bottom: calc(2rem * var(--tw-space-y-reverse));
|
||||||
|
}
|
||||||
|
|
||||||
.overflow-x-auto {
|
.overflow-x-auto {
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
}
|
}
|
||||||
|
@ -1278,6 +1305,11 @@ video {
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 640px) {
|
@media (min-width: 640px) {
|
||||||
|
.sm\:my-16 {
|
||||||
|
margin-top: 4rem;
|
||||||
|
margin-bottom: 4rem;
|
||||||
|
}
|
||||||
|
|
||||||
.sm\:mt-0 {
|
.sm\:mt-0 {
|
||||||
margin-top: 0px;
|
margin-top: 0px;
|
||||||
}
|
}
|
||||||
|
@ -1292,6 +1324,12 @@ video {
|
||||||
margin-bottom: calc(0px * var(--tw-space-y-reverse));
|
margin-bottom: calc(0px * var(--tw-space-y-reverse));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.sm\:space-x-8 > :not([hidden]) ~ :not([hidden]) {
|
||||||
|
--tw-space-x-reverse: 0;
|
||||||
|
margin-right: calc(2rem * var(--tw-space-x-reverse));
|
||||||
|
margin-left: calc(2rem * calc(1 - var(--tw-space-x-reverse)));
|
||||||
|
}
|
||||||
|
|
||||||
.sm\:text-center {
|
.sm\:text-center {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,19 @@ pub fn P(props: PProps) -> Element {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[component]
|
||||||
|
pub fn Title(props: PProps) -> Element {
|
||||||
|
rsx! {
|
||||||
|
div {
|
||||||
|
class: "{props.class}",
|
||||||
|
h1 {
|
||||||
|
class: "mb-2 text-2xl font-bold tracking-tight text-gray-900 dark:text-white",
|
||||||
|
{props.children}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[component]
|
#[component]
|
||||||
pub fn H1(props: PProps) -> Element {
|
pub fn H1(props: PProps) -> Element {
|
||||||
rsx! {
|
rsx! {
|
||||||
|
@ -84,7 +97,7 @@ pub fn Card(prop: CardProp) -> Element {
|
||||||
Picture {src: "{prop.picture}"},
|
Picture {src: "{prop.picture}"},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
H5 { "{prop.name}", span { class: "text-grey-600 dark:text-grey-500 text-lg", " {prop.gender}" } },
|
Title { "{prop.name}", span { class: "text-grey-600 dark:text-grey-500 text-lg", " {prop.gender}" } },
|
||||||
{ prop.children }
|
{ prop.children }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
34
src/cv.rs
|
@ -8,15 +8,24 @@ pub fn CV() -> Element {
|
||||||
rsx! {
|
rsx! {
|
||||||
div {
|
div {
|
||||||
class: "flex flex-col",
|
class: "flex flex-col",
|
||||||
|
div {
|
||||||
|
class: "flex flex-col sm:flex-row justify-center items-center sm:space-x-8 space-y-8 sm:space-y-0",
|
||||||
|
img {
|
||||||
|
class: "rounded-full w-24 h-24",
|
||||||
|
alt: "headshot",
|
||||||
|
src: "/pictures/headshot.webp"
|
||||||
|
}
|
||||||
Introduction {},
|
Introduction {},
|
||||||
|
Socials {}
|
||||||
|
},
|
||||||
|
|
||||||
HR {}
|
HR {}
|
||||||
div {
|
div {
|
||||||
class: "flex justify-between",
|
class: "flex flex-col justify-between sm:flex-row",
|
||||||
WorkExperience {},
|
WorkExperience {},
|
||||||
Miscellaneous {},
|
Miscellaneous {},
|
||||||
},
|
},
|
||||||
HR {},
|
HR {},
|
||||||
Socials {}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -27,12 +36,9 @@ fn Introduction() -> Element {
|
||||||
rsx! {
|
rsx! {
|
||||||
div {
|
div {
|
||||||
class: "flex",
|
class: "flex",
|
||||||
img {
|
P { { translate!(i18, "cv.introduction_0") } },
|
||||||
class: "rounded-full w-16 h-16 mx-16",
|
P { { translate!(i18, "cv.introduction_1") } }
|
||||||
src: "/pictures/headshot.webp"
|
},
|
||||||
}
|
|
||||||
P { { translate!(i18, "cv.introduction") }}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -189,7 +195,7 @@ fn CVEntry(props: CVEntryProps) -> Element {
|
||||||
time { class:"mb-1 text-sm font-normal leading-none text-gray-400 dark:text-gray-500", "{props.time}"},
|
time { class:"mb-1 text-sm font-normal leading-none text-gray-400 dark:text-gray-500", "{props.time}"},
|
||||||
h6 { class: "text-lg font-semibold text-gray-900 dark:text-white", "{props.title}"}
|
h6 { class: "text-lg font-semibold text-gray-900 dark:text-white", "{props.title}"}
|
||||||
ul {
|
ul {
|
||||||
class: "flex",
|
class: "flex flex-wrap gap-2",
|
||||||
for (index, value) in props.technologies.iter().enumerate() {
|
for (index, value) in props.technologies.iter().enumerate() {
|
||||||
li { key: "{index}", RandomBadge { text: "{value}"} }
|
li { key: "{index}", RandomBadge { text: "{value}"} }
|
||||||
}
|
}
|
||||||
|
@ -228,9 +234,17 @@ fn random_badge_color(seed: usize) -> String {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn Socials() -> Element {
|
fn Socials() -> Element {
|
||||||
|
let i18 = use_i18();
|
||||||
|
|
||||||
rsx! {
|
rsx! {
|
||||||
div {
|
div {
|
||||||
"todo!()"
|
class: "ms-8 max-w-3/4",
|
||||||
|
H4 { { translate!(i18, "cv.socials.title") } },
|
||||||
|
div {
|
||||||
|
class: "flex justify-center items-center space-x-4",
|
||||||
|
P { Link { to:"https://www.linkedin.com/in/tudattr/", class:"hover:underline", new_tab: true, img { class: "h-8", src:"/pictures/LI-Bug.svg.original.svg", alt:"LinkedIn Logo" } }},
|
||||||
|
P { Link { to:"https://git.tudattr.dev/tudattr", class:"hover:underline", new_tab: true, img { class: "h-8", src:"/pictures/Gitea_Logo.svg", alt:"Gitea Logo" } }},
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,13 +18,14 @@ pub fn Home() -> Element {
|
||||||
class: "mb-2",
|
class: "mb-2",
|
||||||
P { { translate!(i18, "home.card.l1") } },
|
P { { translate!(i18, "home.card.l1") } },
|
||||||
P { { translate!(i18, "home.card.l2") } },
|
P { { translate!(i18, "home.card.l2") } },
|
||||||
P { { translate!(i18, "home.card.l3") },
|
P { { translate!(i18, "home.card.l3") } },
|
||||||
|
P { { translate!(i18, "home.card.l4") }
|
||||||
Link {
|
Link {
|
||||||
to: "https://git.tudattr.dev/explore/repos",
|
to: "https://git.tudattr.dev/explore/repos",
|
||||||
new_tab: true,
|
new_tab: true,
|
||||||
class: "items-center font-medium hover:underline",
|
class: "items-center font-medium hover:underline",
|
||||||
"gitea"},
|
"gitea"},
|
||||||
{ translate!(i18, "home.card.l3_1") }
|
{ translate!(i18, "home.card.l4_1") }
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
|
@ -18,13 +18,14 @@
|
||||||
"gender": "",
|
"gender": "",
|
||||||
"l1": "Hallihallo! 👋🏻👋🏼👋🏽👋🏾👋🏿",
|
"l1": "Hallihallo! 👋🏻👋🏼👋🏽👋🏾👋🏿",
|
||||||
"l2": "Willkommen auf meiner kleinen Webseite im World Wide Web.",
|
"l2": "Willkommen auf meiner kleinen Webseite im World Wide Web.",
|
||||||
"l3": "Während du hier bist, schau dir doch meine Projekte auf ",
|
"l3": "Mein Name ist Tuan und ich bin Linux-Bastler, IT-Security Nerd und Automatisierer aus Leidenschaft.",
|
||||||
"l3_1": " an.",
|
"l4": "Während du hier bist, schau dir doch meine Projekte auf ",
|
||||||
|
"l4_1": " an.",
|
||||||
"contact_button": "Get in touch."
|
"contact_button": "Get in touch."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"cv": {
|
"cv": {
|
||||||
"introduction": "Während meines Bachelorstudiums habe ich viele Erfahrungen in der Industrie und im Studium gesammelt. Meine beruflichen und persönlichen Interessen sind DevOps/IaC, Systems/Software Security und Computer Networking. All diese Interessen vertiefe ich in persönlichen Projekten wie meinem Homelab und CTF-Challenges.",
|
"introduction_0": "Während meines Bachelorstudiums habe ich viele Erfahrungen in der Industrie und im Studium gesammelt. Meine beruflichen und persönlichen Interessen sind DevOps/IaC, Systems/Software Security und Computer Networking. All diese Interessen vertiefe ich in persönlichen Projekten wie meinem Homelab und CTF-Challenges.",
|
||||||
"workexperience": {
|
"workexperience": {
|
||||||
"title": "Berufserfahrung",
|
"title": "Berufserfahrung",
|
||||||
"se1_gefeba": {
|
"se1_gefeba": {
|
||||||
|
@ -53,6 +54,9 @@
|
||||||
"description": "Während meiner Tätigkeit als wissenschaftlicher Mitarbeiter in der Network Communication System Research Group an der Universität Duisburg-Essen habe ich an der Forschung rund um Software Defined Networking, 5G, Staukontrollalgorithmen und föderiertes maschinelles Lernen mitgearbeitet. Ich habe die On-Premise- und Cloud-Infrastruktur, das Inventarsystem und die Online-Präsenz der Forschungsgruppe aufgebaut und verwaltet."
|
"description": "Während meiner Tätigkeit als wissenschaftlicher Mitarbeiter in der Network Communication System Research Group an der Universität Duisburg-Essen habe ich an der Forschung rund um Software Defined Networking, 5G, Staukontrollalgorithmen und föderiertes maschinelles Lernen mitgearbeitet. Ich habe die On-Premise- und Cloud-Infrastruktur, das Inventarsystem und die Online-Präsenz der Forschungsgruppe aufgebaut und verwaltet."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"socials": {
|
||||||
|
"title": "Profile"
|
||||||
|
},
|
||||||
"education": {
|
"education": {
|
||||||
"title": "Bildungsweg",
|
"title": "Bildungsweg",
|
||||||
"bachelor": {
|
"bachelor": {
|
||||||
|
|
|
@ -18,13 +18,14 @@
|
||||||
"gender": "(He/Him)",
|
"gender": "(He/Him)",
|
||||||
"l1": "Hey there! 👋🏻👋🏼👋🏽👋🏾👋🏿",
|
"l1": "Hey there! 👋🏻👋🏼👋🏽👋🏾👋🏿",
|
||||||
"l2": "Welcome to my little place on the internet.",
|
"l2": "Welcome to my little place on the internet.",
|
||||||
"l3": "While you're here, why don't you check out my projects over on ",
|
"l3": "My name is Tuan and I'm passionate about Linux, system security, automation, network performance tweaking and all things tech.",
|
||||||
"l3_1": "?",
|
"l4": "While you're here, why don't you check out my projects over on ",
|
||||||
|
"l4_1": "?",
|
||||||
"contact_button": "Get in touch."
|
"contact_button": "Get in touch."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"cv": {
|
"cv": {
|
||||||
"introduction": "While studying for my bachelors degree I accumulated a lot of industry and academic experience. My professional and personal intererests are DevOps/IaC, Systems/Software Security and Computer Networking. All of which I deepen in personal projects such as my homelab and CTF challenges.",
|
"introduction_0": "While studying for my bachelors degree I accumulated a lot of industry and academic experience. My professional and personal intererests are DevOps/IaC, Systems/Software Security and Computer Networking. All of which I deepen in personal projects such as my homelab and CTF challenges.",
|
||||||
"workexperience": {
|
"workexperience": {
|
||||||
"title": "Work Experience",
|
"title": "Work Experience",
|
||||||
"se1_gefeba": {
|
"se1_gefeba": {
|
||||||
|
@ -53,6 +54,9 @@
|
||||||
"description": "While working for the Network Communication System Research Group at the University Duisburg-Essen as a research assistant I've assisted in research around software defined networking, 5G, congestion control algorithms and federated machine learning. I've established and managed the research groups on-premise and cloud infractructure, inventory system and online presence."
|
"description": "While working for the Network Communication System Research Group at the University Duisburg-Essen as a research assistant I've assisted in research around software defined networking, 5G, congestion control algorithms and federated machine learning. I've established and managed the research groups on-premise and cloud infractructure, inventory system and online presence."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"socials": {
|
||||||
|
"title": "Socials"
|
||||||
|
},
|
||||||
"education": {
|
"education": {
|
||||||
"title": "Education",
|
"title": "Education",
|
||||||
"bachelor": {
|
"bachelor": {
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use dioxus::prelude::*;
|
use dioxus::prelude::*;
|
||||||
|
|
||||||
mod footer;
|
pub mod footer;
|
||||||
mod header;
|
pub mod header;
|
||||||
|
|
||||||
use crate::{Body, Route};
|
use crate::{Body, Route};
|
||||||
use footer::Footer;
|
use footer::Footer;
|
||||||
|
@ -9,17 +9,13 @@ use header::Header;
|
||||||
|
|
||||||
pub fn Layout() -> Element {
|
pub fn Layout() -> Element {
|
||||||
rsx! {
|
rsx! {
|
||||||
meta {
|
|
||||||
name: "description",
|
|
||||||
content: "Explore Tuan-Dat Tran's personal website featuring his CV, publications, projects, and consulting services. Get insights into his professional journey and connect for collaboration opportunities.",
|
|
||||||
}
|
|
||||||
div {
|
div {
|
||||||
class: "flex flex-col min-h-screen",
|
class: "flex flex-col min-h-screen",
|
||||||
Header {},
|
Header {},
|
||||||
Body {
|
Body {
|
||||||
Outlet::<Route> {},
|
Outlet::<Route> {},
|
||||||
}
|
|
||||||
Footer {},
|
Footer {},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
52
src/main.rs
|
@ -2,9 +2,12 @@
|
||||||
|
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
|
use components::H1;
|
||||||
use dioxus::prelude::*;
|
use dioxus::prelude::*;
|
||||||
use dioxus_sdk::i18n::*;
|
use dioxus_sdk::i18n::*;
|
||||||
|
|
||||||
|
use layout::footer::Footer;
|
||||||
|
use layout::header::Header;
|
||||||
use tracing::Level;
|
use tracing::Level;
|
||||||
|
|
||||||
pub mod components;
|
pub mod components;
|
||||||
|
@ -60,6 +63,41 @@ fn App() -> Element {
|
||||||
});
|
});
|
||||||
|
|
||||||
rsx! {
|
rsx! {
|
||||||
|
meta {
|
||||||
|
name: "description",
|
||||||
|
content: "Visit Tuan-Dat Tran's website for his CV, publications, projects, and consulting services. Connect for collaboration.",
|
||||||
|
},
|
||||||
|
script {
|
||||||
|
r#type: "application/ld+json",
|
||||||
|
"
|
||||||
|
{{
|
||||||
|
\"@context\": \"https://schema.org\",
|
||||||
|
\"@type\": \"ProfilePage\",
|
||||||
|
\"mainEntity\": {{
|
||||||
|
\"@type\": \"Person\",
|
||||||
|
\"name\": \"Tuan-Dat Tran\",
|
||||||
|
\"alternateName\": \"TuDatTr\",
|
||||||
|
\"image\": \"https://www.tudattr.dev/pictures/headshot.webp\",
|
||||||
|
}}
|
||||||
|
}}
|
||||||
|
"
|
||||||
|
}
|
||||||
|
// meta {
|
||||||
|
// property: "og:title",
|
||||||
|
// content: "Tuan-Dat Trans Personal Website",
|
||||||
|
// }
|
||||||
|
// meta {
|
||||||
|
// property: "og:description",
|
||||||
|
// content: "Explore Tuan-Dat Tran's personal website featuring his CV, publications, projects, and consulting services. Get insights into his professional journey and connect for collaboration opportunities.",
|
||||||
|
// }
|
||||||
|
// meta {
|
||||||
|
// property: "og:image",
|
||||||
|
// content: "https://www.tudattr.dev/pictures/headshot.webp",
|
||||||
|
// }
|
||||||
|
// meta {
|
||||||
|
// property: "og:url",
|
||||||
|
// content: "https://tudattr.dev",
|
||||||
|
// }
|
||||||
div {
|
div {
|
||||||
class: "bg-white dark:bg-gray-900 min-h-screen",
|
class: "bg-white dark:bg-gray-900 min-h-screen",
|
||||||
Router::<Route> {},
|
Router::<Route> {},
|
||||||
|
@ -71,12 +109,16 @@ fn App() -> Element {
|
||||||
fn PageNotFound(route: Vec<String>) -> Element {
|
fn PageNotFound(route: Vec<String>) -> Element {
|
||||||
rsx! {
|
rsx! {
|
||||||
div {
|
div {
|
||||||
class: "h-screen items-center justify-center",
|
class: "flex flex-col min-h-screen items",
|
||||||
img {
|
Header {},
|
||||||
class: "size-auto",
|
div {
|
||||||
src: "https://raw.githubusercontent.com/SAWARATSUKI/ServiceLogos/main/404Notfound/NotFound.png"
|
class: "container mx-auto p-4 flex items-center justify-center max-w-md w-full",
|
||||||
}
|
H1 {
|
||||||
|
"Site not found (404)"
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
Footer {}
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|