+
{workshop.title}
-
{workshop.description}
{workshop.date.split('-').reverse().join('-')}
{workshop.time_start.slice(0, 5).split(':').join('h')} - {workshop.time_end.slice(0, 5).split(':').join('h')}
@@ -274,14 +309,25 @@ export default function Header() {
-
-
diff --git a/frontend-plataforma-tutoriais/src/components/header/styles.module.css b/frontend-plataforma-tutoriais/src/components/header/styles.module.css
index 0a6cc11..ceb45e3 100644
--- a/frontend-plataforma-tutoriais/src/components/header/styles.module.css
+++ b/frontend-plataforma-tutoriais/src/components/header/styles.module.css
@@ -114,12 +114,12 @@
width: 200px;
text-decoration: none;
color: var(--text-white);
- background-color: var(--primary-color);
- border-radius: 8px;
+ background: var(--bg-gradient);
+ box-shadow: var(--shadow-primary);
+ border-radius: var(--border-radius-button);
padding: 10px 12px;
font-weight: 600;
transition: all 0.3s ease;
-
&:hover {
background-color: var(--secondary-color);
}
@@ -174,6 +174,9 @@
}
.boxVideo {
+ background: linear-gradient(90deg, #e0e0e0 25%, #f0f0f0 50%, #e0e0e0 75%);
+ background-size: 200% 100%;
+ animation: skeleton 1.5s infinite;
height: 200px;
position: relative;
overflow: hidden;
@@ -181,7 +184,12 @@
}
.boxVideo::after {
- content: '';
+ content: '\F4F4';
+ color: var(--bg-grey);
+ font-size: 4rem;
+ align-content: center;
+ text-align: center;
+ font-family: 'bootstrap-icons';
position: absolute;
top: 0;
left: 0;
@@ -189,6 +197,11 @@
height: 100%;
background: linear-gradient(180deg, rgba(0, 0, 0, 0) 40%, rgba(0, 0, 0, 0.8) 100%);
z-index: 1;
+ transition: all 0.3s ease;
+}
+
+.boxVideo:hover::after {
+ color: var(--bg-primary-color);
}
.boxVideoInfo{
@@ -268,12 +281,17 @@
}
.thumbnailWorkshop{
+ background: linear-gradient(90deg, #e0e0e0 25%, #f0f0f0 50%, #e0e0e0 75%);
+ background-size: 200% 100%;
+ animation: skeleton 1.5s infinite;
width: 100%;
object-fit: cover;
border-top-right-radius: var(--border-radius);
border-top-left-radius: var(--border-radius);
transition: all 0.3s ease;
height: 150px;
+ position: relative;
+ overflow: hidden;
}
.icon{
@@ -284,6 +302,11 @@
animation: spin 1s linear infinite;
}
+@keyframes skeleton {
+ 0% { background-position: 200% 0; }
+ 100% { background-position: -200% 0; }
+}
+
@keyframes spin {
from {
transform: rotate(0deg);
diff --git a/frontend-plataforma-tutoriais/src/components/sidebar/styles.module.css b/frontend-plataforma-tutoriais/src/components/sidebar/styles.module.css
index 6b19f61..15dc290 100644
--- a/frontend-plataforma-tutoriais/src/components/sidebar/styles.module.css
+++ b/frontend-plataforma-tutoriais/src/components/sidebar/styles.module.css
@@ -91,7 +91,7 @@
text-decoration: none;
color: var(--text-white);
background-color: var(--primary-color);
- border-radius: 8px;
+ border-radius: var(--border-radius-button);
padding: 10px 12px;
font-weight: 600;
transition: all 0.3s ease;
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetVideos.ts b/frontend-plataforma-tutoriais/src/hooks/useGetVideos.ts
index 55aad6a..8bd101e 100644
--- a/frontend-plataforma-tutoriais/src/hooks/useGetVideos.ts
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetVideos.ts
@@ -1,22 +1,42 @@
import type { ApiErrorResponse, Video } from "../types";
+type GetVideosParams = {
+ page?: number;
+ search?: string;
+ category?: string;
+ status?: "active" | "inactive";
+ watched?: 0 | 1;
+};
+
export function useGetVideos() {
- async function getVideos(searchQuery?: string) {
- const response = await fetch(`http://127.0.0.1:8000/api/videos?search=${encodeURIComponent(searchQuery ?? "")}`, {
- method: "GET",
- headers: {
- Accept: "application/json",
- "Content-Type": "application/json",
- Authorization: `Bearer ${localStorage.getItem("token")}`
- },
- });
+ async function getVideos(params: GetVideosParams) {
+ const query = new URLSearchParams();
+
+ if (params.page !== undefined) query.append("page", params.page.toString());
+ if (params.search) query.append("search", params.search);
+ if (params.category) query.append("category", params.category);
+ if (params.status) query.append("status", params.status);
+ if (params.watched !== undefined) query.append("watched", params.watched.toString());
+
+ const response = await fetch(`http://127.0.0.1:8000/api/videos?${query.toString()}`, {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ }
+ );
const data = await response.json();
if (response.ok) {
- return data.data as Video[];
+ return {
+ videos: data.data as Video[],
+ meta: data.meta
+ };
} else {
- return (data as ApiErrorResponse);
+ return data as ApiErrorResponse;
}
}
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetVideosLength.ts b/frontend-plataforma-tutoriais/src/hooks/useGetVideosLength.ts
new file mode 100644
index 0000000..db1f9fd
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetVideosLength.ts
@@ -0,0 +1,32 @@
+import type { ApiErrorResponse } from "../types";
+
+export function useGetVideosLength() {
+ async function getVideosLength() {
+ try {
+ const response = await fetch("http://127.0.0.1:8000/api/videos-length", {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ });
+
+ const data = await response.json();
+
+ if (response.ok) {
+ return {
+ videos: data.data.videos,
+ videosWatched: data.data.videosWatched,
+ };
+ }
+
+ return data as ApiErrorResponse;
+
+ } catch (error) {
+ return error as ApiErrorResponse;
+ }
+ }
+
+ return { getVideosLength };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetVideosSearch.ts b/frontend-plataforma-tutoriais/src/hooks/useGetVideosSearch.ts
new file mode 100644
index 0000000..53b115d
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetVideosSearch.ts
@@ -0,0 +1,30 @@
+import type { ApiErrorResponse, Video } from "../types";
+
+export function useGetVideosSearch() {
+ async function getVideosSearch(searchQuery: string = "") {
+ const response = await fetch(
+ `http://127.0.0.1:8000/api/videos-search?search=${encodeURIComponent(searchQuery)}`,
+ {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ }
+ );
+
+ const data = await response.json();
+
+ if (response.ok) {
+ return {
+ videos: data.data as Video[],
+ meta: data.meta
+ };
+ } else {
+ return data as ApiErrorResponse;
+ }
+ }
+
+ return { getVideosSearch };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetWorkshops.ts b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshops.ts
index 8fa9d2b..0090bc4 100644
--- a/frontend-plataforma-tutoriais/src/hooks/useGetWorkshops.ts
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshops.ts
@@ -1,8 +1,22 @@
import type { ApiErrorResponse, Workshop } from "../types";
+type GetWorkshopsParams = {
+ search?: string;
+ status?: string;
+ page?: number;
+ per_page?: number;
+};
+
export function useGetWorkshops() {
- async function getWorkshops(searchQuery?: string) {
- const response = await fetch(`http://127.0.0.1:8000/api/workshops?search=${encodeURIComponent(searchQuery ?? "")}`, {
+ async function getWorkshops(params: GetWorkshopsParams) {
+ const query = new URLSearchParams();
+
+ if (params.search) query.append("search", params.search);
+ if (params.status) query.append("status", params.status);
+ if (params.page) query.append("page", params.page.toString());
+ if (params.per_page) query.append("per_page", params.per_page.toString());
+
+ const response = await fetch(`http://127.0.0.1:8000/api/workshops?${query.toString()}`, {
method: "GET",
headers: {
Accept: "application/json",
@@ -14,7 +28,10 @@ export function useGetWorkshops() {
const data = await response.json();
if (response.ok) {
- return data.data as Workshop[];
+ return {
+ workshops: data.data as Workshop[],
+ meta: data.meta
+ };
} else {
return (data as ApiErrorResponse);
}
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsLength.ts b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsLength.ts
new file mode 100644
index 0000000..ff911e9
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsLength.ts
@@ -0,0 +1,32 @@
+import type { ApiErrorResponse } from "../types";
+
+export function useGetWorkshopsLength() {
+ async function getWorkshopsLength() {
+ try {
+ const response = await fetch("http://127.0.0.1:8000/api/workshops-length", {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ });
+
+ const data = await response.json();
+
+ if (response.ok) {
+ return {
+ workshops: data.data.workshops,
+ workshopsInscribed: data.data.workshopsInscribed,
+ };
+ }
+
+ return data as ApiErrorResponse;
+
+ } catch (error) {
+ return error as ApiErrorResponse;
+ }
+ }
+
+ return { getWorkshopsLength };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsSearch.ts b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsSearch.ts
new file mode 100644
index 0000000..1cfeee2
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useGetWorkshopsSearch.ts
@@ -0,0 +1,30 @@
+import type { ApiErrorResponse, Workshop } from "../types";
+
+export function useGetWorkshopsSearch() {
+ async function getWorkshopsSearch(searchQuery: string = "") {
+ const response = await fetch(
+ `http://127.0.0.1:8000/api/workshops-search?search=${encodeURIComponent(searchQuery)}`,
+ {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ }
+ );
+
+ const data = await response.json();
+
+ if (response.ok) {
+ return {
+ workshops: data.data as Workshop[],
+ meta: data.meta
+ };
+ } else {
+ return data as ApiErrorResponse;
+ }
+ }
+
+ return { getWorkshopsSearch };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/useNextVideos.ts b/frontend-plataforma-tutoriais/src/hooks/useNextVideos.ts
new file mode 100644
index 0000000..8c91421
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useNextVideos.ts
@@ -0,0 +1,23 @@
+import type { ApiErrorResponse, Video } from "../types";
+
+export function useNextVideos() {
+ async function getNextVideos() {
+ const response = await fetch("http://127.0.0.1:8000/api/next-videos", {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ });
+ const data = await response.json();
+ if (response.ok) {
+ return {
+ videos: data.data as Video[],
+ };
+ } else {
+ return data as ApiErrorResponse;
+ }
+ }
+ return { getNextVideos };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/useNextWorkshops.ts b/frontend-plataforma-tutoriais/src/hooks/useNextWorkshops.ts
new file mode 100644
index 0000000..e681fad
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useNextWorkshops.ts
@@ -0,0 +1,23 @@
+import type { ApiErrorResponse, NextWorkshopsResponse } from "../types";
+
+export function useNextWorkshops() {
+ async function getNextWorkshops() {
+ const response = await fetch("http://127.0.0.1:8000/api/next-workshops", {
+ method: "GET",
+ headers: {
+ Accept: "application/json",
+ "Content-Type": "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ });
+ const data = await response.json();
+ if (response.ok) {
+ return {
+ workshops: data.data as NextWorkshopsResponse[],
+ };
+ } else {
+ return data as ApiErrorResponse;
+ }
+ }
+ return { getNextWorkshops };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/hooks/usePreloadImages.ts b/frontend-plataforma-tutoriais/src/hooks/usePreloadImages.ts
deleted file mode 100644
index 35ff3fa..0000000
--- a/frontend-plataforma-tutoriais/src/hooks/usePreloadImages.ts
+++ /dev/null
@@ -1,17 +0,0 @@
-import { useCallback } from "react";
-
-export function usePreloadImages() {
- const preloadImages = useCallback ((urls: string[]): Promise
=> {
- return Promise.all(
- urls.map(url => new Promise((resolve) => {
- const img = new Image();
- img.onload = () => resolve();
- img.onerror = () => resolve();
- img.src = url;
- }))
- );
- }, []);
-
- return { preloadImages };
-}
-
diff --git a/frontend-plataforma-tutoriais/src/hooks/useVideoWatch.ts b/frontend-plataforma-tutoriais/src/hooks/useVideoWatch.ts
new file mode 100644
index 0000000..4cab5c4
--- /dev/null
+++ b/frontend-plataforma-tutoriais/src/hooks/useVideoWatch.ts
@@ -0,0 +1,29 @@
+import { useState, useRef } from "react";
+
+export function useVideoWatch(videoId: number, initialWatched: boolean) {
+ const [watched, setWatched] = useState(initialWatched);
+ const alreadySent = useRef(false);
+
+ const markAsWatched = async () => {
+ if (alreadySent.current || watched ) return;
+
+ alreadySent.current = true;
+
+ try {
+ await fetch(`http://127.0.0.1:8000/api/video/${videoId}/watch`, {
+ method: "POST",
+ headers: {
+ Accept: "application/json",
+ Authorization: `Bearer ${localStorage.getItem("token")}`
+ },
+ });
+
+ setWatched(true);
+ } catch (error) {
+ console.error("Erro ao marcar o vídeo como assistido: ", error);
+ alreadySent.current = false;
+ }
+ }
+
+ return { watched, markAsWatched };
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/index.css b/frontend-plataforma-tutoriais/src/index.css
index 986ce25..0bd9dc0 100644
--- a/frontend-plataforma-tutoriais/src/index.css
+++ b/frontend-plataforma-tutoriais/src/index.css
@@ -1,157 +1,164 @@
:root {
- --primary-color: #B20112;
- --secondary-color: #C4534A;
- --tertiary-color: #0054B0;
- --success-color: #08a35d;
- --neutral-color: #8A716E;
- --primary-contrast-color: #410002;
+ --primary-color: #B20112;
+ --secondary-color: #C4534A;
+ --tertiary-color: #0054B0;
+ --success-color: #08a35d;
+ --neutral-color: #8A716E;
+ --primary-contrast-color: #410002;
- --text-primary-color: var(--primary-color);
- --text-secondary-color: var(--secondary-color);
- --text-tertiary-color: var(--tertiary-color);
- --text-neutral-color: var(--neutral-color);
- --text-primary-contrast-color: var(--primary-contrast-color);
- --text-white: #ffffff;
- --text-black: #000000;
+ --text-primary-color: var(--primary-color);
+ --text-secondary-color: var(--secondary-color);
+ --text-tertiary-color: var(--tertiary-color);
+ --text-neutral-color: var(--neutral-color);
+ --text-primary-contrast-color: var(--primary-contrast-color);
+ --text-white: #ffffff;
+ --text-black: #000000;
- --size-font-title: 2.2rem;
- --size-font-subtitle: 1.7rem;
- --size-font-text: 1.2rem;
- --size-font-small: 1rem;
+ --size-font-title: clamp(1.8rem, calc(1.607vw + 0.629rem), 2.2rem);
+ --size-font-subtitle: clamp(1.55rem, calc(1.116vw + 0.614rem), 1.7rem);
+ --size-font-text: clamp(1.2rem, calc(0.394vw + 0.811rem), 1.2rem);
+ --size-font-small: clamp(1rem, calc(0.295vw + 0.708rem), 1rem);
+
+ --border-radius: 15px;
+ --border-radius-input: 5px;
+ --border-radius-button: 10px;
+ --border-primary-color: var(--primary-color);
+ --border-secondary-color: var(--secondary-color);
+ --border-tertiary-color: var(--tertiary-color);
+ --border-neutral-color: var(--neutral-color);
-
- --border-radius: 15px;
- --border-radius-input: 5px;
- --border-radius-button: 10px;
- --border-primary-color: var(--primary-color);
- --border-secondary-color: var(--secondary-color);
- --border-tertiary-color: var(--tertiary-color);
- --border-neutral-color: var(--neutral-color);
+ --bg-white: #ffffff;
+ --bg-grey: #F3F3F3;
+ --bg-primary-color: var(--primary-color);
+ --bg-primary-color-opacity: #FFEDEA;
+ --bg-secondary-color: var(--secondary-color);
+ --bg-tertiary-color: var(--tertiary-color);
+ --bg-tertiary-color-opacity: #CFE2FF;
+ --bg-success-color-opacity: #D1E7DD;
+ --bg-neutral-color: var(--neutral-color);
+ --bg-gradient: linear-gradient(135deg, #b20112 0%, #8e010e 100%);
- --bg-white: #ffffff;
- --bg-grey: #F3F3F3;
- --bg-primary-color: var(--primary-color);
- --bg-primary-color-opacity: #FFEDEA;
- --bg-secondary-color: var(--secondary-color);
- --bg-tertiary-color: var(--tertiary-color);
- --bg-tertiary-color-opacity: #CFE2FF;
- --bg-success-color-opacity: #D1E7DD;
- --bg-neutral-color: var(--neutral-color);
- --bg-gradient: linear-gradient(135deg, #b20112 0%, #8e010e 100%);
+ --text-h: #08060d;
+ --bg: #fff;
+ --border: #e5e4e7;
+ --code-bg: #f4f3ec;
+ --accent: #aa3bff;
+ --accent-bg: rgba(170, 59, 255, 0.1);
+ --accent-border: rgba(170, 59, 255, 0.5);
+ --social-bg: rgba(244, 243, 236, 0.5);
+ --shadow:rgba(0, 0, 0, 0.1) 0 10px 15px -3px,
+ rgba(0, 0, 0, 0.05) 0 4px 6px -2px;
+ --shadow-primary:0 4px 10px rgba(178, 1, 18, 0.4),
+ 0 2px 6px rgba(142, 1, 14, 0.3);
+ ;
- --text-h: #08060d;
- --bg: #fff;
- --border: #e5e4e7;
- --code-bg: #f4f3ec;
- --accent: #aa3bff;
- --accent-bg: rgba(170, 59, 255, 0.1);
- --accent-border: rgba(170, 59, 255, 0.5);
- --social-bg: rgba(244, 243, 236, 0.5);
- --shadow:rgba(0, 0, 0, 0.1) 0 10px 15px -3px, rgba(0, 0, 0, 0.05) 0 4px 6px -2px;
- --shadow-primary:0 4px 10px rgba(178, 1, 18, 0.4), 0 2px 6px rgba(142, 1, 14, 0.3);;
+ --font-family: 'Manrope',
+ sans-serif;
- --font-family: 'Manrope', sans-serif;
+ font: 18px/145% var(--sans);
+ letter-spacing: 0.18px;
+ color-scheme: light dark;
+ color: var(--text);
+ background: var(--bg);
+ font-synthesis: none;
+ text-rendering: optimizeLegibility;
+ -webkit-font-smoothing: antialiased;
+ -moz-osx-font-smoothing: grayscale;
- font: 18px/145% var(--sans);
- letter-spacing: 0.18px;
- color-scheme: light dark;
- color: var(--text);
- background: var(--bg);
- font-synthesis: none;
- text-rendering: optimizeLegibility;
- -webkit-font-smoothing: antialiased;
- -moz-osx-font-smoothing: grayscale;
-
- @media (max-width: 1024px) {
- font-size: 16px;
- }
+ @media (max-width: 1024px) {
+ font-size: 16px;
+ }
}
@media (prefers-color-scheme: dark) {
- :root {
- --text: #9ca3af;
- --text-h: #f3f4f6;
- --bg: #16171d;
- --border: #2e303a;
- --code-bg: #1f2028;
- --accent: #c084fc;
- --accent-bg: rgba(192, 132, 252, 0.15);
- --accent-border: rgba(192, 132, 252, 0.5);
- --social-bg: rgba(47, 48, 58, 0.5);
- --shadow: 0 0 10px 0 rgba(0, 0, 0, 0.1);
- }
+ :root {
+ --text: #9ca3af;
+ --text-h: #f3f4f6;
+ --bg: #16171d;
+ --border: #2e303a;
+ --code-bg: #1f2028;
+ --accent: #c084fc;
+ --accent-bg: rgba(192, 132, 252, 0.15);
+ --accent-border: rgba(192, 132, 252, 0.5);
+ --social-bg: rgba(47, 48, 58, 0.5);
+ --shadow: 0 0 10px 0 rgba(0, 0, 0, 0.1);
+ }
- #social .button-icon {
- filter: invert(1) brightness(2);
- }
+ #social .button-icon {
+ filter: invert(1) brightness(2);
+ }
}
#root {
- max-width: 100%;
- margin: 0 auto;
- text-align: center;
- border-inline: 1px solid var(--border);
- min-height: 100svh;
- display: flex;
- flex-direction: column;
- box-sizing: border-box;
- background: var(--bg-white);
+ max-width: 100%;
+ margin: 0 auto;
+ text-align: center;
+ border-inline: 1px solid var(--border);
+ min-height: 100svh;
+ display: flex;
+ flex-direction: column;
+ box-sizing: border-box;
+ background: var(--bg-white);
}
-*{
+* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
body {
- margin: 0;
- font-family: var(--font-family);
- box-sizing: border-box;
- margin: 0;
- padding: 0;
+ margin: 0;
+ font-family: var(--font-family);
+ box-sizing: border-box;
+ margin: 0;
+ padding: 0;
}
h1,
h2 {
- font-weight: 500;
- color: var(--text-h);
+ font-weight: 500;
+ color: var(--text-h);
}
h1 {
- font-size: 56px;
- letter-spacing: -1.68px;
- margin: 32px 0;
- @media (max-width: 1024px) {
- font-size: 36px;
- margin: 20px 0;
- }
+ font-size: 56px;
+ letter-spacing: -1.68px;
+ margin: 32px 0;
+
+ @media (max-width: 1024px) {
+ font-size: 36px;
+ margin: 20px 0;
+ }
}
+
h2 {
- font-size: 24px;
- line-height: 118%;
- letter-spacing: -0.24px;
- margin: 0 0 8px;
- @media (max-width: 1024px) {
- font-size: 20px;
- }
+ font-size: 24px;
+ line-height: 118%;
+ letter-spacing: -0.24px;
+ margin: 0 0 8px;
+
+ @media (max-width: 1024px) {
+ font-size: 20px;
+ }
}
+
p {
- margin: 0;
+ margin: 0;
}
code,
.counter {
- display: inline-flex;
- border-radius: 4px;
- color: var(--text-h);
+ display: inline-flex;
+ border-radius: 4px;
+ color: var(--text-h);
}
code {
- font-size: 15px;
- line-height: 135%;
- padding: 4px 8px;
- background: var(--code-bg);
+ font-size: 15px;
+ line-height: 135%;
+ padding: 4px 8px;
+ background: var(--code-bg);
}
/* Pagination */
@@ -161,7 +168,7 @@ html body .pagination .page-item.active .page-link {
color: var(--text-primary-color) !important;
}
-.page-link{
+.page-link {
color: var(--text-neutral-color) !important;
}
@@ -169,10 +176,7 @@ input:focus,
textarea:focus,
select:focus,
button:focus {
- outline: none !important;
- box-shadow: none !important;
- border: none;
-}
-
-
-
+ outline: none !important;
+ box-shadow: none !important;
+ border: none;
+}
\ No newline at end of file
diff --git a/frontend-plataforma-tutoriais/src/main.tsx b/frontend-plataforma-tutoriais/src/main.tsx
index b797766..18ca7dd 100644
--- a/frontend-plataforma-tutoriais/src/main.tsx
+++ b/frontend-plataforma-tutoriais/src/main.tsx
@@ -3,6 +3,7 @@ import { createRoot } from 'react-dom/client'
import { RouterProvider } from 'react-router'
import {router} from './routes'
import './index.css'
+import 'bootstrap-icons/font/bootstrap-icons.css';
createRoot(document.getElementById('root')!).render(
diff --git a/frontend-plataforma-tutoriais/src/pages/private/_layout.tsx b/frontend-plataforma-tutoriais/src/pages/private/_layout.tsx
index e4c8823..8c3610e 100644
--- a/frontend-plataforma-tutoriais/src/pages/private/_layout.tsx
+++ b/frontend-plataforma-tutoriais/src/pages/private/_layout.tsx
@@ -1,9 +1,9 @@
-import { Link, Outlet, useNavigate } from "react-router";
+import { Outlet, useNavigate } from "react-router";
import styles from "./styles.module.css";
import Header from "../../components/header";
import Sidebar from "../../components/sidebar";
-import Footer from "../../components/footer";
import { useEffect } from "react";
+import Chatbot from "../../components/chatbot";
export default function ProtectedLayout() {
@@ -29,8 +29,11 @@ export default function ProtectedLayout() {
+
+
{/* */}
+
>
);
diff --git a/frontend-plataforma-tutoriais/src/pages/private/admin/createUser/styles.module.css b/frontend-plataforma-tutoriais/src/pages/private/admin/createUser/styles.module.css
index 95f234e..f280ba8 100644
--- a/frontend-plataforma-tutoriais/src/pages/private/admin/createUser/styles.module.css
+++ b/frontend-plataforma-tutoriais/src/pages/private/admin/createUser/styles.module.css
@@ -24,6 +24,7 @@
.LinkIcon, .linkText{
color: var(--text-black);
+ font-size: var(--size-font-text);
}
.button:hover .LinkIcon, .button:hover .linkText{
diff --git a/frontend-plataforma-tutoriais/src/pages/private/admin/createVideo/index.tsx b/frontend-plataforma-tutoriais/src/pages/private/admin/createVideo/index.tsx
index 9005230..bd1d1c4 100644
--- a/frontend-plataforma-tutoriais/src/pages/private/admin/createVideo/index.tsx
+++ b/frontend-plataforma-tutoriais/src/pages/private/admin/createVideo/index.tsx
@@ -19,6 +19,7 @@ export default function CreateVideo() {
const [checkingRole, setCheckingRole] = useState(true);
const [creating, setCreating] = useState(false);
const [categories, setCategories] = useState