From c40d10f5f5809970aa2ea5f6f8b4b5b8649045fe Mon Sep 17 00:00:00 2001 From: MTRNord Date: Sat, 11 Jan 2025 18:39:19 +0100 Subject: [PATCH] Improve the output and allow logout --- src/app/api/redirect/route.ts | 10 ++++++---- src/app/importActions.ts | 13 ++++++++----- src/app/logoutAction.ts | 9 +++++++++ src/app/logoutComponent.tsx | 16 ++++++++++++++++ src/app/page.tsx | 11 +++++++++-- src/utils/initMiroAPI.ts | 12 ++++++++++++ 6 files changed, 60 insertions(+), 11 deletions(-) create mode 100644 src/app/logoutAction.ts create mode 100644 src/app/logoutComponent.tsx diff --git a/src/app/api/redirect/route.ts b/src/app/api/redirect/route.ts index 3523d38..b4aeb0b 100644 --- a/src/app/api/redirect/route.ts +++ b/src/app/api/redirect/route.ts @@ -1,10 +1,12 @@ -import type {NextRequest} from 'next/server'; -import {redirect} from 'next/navigation'; +import type { NextRequest } from 'next/server'; +import { redirect } from 'next/navigation'; import initMiroAPI from '../../../utils/initMiroAPI'; // handle redirect with code and exchange it for the access token export async function GET(request: NextRequest) { - const {miro, userId} = initMiroAPI(); + const { miro, userId } = initMiroAPI(); + + const awaitedUserId = await userId(); // Make sure the code is in query parameters const code = request.nextUrl.searchParams.get('code'); @@ -14,7 +16,7 @@ export async function GET(request: NextRequest) { } try { - await miro.exchangeCodeForAccessToken(userId, code); + await miro.exchangeCodeForAccessToken(awaitedUserId, code); } catch (error) { redirect('/?error'); } diff --git a/src/app/importActions.ts b/src/app/importActions.ts index a959a61..bb4d231 100644 --- a/src/app/importActions.ts +++ b/src/app/importActions.ts @@ -1,6 +1,6 @@ 'use server'; -import { AppCardItem, CardItem, DocumentItem, EmbedItem, FrameItem, ImageItem, Item, ShapeItem, StickyNoteItem, TextItem } from '@mirohq/miro-api'; +import { AppCardItem, Board, CardItem, DocumentItem, EmbedItem, FrameItem, ImageItem, Item, ShapeItem, StickyNoteItem, TextItem } from '@mirohq/miro-api'; import initMiroAPI from '../utils/initMiroAPI'; import { Path, Shape, Slide, Whiteboard, Image, neoboardWhiteboardWidth, neoboardWhiteboardHeight } from './neoboardTypes'; import { stripHtml } from 'string-strip-html'; @@ -81,7 +81,7 @@ export async function importBoard(prevState: FormState, formData: FormData): Pro if (item.parent.id === frameId) { switch (item.type) { case 'shape': - const shape = convertShape(item as ShapeItem); + const shape = await convertShape(item as ShapeItem, board); if (shape) { slide.elements.push(shape); } @@ -196,7 +196,7 @@ function transformBoardIntoNeoboardCoordinates(board: Whiteboard): Whiteboard { return transformedBoard; } -function convertShape(item: ShapeItem): Shape | undefined { +async function convertShape(item: ShapeItem, board: Board): Promise { let shapeType = item.data?.shape; let borderRadius = 0; if (shapeType !== 'rectangle' && shapeType !== 'circle' && shapeType !== 'ellipse' && shapeType !== 'triangle' && shapeType !== 'round_rectangle') { @@ -205,9 +205,12 @@ function convertShape(item: ShapeItem): Shape | undefined { } if (shapeType === 'round_rectangle') { shapeType = 'rectangle'; - borderRadius = 10; + borderRadius = 25; } + // Get shape item from Miro API + item = await board.getItem(item.id) as ShapeItem; + const opacity = item.style?.fillOpacity ?? (item.style?.fillColor ? "1.0" : "0.0"); const borderColorOpacity = item.style?.borderOpacity ?? "1.0"; @@ -325,7 +328,7 @@ function convertStickyNote(item: StickyNoteItem): Shape { fillColor: item.style?.fillColor ?? "#ffffff", strokeColor: 'transparent', strokeWidth: 0, - borderRadius: 10, + borderRadius: 25, text: stripHtml(item.data?.content ?? "").result, textAlignment: (item.style?.textAlign ?? "left") as "left" | "center" | "right", textColor: 'black', diff --git a/src/app/logoutAction.ts b/src/app/logoutAction.ts new file mode 100644 index 0000000..a69ec7b --- /dev/null +++ b/src/app/logoutAction.ts @@ -0,0 +1,9 @@ +"use server"; + +import { redirect } from "next/navigation"; +import { logoutMiroAPI } from "../utils/initMiroAPI"; + +export default async function logout() { + await logoutMiroAPI(); + redirect("/"); +} \ No newline at end of file diff --git a/src/app/logoutComponent.tsx b/src/app/logoutComponent.tsx new file mode 100644 index 0000000..25eb1da --- /dev/null +++ b/src/app/logoutComponent.tsx @@ -0,0 +1,16 @@ +"use server"; + +import logout from "./logoutAction"; + +export default async function Logout() { + return ( +
+

Logout

+
+ +
+
+ ) +} \ No newline at end of file diff --git a/src/app/page.tsx b/src/app/page.tsx index f9b146a..2167194 100644 --- a/src/app/page.tsx +++ b/src/app/page.tsx @@ -5,6 +5,7 @@ import { Board } from '@mirohq/miro-api'; import initMiroAPI from '../utils/initMiroAPI'; import '../assets/style.scss'; import SelectionFormComponent from './selectionFormComponent'; +import Logout from './logoutComponent'; const getBoards = async () => { const { miro, userId } = initMiroAPI(); @@ -41,7 +42,7 @@ export default async function Page() { return (

Login

- + Login
@@ -52,5 +53,11 @@ export default async function Page() { return
Loading...
; } - return (<>); + return ( + <> + +
+ + + ); } diff --git a/src/utils/initMiroAPI.ts b/src/utils/initMiroAPI.ts index f39be1d..7acce7c 100644 --- a/src/utils/initMiroAPI.ts +++ b/src/utils/initMiroAPI.ts @@ -4,6 +4,18 @@ import { State } from '@mirohq/miro-api/dist/storage'; const tokensCookie = 'miro_tokens'; +export async function logoutMiroAPI() { + const { miro, userId } = initMiroAPI(); + const userIdAwaited = await userId(); + + if (userIdAwaited) { + await miro.as(userIdAwaited).revokeToken(); + } + + const cookieInstance = await cookies(); + cookieInstance.delete('miro_tokens'); +} + export default function initMiroAPI() { const cookieInstance = cookies(); -- 2.45.2