Files
ERitors-Scribe-Desktop/app/page.tsx
natreex 9e51cc93a8 Remove SyncService and introduce context-based offline mode and state management
- Delete `SyncService` and its associated bidirectional synchronization logic.
- Add multiple context providers (`OfflineProvider`, `AlertProvider`, `LangContext`, `UserContext`, `SessionContext`, `WorldContext`, `SettingBookContext`) for contextual state management.
- Implement `SecureStorage` for OS-level secure data encryption and replace dependency on `SyncService` synchronization.
- Update localization files (`en.json`, `fr.json`) with offline mode and error-related strings.
2025-11-19 22:01:24 -05:00

552 lines
23 KiB
TypeScript

'use client';
import {useContext, useEffect, useState} from 'react';
import {BookContext} from "@/context/BookContext";
import {ChapterProps} from "@/lib/models/Chapter";
import {ChapterContext} from '@/context/ChapterContext';
import {EditorContext} from '@/context/EditorContext'
import {Editor, useEditor} from "@tiptap/react";
import StarterKit from "@tiptap/starter-kit";
import Underline from "@tiptap/extension-underline";
import TextAlign from "@tiptap/extension-text-align";
import {AlertContext, AlertProvider} from "@/context/AlertContext";
import System from "@/lib/models/System";
import {SessionContext} from '@/context/SessionContext';
import {SessionProps} from "@/lib/models/Session";
import User, {UserProps} from "@/lib/models/User";
import {BookProps} from "@/lib/models/Book";
// Removed Next.js router imports for Electron
import ScribeTopBar from "@/components/ScribeTopBar";
import ScribeControllerBar from "@/components/ScribeControllerBar";
import ScribeLeftBar from "@/components/leftbar/ScribeLeftBar";
import ScribeEditor from "@/components/editor/ScribeEditor";
import ComposerRightBar from "@/components/rightbar/ComposerRightBar";
import ScribeFooterBar from "@/components/ScribeFooterBar";
import GuideTour, {GuideStep} from "@/components/GuideTour";
import {FontAwesomeIcon} from "@fortawesome/react-fontawesome";
import {faBookMedical, faFeather} from "@fortawesome/free-solid-svg-icons";
import TermsOfUse from "@/components/TermsOfUse";
import frMessages from '@/lib/locales/fr.json';
import enMessages from '@/lib/locales/en.json';
// Removed Next.js Image import for Electron
import {NextIntlClientProvider, useTranslations} from "next-intl";
import {LangContext} from "@/context/LangContext";
import {AIUsageContext} from "@/context/AIUsageContext";
import OfflineProvider from "@/context/OfflineProvider";
import OfflineContext from "@/context/OfflineContext";
import OfflinePinSetup from "@/components/offline/OfflinePinSetup";
import OfflinePinVerify from "@/components/offline/OfflinePinVerify";
const messagesMap = {
fr: frMessages,
en: enMessages
};
function ScribeContent() {
const t = useTranslations();
const {lang: locale} = useContext(LangContext);
const {errorMessage} = useContext(AlertContext);
const {initializeDatabase} = useContext(OfflineContext);
const editor: Editor | null = useEditor({
extensions: [
StarterKit,
Underline,
TextAlign.configure({
types: ['heading', 'paragraph'],
}),
],
injectCSS: false,
immediatelyRender: false,
});
// Router removed for Electron - using window.location instead
const [session, setSession] = useState<SessionProps>({user: null, accessToken: '', isConnected: false});
const [currentChapter, setCurrentChapter] = useState<ChapterProps | undefined>(undefined);
const [currentBook, setCurrentBook] = useState<BookProps | null>(null);
const [currentCredits, setCurrentCredits] = useState<number>(160);
const [amountSpent, setAmountSpent] = useState<number>(session.user?.aiUsage || 0);
const [isLoading, setIsLoading] = useState<boolean>(true);
const [sessionAttempts, setSessionAttempts] = useState<number>(0)
const [isTermsAccepted, setIsTermsAccepted] = useState<boolean>(false);
const [homeStepsGuide, setHomeStepsGuide] = useState<boolean>(false);
const [showPinSetup, setShowPinSetup] = useState<boolean>(false);
const [showPinVerify, setShowPinVerify] = useState<boolean>(false);
const homeSteps: GuideStep[] = [
{
id: 0,
x: 50,
y: 50,
title: t("homePage.guide.welcome", {name: session.user?.name || ''}),
content: (
<div>
<p>{t("homePage.guide.step0.description1")}</p>
<br/>
<p>{t("homePage.guide.step0.description2")}</p>
</div>
),
},
{
id: 1, position: 'right',
targetSelector: `[data-guide="left-panel-container"]`,
title: t("homePage.guide.step1.title"),
content: (
<div>
<p className={'flex items-center space-x-2'}>
<strong>
<FontAwesomeIcon icon={faBookMedical} className={'w-5 h-5'}/> :
</strong>
{t("homePage.guide.step1.addBook")}
</p>
<br/>
<p><strong><FontAwesomeIcon icon={faFeather}
className={'w-5 h-5'}/> :</strong> {t("homePage.guide.step1.generateStory")}
</p>
</div>
),
},
{
id: 2,
title: t("homePage.guide.step2.title"), position: 'bottom',
targetSelector: `[data-guide="search-bar"]`,
content: (
<div>
<p>{t("homePage.guide.step2.description")}</p>
</div>
),
},
{
id: 3,
title: t("homePage.guide.step3.title"),
targetSelector: `[data-guide="user-dropdown"]`,
position: 'auto',
content: (
<div>
<p>{t("homePage.guide.step3.description")}</p>
</div>
),
},
{
id: 4,
title: t("homePage.guide.step4.title"),
content: (
<div>
<p>{t("homePage.guide.step4.description1")}</p>
<br/>
<p>{t("homePage.guide.step4.description2")}</p>
</div>
),
},
];
useEffect((): void => {
checkAuthentification().then()
}, []);
useEffect((): void => {
if (session.isConnected) {
setIsTermsAccepted(session.user?.termsAccepted ?? false);
setHomeStepsGuide(User.guideTourDone(session.user?.guideTour ?? [], 'home-basic'));
setIsLoading(false);
} else {
if (sessionAttempts > 2) {
// Redirect handled by checkAuthentification
}
}
setSessionAttempts(sessionAttempts + 1);
}, [session]);
useEffect((): void => {
if (currentBook) {
getLastChapter().then();
}
}, [currentBook]);
// Check for PIN setup after successful connection
useEffect(() => {
async function checkPinSetup() {
if (session.isConnected && window.electron) {
try {
const offlineStatus = await window.electron.offlineModeGet();
console.log('[Page] Session connected, offline status:', offlineStatus);
if (!offlineStatus.hasPin) {
console.log('[Page] No PIN configured, will show setup dialog');
// Show PIN setup dialog after a short delay
setTimeout(() => {
console.log('[Page] Showing PIN setup dialog');
setShowPinSetup(true);
}, 2000); // 2 seconds delay after page load
}
} catch (error) {
console.error('[Page] Error checking offline mode:', error);
}
}
}
checkPinSetup();
}, [session.isConnected]); // Run when session connection status changes
async function handlePinVerifySuccess(userId: string): Promise<void> {
console.log('[OfflinePin] PIN verified successfully for user:', userId);
try {
// Initialize database with user's encryption key
if (window.electron) {
const encryptionKey:string|null = await window.electron.getUserEncryptionKey(userId);
if (encryptionKey) {
await window.electron.dbInitialize(userId, encryptionKey);
// Load user from local DB
const localUser = await window.electron.invoke('db:user:info');
if (localUser && localUser.success) {
// Use local data and continue in offline mode
setSession({
isConnected: true,
user: localUser.data,
accessToken: 'offline', // Special offline token
});
setShowPinVerify(false);
setCurrentCredits(localUser.data.creditsBalance || 0);
setAmountSpent(localUser.data.aiUsage || 0);
console.log('[OfflinePin] Running in offline mode');
} else {
errorMessage(t("homePage.errors.localDataError"));
if (window.electron) {
//window.electron.logout();
}
}
} else {
errorMessage(t("homePage.errors.encryptionKeyError"));
if (window.electron) {
//window.electron.logout();
}
}
}
} catch (error) {
console.error('[OfflinePin] Error initializing offline mode:', error);
errorMessage(t("homePage.errors.offlineModeError"));
if (window.electron) {
//window.electron.logout();
}
}
}
async function handleHomeTour(): Promise<void> {
try {
const response: boolean = await System.authPostToServer<boolean>('logs/tour', {
plateforme: 'web',
tour: 'home-basic'
},
session.accessToken,
locale
);
if (response) {
setSession(User.setNewGuideTour(session, 'home-basic'));
setHomeStepsGuide(false);
}
} catch (e: unknown) {
if (e instanceof Error) {
errorMessage(e.message);
} else {
errorMessage(t("homePage.errors.termsError"));
}
}
}
async function checkAuthentification(): Promise<void> {
let token: string | null = null;
if (typeof window !== 'undefined' && window.electron) {
try {
token = await window.electron.getToken();
} catch (e) {
console.error('Error getting token from electron:', e);
}
}
if (token) {
try {
const user: UserProps = await System.authGetQueryToServer<UserProps>('user/infos', token, locale);
if (!user) {
errorMessage(t("homePage.errors.userNotFound"));
// Token invalide, supprimer et logout
if (window.electron) {
await window.electron.removeToken();
window.electron.logout();
}
return;
}
console.log('user: ' , user);
// Initialize user in Electron (sets userId and creates/gets encryption key)
if (window.electron && user.id) {
try {
const initResult = await window.electron.initUser(user.id);
if (!initResult.success) {
console.error('[Page] Failed to initialize user:', initResult.error);
} else {
console.log('[Page] User initialized successfully, key created:', initResult.keyCreated);
// Check if PIN is configured for offline mode
try {
const offlineStatus = await window.electron.offlineModeGet();
console.log('[Page] Offline status:', offlineStatus);
if (!offlineStatus.hasPin) {
// First login or no PIN configured yet
// Show PIN setup dialog after a short delay
console.log('[Page] No PIN configured, will show setup dialog');
setTimeout(() => {
console.log('[Page] Showing PIN setup dialog');
setShowPinSetup(true);
}, 2000); // 2 seconds delay after successful login
}
} catch (error) {
console.error('[Page] Error checking offline mode:', error);
}
}
} catch (error) {
console.error('[Page] Error initializing user:', error);
}
}
setSession({
isConnected: true,
user: user,
accessToken: token,
});
setCurrentCredits(user.creditsBalance)
setAmountSpent(user.aiUsage)
// Initialiser la DB locale en Electron
if (window.electron && user.id) {
try {
const dbInitialized = await initializeDatabase(user.id);
if (dbInitialized) {
console.log('Database initialized successfully');
// Sync user to local DB (only if not exists)
try {
await window.electron.invoke('db:user:sync', {
userId: user.id,
firstName: user.name,
lastName: user.lastName,
username: user.username,
email: user.email
});
console.log('User synced to local DB');
} catch (syncError) {
console.error('Failed to sync user to local DB:', syncError);
// Non-blocking error, continue anyway
}
}
} catch (error) {
console.error('Failed to initialize database:', error);
}
}
} catch (e: unknown) {
console.log('[Auth] Server error, checking offline mode...');
if (window.electron) {
try {
const offlineStatus = await window.electron.offlineModeGet();
if (offlineStatus.hasPin && offlineStatus.lastUserId) {
console.log('[Auth] Server unreachable but PIN configured, showing PIN verification');
setShowPinVerify(true);
setIsLoading(false);
return;
} else {
if (window.electron) {
await window.electron.removeToken();
window.electron.logout();
}
}
} catch (offlineError) {
console.error('[Auth] Error checking offline mode:', offlineError);
}
}
// If not in offline mode or no PIN configured, show error and logout
if (e instanceof Error) {
errorMessage(e.message);
} else {
errorMessage(t("homePage.errors.authenticationError"));
}
}
} else {
if (window.electron) {
try {
const offlineStatus = await window.electron.offlineModeGet();
if (offlineStatus.hasPin && offlineStatus.lastUserId) {
console.log('[Auth] No token but PIN configured, showing PIN verification for offline mode');
setShowPinVerify(true);
setIsLoading(false);
return;
}
} catch (error) {
console.error('[Auth] Error checking offline mode:', error);
}
window.electron.logout();
}
}
}
async function handleTermsAcceptance(): Promise<void> {
try {
const response: boolean = await System.authPostToServer<boolean>(`user/terms/accept`, {
version: '2025-07-1'
}, session.accessToken, locale);
if (response) {
setIsTermsAccepted(true);
setHomeStepsGuide(true);
const newSession: SessionProps = {
...session,
user: {
...session?.user as UserProps,
termsAccepted: true
}
}
setSession(newSession);
} else {
errorMessage(t("homePage.errors.termsAcceptError"));
}
} catch (e: unknown) {
if (e instanceof Error) {
errorMessage(e.message);
} else {
errorMessage(t("homePage.errors.termsAcceptError"));
}
}
}
async function getLastChapter(): Promise<void> {
if (session?.accessToken) {
try {
const response: ChapterProps | null = await System.authGetQueryToServer<ChapterProps | null>(`chapter/last-chapter`, session.accessToken, locale, {bookid: currentBook?.bookId});
if (response) {
setCurrentChapter(response)
} else {
setCurrentChapter(undefined);
}
} catch (e: unknown) {
if (e instanceof Error) {
errorMessage(e.message);
} else {
errorMessage(t("homePage.errors.lastChapterError"));
}
}
}
}
if (isLoading) {
return (
<div
className="bg-background text-text-primary h-screen flex flex-col items-center justify-center font-['Lora']">
<div className="flex flex-col items-center space-y-6">
<div className="animate-pulse">
<img src="/eritors-favicon-white.png" alt="ERitors Logo" style={{width: 400, height: 400}} />
</div>
<div className="flex space-x-2">
<div className="w-2 h-2 bg-primary rounded-full animate-bounce"></div>
<div className="w-2 h-2 bg-primary rounded-full animate-bounce delay-100"></div>
<div className="w-2 h-2 bg-primary rounded-full animate-bounce delay-200"></div>
</div>
<p className="text-text-secondary text-sm">
{t("homePage.loading")}
</p>
</div>
</div>
)
}
return (
<SessionContext.Provider value={{session: session, setSession: setSession}}>
<BookContext.Provider value={{book: currentBook, setBook: setCurrentBook}}>
<ChapterContext.Provider value={{chapter: currentChapter, setChapter: setCurrentChapter}}>
<AIUsageContext.Provider value={{
totalCredits: currentCredits,
setTotalCredits: setCurrentCredits,
totalPrice: amountSpent,
setTotalPrice: setAmountSpent
}}>
<div
className="bg-background text-text-primary h-screen flex flex-col font-['Lora']">
<ScribeTopBar/>
<EditorContext.Provider value={{editor: editor}}>
<ScribeControllerBar/>
<div className="flex-1 flex overflow-hidden">
<ScribeLeftBar/>
<ScribeEditor/>
<ComposerRightBar/>
</div>
<ScribeFooterBar/>
</EditorContext.Provider>
</div>
{
homeStepsGuide &&
<GuideTour stepId={0} steps={homeSteps} onComplete={handleHomeTour}
onClose={(): void => setHomeStepsGuide(false)}/>
}
{
!isTermsAccepted && <TermsOfUse onAccept={handleTermsAcceptance}/>
}
{
showPinSetup && window.electron && (
<OfflinePinSetup
showOnFirstLogin={true}
onClose={() => setShowPinSetup(false)}
onSuccess={() => {
setShowPinSetup(false);
console.log('[Page] PIN configured successfully');
}}
/>
)
}
{
showPinVerify && window.electron && (
<OfflinePinVerify
onSuccess={handlePinVerifySuccess}
onCancel={():void => {
//window.electron.logout();
}}
/>
)
}
</AIUsageContext.Provider>
</ChapterContext.Provider>
</BookContext.Provider>
</SessionContext.Provider>
);
}
export default function Scribe() {
const [locale, setLocale] = useState<'fr' | 'en'>('fr');
useEffect((): void => {
const lang: "fr" | "en" | null = System.getCookie('lang') as "fr" | "en" | null;
if (lang) {
setLocale(lang);
}
}, []);
const messages = messagesMap[locale];
return (
<LangContext.Provider value={{lang: locale, setLang: setLocale}}>
<NextIntlClientProvider locale={locale} messages={messages}>
<OfflineProvider>
<AlertProvider>
<ScribeContent/>
</AlertProvider>
</OfflineProvider>
</NextIntlClientProvider>
</LangContext.Provider>
);
}