prettier fix

This commit is contained in:
Илья Глазунов 2026-01-15 23:26:09 +03:00
parent 528dfe01a1
commit 669c019fd8
21 changed files with 4857 additions and 4639 deletions

View File

@ -10,16 +10,16 @@ packaging.
Install dependencies and run the development app: Install dependencies and run the development app:
``` ```
1. npm install 1. pnpm install
2. npm start 2. pnpm start
``` ```
## Style ## Style
Run `npx prettier --write .` before committing. Prettier uses the settings in Run `pnpm prettier --write .` before committing. Prettier uses the settings in
`.prettierrc` (four-space indentation, print width 150, semicolons and single `.prettierrc` (four-space indentation, print width 150, semicolons and single
quotes). `src/assets` and `node_modules` are ignored via `.prettierignore`. quotes). `src/assets` and `node_modules` are ignored via `.prettierignore`.
The project does not provide linting; `npm run lint` simply prints The project does not provide linting; `pnpm run lint` simply prints
"No linting configured". "No linting configured".
## Code standards ## Code standards

View File

@ -76,8 +76,8 @@ module.exports = {
genericName: 'AI Assistant', genericName: 'AI Assistant',
description: 'AI assistant for interviews and learning', description: 'AI assistant for interviews and learning',
categories: ['Development', 'Education'], categories: ['Development', 'Education'],
icon: 'src/assets/logo.png' icon: 'src/assets/logo.png',
} },
}, },
}, },
], ],

3
pnpm-lock.yaml generated
View File

@ -5,7 +5,6 @@ settings:
excludeLinksFromLockfile: false excludeLinksFromLockfile: false
importers: importers:
.: .:
dependencies: dependencies:
'@google/genai': '@google/genai':
@ -59,7 +58,6 @@ importers:
version: 39.2.7 version: 39.2.7
packages: packages:
'@electron-forge/cli@7.11.1': '@electron-forge/cli@7.11.1':
resolution: { integrity: sha512-pk8AoLsr7t7LBAt0cFD06XFA6uxtPdvtLx06xeal7O9o7GHGCbj29WGwFoJ8Br/ENM0Ho868S3PrAn1PtBXt5g== } resolution: { integrity: sha512-pk8AoLsr7t7LBAt0cFD06XFA6uxtPdvtLx06xeal7O9o7GHGCbj29WGwFoJ8Br/ENM0Ho868S3PrAn1PtBXt5g== }
engines: { node: '>= 16.4.0' } engines: { node: '>= 16.4.0' }
@ -2176,7 +2174,6 @@ packages:
engines: { node: '>=18' } engines: { node: '>=18' }
snapshots: snapshots:
'@electron-forge/cli@7.11.1(encoding@0.1.13)': '@electron-forge/cli@7.11.1(encoding@0.1.13)':
dependencies: dependencies:
'@electron-forge/core': 7.11.1(encoding@0.1.13) '@electron-forge/core': 7.11.1(encoding@0.1.13)

View File

@ -3,7 +3,11 @@ import { html, css, LitElement } from '../../assets/lit-core-2.7.4.min.js';
export class AppHeader extends LitElement { export class AppHeader extends LitElement {
static styles = css` static styles = css`
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
user-select: none; user-select: none;
} }
@ -155,7 +159,9 @@ export class AppHeader extends LitElement {
white-space: normal; white-space: normal;
opacity: 0; opacity: 0;
visibility: hidden; visibility: hidden;
transition: opacity 0.15s ease, visibility 0.15s ease; transition:
opacity 0.15s ease,
visibility 0.15s ease;
pointer-events: none; pointer-events: none;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
z-index: 1000; z-index: 1000;
@ -224,7 +230,9 @@ export class AppHeader extends LitElement {
white-space: nowrap; white-space: nowrap;
opacity: 0; opacity: 0;
visibility: hidden; visibility: hidden;
transition: opacity 0.15s ease, visibility 0.15s ease; transition:
opacity 0.15s ease,
visibility 0.15s ease;
pointer-events: none; pointer-events: none;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
z-index: 1000; z-index: 1000;
@ -421,7 +429,7 @@ export class AppHeader extends LitElement {
const names = { const names = {
'gemini': 'Gemini', 'gemini': 'Gemini',
'openai-realtime': 'OpenAI Realtime', 'openai-realtime': 'OpenAI Realtime',
'openai-sdk': 'OpenAI SDK' 'openai-sdk': 'OpenAI SDK',
}; };
return names[this.aiProvider] || this.aiProvider; return names[this.aiProvider] || this.aiProvider;
} }
@ -471,39 +479,59 @@ export class AppHeader extends LitElement {
<span>${elapsedTime}</span> <span>${elapsedTime}</span>
<div class="status-wrapper"> <div class="status-wrapper">
<span class="status-text ${isError ? 'error' : ''}">${shortStatus}</span> <span class="status-text ${isError ? 'error' : ''}">${shortStatus}</span>
${isError ? html` ${isError
? html`
<div class="status-tooltip"> <div class="status-tooltip">
<div class="tooltip-label">Error Details</div> <div class="tooltip-label">Error Details</div>
<div class="tooltip-content">${this.statusText}</div> <div class="tooltip-content">${this.statusText}</div>
</div> </div>
` : ''} `
: ''}
</div> </div>
${this.isClickThrough ? html`<span class="click-through-indicator">click-through</span>` : ''} ${this.isClickThrough ? html`<span class="click-through-indicator">click-through</span>` : ''}
` `
: ''} : ''}
${this.currentView === 'main' ${this.currentView === 'main'
? html` ? html`
${this.updateAvailable ? html` ${this.updateAvailable
? html`
<button class="update-button" @click=${this._openUpdatePage}> <button class="update-button" @click=${this._openUpdatePage}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" fill="currentColor">
<path fill-rule="evenodd" d="M13.836 2.477a.75.75 0 0 1 .75.75v3.182a.75.75 0 0 1-.75.75h-3.182a.75.75 0 0 1 0-1.5h1.37l-.84-.841a4.5 4.5 0 0 0-7.08.932.75.75 0 0 1-1.3-.75 6 6 0 0 1 9.44-1.242l.842.84V3.227a.75.75 0 0 1 .75-.75Zm-.911 7.5A.75.75 0 0 1 13.199 11a6 6 0 0 1-9.44 1.241l-.84-.84v1.371a.75.75 0 0 1-1.5 0V9.591a.75.75 0 0 1 .75-.75H5.35a.75.75 0 0 1 0 1.5H3.98l.841.841a4.5 4.5 0 0 0 7.08-.932.75.75 0 0 1 1.025-.273Z" clip-rule="evenodd" /> <path
fill-rule="evenodd"
d="M13.836 2.477a.75.75 0 0 1 .75.75v3.182a.75.75 0 0 1-.75.75h-3.182a.75.75 0 0 1 0-1.5h1.37l-.84-.841a4.5 4.5 0 0 0-7.08.932.75.75 0 0 1-1.3-.75 6 6 0 0 1 9.44-1.242l.842.84V3.227a.75.75 0 0 1 .75-.75Zm-.911 7.5A.75.75 0 0 1 13.199 11a6 6 0 0 1-9.44 1.241l-.84-.84v1.371a.75.75 0 0 1-1.5 0V9.591a.75.75 0 0 1 .75-.75H5.35a.75.75 0 0 1 0 1.5H3.98l.841.841a4.5 4.5 0 0 0 7.08-.932.75.75 0 0 1 1.025-.273Z"
clip-rule="evenodd"
/>
</svg> </svg>
Update available Update available
</button> </button>
` : ''} `
: ''}
<button class="icon-button" @click=${this.onHistoryClick}> <button class="icon-button" @click=${this.onHistoryClick}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path fill-rule="evenodd" d="M10 18a8 8 0 1 0 0-16 8 8 0 0 0 0 16Zm.75-13a.75.75 0 0 0-1.5 0v5c0 .414.336.75.75.75h4a.75.75 0 0 0 0-1.5h-3.25V5Z" clip-rule="evenodd" /> <path
fill-rule="evenodd"
d="M10 18a8 8 0 1 0 0-16 8 8 0 0 0 0 16Zm.75-13a.75.75 0 0 0-1.5 0v5c0 .414.336.75.75.75h4a.75.75 0 0 0 0-1.5h-3.25V5Z"
clip-rule="evenodd"
/>
</svg> </svg>
</button> </button>
<button class="icon-button" @click=${this.onCustomizeClick}> <button class="icon-button" @click=${this.onCustomizeClick}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path fill-rule="evenodd" d="M7.84 1.804A1 1 0 0 1 8.82 1h2.36a1 1 0 0 1 .98.804l.331 1.652a6.993 6.993 0 0 1 1.929 1.115l1.598-.54a1 1 0 0 1 1.186.447l1.18 2.044a1 1 0 0 1-.205 1.251l-1.267 1.113a7.047 7.047 0 0 1 0 2.228l1.267 1.113a1 1 0 0 1 .206 1.25l-1.18 2.045a1 1 0 0 1-1.187.447l-1.598-.54a6.993 6.993 0 0 1-1.929 1.115l-.33 1.652a1 1 0 0 1-.98.804H8.82a1 1 0 0 1-.98-.804l-.331-1.652a6.993 6.993 0 0 1-1.929-1.115l-1.598.54a1 1 0 0 1-1.186-.447l-1.18-2.044a1 1 0 0 1 .205-1.251l1.267-1.114a7.05 7.05 0 0 1 0-2.227L1.821 7.773a1 1 0 0 1-.206-1.25l1.18-2.045a1 1 0 0 1 1.187-.447l1.598.54A6.992 6.992 0 0 1 7.51 3.456l.33-1.652ZM10 13a3 3 0 1 0 0-6 3 3 0 0 0 0 6Z" clip-rule="evenodd" /> <path
fill-rule="evenodd"
d="M7.84 1.804A1 1 0 0 1 8.82 1h2.36a1 1 0 0 1 .98.804l.331 1.652a6.993 6.993 0 0 1 1.929 1.115l1.598-.54a1 1 0 0 1 1.186.447l1.18 2.044a1 1 0 0 1-.205 1.251l-1.267 1.113a7.047 7.047 0 0 1 0 2.228l1.267 1.113a1 1 0 0 1 .206 1.25l-1.18 2.045a1 1 0 0 1-1.187.447l-1.598-.54a6.993 6.993 0 0 1-1.929 1.115l-.33 1.652a1 1 0 0 1-.98.804H8.82a1 1 0 0 1-.98-.804l-.331-1.652a6.993 6.993 0 0 1-1.929-1.115l-1.598.54a1 1 0 0 1-1.186-.447l-1.18-2.044a1 1 0 0 1 .205-1.251l1.267-1.114a7.05 7.05 0 0 1 0-2.227L1.821 7.773a1 1 0 0 1-.206-1.25l1.18-2.045a1 1 0 0 1 1.187-.447l1.598.54A6.992 6.992 0 0 1 7.51 3.456l.33-1.652ZM10 13a3 3 0 1 0 0-6 3 3 0 0 0 0 6Z"
clip-rule="evenodd"
/>
</svg> </svg>
</button> </button>
<button class="icon-button" @click=${this.onHelpClick}> <button class="icon-button" @click=${this.onHelpClick}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path fill-rule="evenodd" d="M18 10a8 8 0 1 1-16 0 8 8 0 0 1 16 0ZM8.94 6.94a.75.75 0 1 1-1.061-1.061 3 3 0 1 1 2.871 5.026v.345a.75.75 0 0 1-1.5 0v-.5c0-.72.57-1.172 1.081-1.287A1.5 1.5 0 1 0 8.94 6.94ZM10 15a1 1 0 1 0 0-2 1 1 0 0 0 0 2Z" clip-rule="evenodd" /> <path
fill-rule="evenodd"
d="M18 10a8 8 0 1 1-16 0 8 8 0 0 1 16 0ZM8.94 6.94a.75.75 0 1 1-1.061-1.061 3 3 0 1 1 2.871 5.026v.345a.75.75 0 0 1-1.5 0v-.5c0-.72.57-1.172 1.081-1.287A1.5 1.5 0 1 0 8.94 6.94ZM10 15a1 1 0 1 0 0-2 1 1 0 0 0 0 2Z"
clip-rule="evenodd"
/>
</svg> </svg>
</button> </button>
` `
@ -516,14 +544,18 @@ export class AppHeader extends LitElement {
</button> </button>
<button @click=${this.onCloseClick} class="icon-button window-close"> <button @click=${this.onCloseClick} class="icon-button window-close">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z" /> <path
d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z"
/>
</svg> </svg>
</button> </button>
` `
: html` : html`
<button @click=${this.isNavigationView() ? this.onBackClick : this.onCloseClick} class="icon-button window-close"> <button @click=${this.isNavigationView() ? this.onBackClick : this.onCloseClick} class="icon-button window-close">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z" /> <path
d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z"
/>
</svg> </svg>
</button> </button>
`} `}

View File

@ -12,7 +12,11 @@ export class CheatingDaddyApp extends LitElement {
static styles = css` static styles = css`
* { * {
box-sizing: border-box; box-sizing: border-box;
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
margin: 0px; margin: 0px;
padding: 0px; padding: 0px;
cursor: default; cursor: default;
@ -152,17 +156,14 @@ export class CheatingDaddyApp extends LitElement {
const [config, prefs, openaiSdkCreds] = await Promise.all([ const [config, prefs, openaiSdkCreds] = await Promise.all([
cheatingDaddy.storage.getConfig(), cheatingDaddy.storage.getConfig(),
cheatingDaddy.storage.getPreferences(), cheatingDaddy.storage.getPreferences(),
cheatingDaddy.storage.getOpenAISDKCredentials() cheatingDaddy.storage.getOpenAISDKCredentials(),
]); ]);
// Check onboarding status // Check onboarding status
this.currentView = config.onboarded ? 'main' : 'onboarding'; this.currentView = config.onboarded ? 'main' : 'onboarding';
// Apply background appearance (color + transparency) // Apply background appearance (color + transparency)
this.applyBackgroundAppearance( this.applyBackgroundAppearance(prefs.backgroundColor ?? '#1e1e1e', prefs.backgroundTransparency ?? 0.8);
prefs.backgroundColor ?? '#1e1e1e',
prefs.backgroundTransparency ?? 0.8
);
// Load preferences // Load preferences
this.selectedProfile = prefs.selectedProfile || 'interview'; this.selectedProfile = prefs.selectedProfile || 'interview';
@ -176,7 +177,7 @@ export class CheatingDaddyApp extends LitElement {
this.modelInfo = { this.modelInfo = {
model: openaiSdkCreds.model || 'gpt-4o', model: openaiSdkCreds.model || 'gpt-4o',
visionModel: openaiSdkCreds.visionModel || 'gpt-4o', visionModel: openaiSdkCreds.visionModel || 'gpt-4o',
whisperModel: openaiSdkCreds.whisperModel || 'whisper-1' whisperModel: openaiSdkCreds.whisperModel || 'whisper-1',
}; };
this._storageLoaded = true; this._storageLoaded = true;
@ -191,18 +192,20 @@ export class CheatingDaddyApp extends LitElement {
hexToRgb(hex) { hexToRgb(hex) {
const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex); const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
return result ? { return result
? {
r: parseInt(result[1], 16), r: parseInt(result[1], 16),
g: parseInt(result[2], 16), g: parseInt(result[2], 16),
b: parseInt(result[3], 16) b: parseInt(result[3], 16),
} : { r: 30, g: 30, b: 30 }; }
: { r: 30, g: 30, b: 30 };
} }
lightenColor(rgb, amount) { lightenColor(rgb, amount) {
return { return {
r: Math.min(255, rgb.r + amount), r: Math.min(255, rgb.r + amount),
g: Math.min(255, rgb.g + amount), g: Math.min(255, rgb.g + amount),
b: Math.min(255, rgb.b + amount) b: Math.min(255, rgb.b + amount),
}; };
} }
@ -525,11 +528,11 @@ export class CheatingDaddyApp extends LitElement {
render() { render() {
const viewClassMap = { const viewClassMap = {
'assistant': 'assistant-view', assistant: 'assistant-view',
'onboarding': 'onboarding-view', onboarding: 'onboarding-view',
'customize': 'settings-view', customize: 'settings-view',
'help': 'help-view', help: 'help-view',
'history': 'history-view', history: 'history-view',
}; };
const mainContentClass = `main-content ${viewClassMap[this.currentView] || 'with-border'}`; const mainContentClass = `main-content ${viewClassMap[this.currentView] || 'with-border'}`;
@ -602,7 +605,7 @@ export class CheatingDaddyApp extends LitElement {
if (result.success) { if (result.success) {
this.screenSources = result.sources; this.screenSources = result.sources;
this.showScreenPicker = true; this.showScreenPicker = true;
return new Promise((resolve) => { return new Promise(resolve => {
this._screenPickerResolve = resolve; this._screenPickerResolve = resolve;
}); });
} else { } else {

View File

@ -9,7 +9,11 @@ export class AssistantView extends LitElement {
} }
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
} }
@ -51,12 +55,24 @@ export class AssistantView extends LitElement {
font-weight: 600; font-weight: 600;
} }
.response-container h1 { font-size: 1.6em; } .response-container h1 {
.response-container h2 { font-size: 1.4em; } font-size: 1.6em;
.response-container h3 { font-size: 1.2em; } }
.response-container h4 { font-size: 1.1em; } .response-container h2 {
.response-container h5 { font-size: 1em; } font-size: 1.4em;
.response-container h6 { font-size: 0.9em; } }
.response-container h3 {
font-size: 1.2em;
}
.response-container h4 {
font-size: 1.1em;
}
.response-container h5 {
font-size: 1em;
}
.response-container h6 {
font-size: 0.9em;
}
.response-container p { .response-container p {
margin: 0.6em 0; margin: 0.6em 0;
@ -268,7 +284,9 @@ export class AssistantView extends LitElement {
white-space: nowrap; white-space: nowrap;
opacity: 0; opacity: 0;
visibility: hidden; visibility: hidden;
transition: opacity 0.15s ease, visibility 0.15s ease; transition:
opacity 0.15s ease,
visibility 0.15s ease;
pointer-events: none; pointer-events: none;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
z-index: 100; z-index: 100;
@ -655,12 +673,17 @@ export class AssistantView extends LitElement {
<div class="capture-buttons"> <div class="capture-buttons">
<button class="region-select-btn" @click=${this.handleRegionSelect} title="Select region to analyze (like Win+Shift+S)"> <button class="region-select-btn" @click=${this.handleRegionSelect} title="Select region to analyze (like Win+Shift+S)">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path fill-rule="evenodd" d="M4.25 2A2.25 2.25 0 0 0 2 4.25v2.5A.75.75 0 0 0 3.5 6.75v-2.5a.75.75 0 0 1 .75-.75h2.5A.75.75 0 0 0 6.75 2h-2.5Zm9.5 0a.75.75 0 0 0 0 1.5h2.5a.75.75 0 0 1 .75.75v2.5a.75.75 0 0 0 1.5 0v-2.5A2.25 2.25 0 0 0 16.25 2h-2.5ZM3.5 13.25a.75.75 0 0 0-1.5 0v2.5A2.25 2.25 0 0 0 4.25 18h2.5a.75.75 0 0 0 0-1.5h-2.5a.75.75 0 0 1-.75-.75v-2.5Zm13.5 0a.75.75 0 0 0 1.5 0v2.5A2.25 2.25 0 0 1 16.25 18h-2.5a.75.75 0 0 1 0-1.5h2.5a.75.75 0 0 0 .75-.75v-2.5Z" clip-rule="evenodd" /> <path
fill-rule="evenodd"
d="M4.25 2A2.25 2.25 0 0 0 2 4.25v2.5A.75.75 0 0 0 3.5 6.75v-2.5a.75.75 0 0 1 .75-.75h2.5A.75.75 0 0 0 6.75 2h-2.5Zm9.5 0a.75.75 0 0 0 0 1.5h2.5a.75.75 0 0 1 .75.75v2.5a.75.75 0 0 0 1.5 0v-2.5A2.25 2.25 0 0 0 16.25 2h-2.5ZM3.5 13.25a.75.75 0 0 0-1.5 0v2.5A2.25 2.25 0 0 0 4.25 18h2.5a.75.75 0 0 0 0-1.5h-2.5a.75.75 0 0 1-.75-.75v-2.5Zm13.5 0a.75.75 0 0 0 1.5 0v2.5A2.25 2.25 0 0 1 16.25 18h-2.5a.75.75 0 0 1 0-1.5h2.5a.75.75 0 0 0 .75-.75v-2.5Z"
clip-rule="evenodd"
/>
</svg> </svg>
<span>Select region</span> <span>Select region</span>
</button> </button>
<div class="screen-answer-btn-wrapper"> <div class="screen-answer-btn-wrapper">
${this.aiProvider === 'gemini' ? html` ${this.aiProvider === 'gemini'
? html`
<div class="tooltip"> <div class="tooltip">
<div class="tooltip-row"> <div class="tooltip-row">
<span class="tooltip-label">Flash</span> <span class="tooltip-label">Flash</span>
@ -672,13 +695,18 @@ export class AssistantView extends LitElement {
</div> </div>
<div class="tooltip-note">Resets every 24 hours</div> <div class="tooltip-note">Resets every 24 hours</div>
</div> </div>
` : ''} `
: ''}
<button class="screen-answer-btn" @click=${this.handleScreenAnswer}> <button class="screen-answer-btn" @click=${this.handleScreenAnswer}>
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path d="M15.98 1.804a1 1 0 0 0-1.96 0l-.24 1.192a1 1 0 0 1-.784.785l-1.192.238a1 1 0 0 0 0 1.962l1.192.238a1 1 0 0 1 .785.785l.238 1.192a1 1 0 0 0 1.962 0l.238-1.192a1 1 0 0 1 .785-.785l1.192-.238a1 1 0 0 0 0-1.962l-1.192-.238a1 1 0 0 1-.785-.785l-.238-1.192ZM6.949 5.684a1 1 0 0 0-1.898 0l-.683 2.051a1 1 0 0 1-.633.633l-2.051.683a1 1 0 0 0 0 1.898l2.051.684a1 1 0 0 1 .633.632l.683 2.051a1 1 0 0 0 1.898 0l.683-2.051a1 1 0 0 1 .633-.633l2.051-.683a1 1 0 0 0 0-1.898l-2.051-.683a1 1 0 0 1-.633-.633L6.95 5.684ZM13.949 13.684a1 1 0 0 0-1.898 0l-.184.551a1 1 0 0 1-.632.633l-.551.183a1 1 0 0 0 0 1.898l.551.183a1 1 0 0 1 .633.633l.183.551a1 1 0 0 0 1.898 0l.184-.551a1 1 0 0 1 .632-.633l.551-.183a1 1 0 0 0 0-1.898l-.551-.184a1 1 0 0 1-.633-.632l-.183-.551Z" /> <path
d="M15.98 1.804a1 1 0 0 0-1.96 0l-.24 1.192a1 1 0 0 1-.784.785l-1.192.238a1 1 0 0 0 0 1.962l1.192.238a1 1 0 0 1 .785.785l.238 1.192a1 1 0 0 0 1.962 0l.238-1.192a1 1 0 0 1 .785-.785l1.192-.238a1 1 0 0 0 0-1.962l-1.192-.238a1 1 0 0 1-.785-.785l-.238-1.192ZM6.949 5.684a1 1 0 0 0-1.898 0l-.683 2.051a1 1 0 0 1-.633.633l-2.051.683a1 1 0 0 0 0 1.898l2.051.684a1 1 0 0 1 .633.632l.683 2.051a1 1 0 0 0 1.898 0l.683-2.051a1 1 0 0 1 .633-.633l2.051-.683a1 1 0 0 0 0-1.898l-2.051-.683a1 1 0 0 1-.633-.633L6.95 5.684ZM13.949 13.684a1 1 0 0 0-1.898 0l-.184.551a1 1 0 0 1-.632.633l-.551.183a1 1 0 0 0 0 1.898l.551.183a1 1 0 0 1 .633.633l.183.551a1 1 0 0 0 1.898 0l.184-.551a1 1 0 0 1 .632-.633l.551-.183a1 1 0 0 0 0-1.898l-.551-.184a1 1 0 0 1-.633-.632l-.183-.551Z"
/>
</svg> </svg>
<span>Full screen</span> <span>Full screen</span>
${this.aiProvider === 'gemini' ? html`<span class="usage-count">(${this.getTotalUsed()}/${this.getTotalAvailable()})</span>` : ''} ${this.aiProvider === 'gemini'
? html`<span class="usage-count">(${this.getTotalUsed()}/${this.getTotalAvailable()})</span>`
: ''}
</button> </button>
</div> </div>
</div> </div>

View File

@ -4,7 +4,11 @@ import { resizeLayout } from '../../utils/windowResize.js';
export class CustomizeView extends LitElement { export class CustomizeView extends LitElement {
static styles = css` static styles = css`
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
user-select: none; user-select: none;
} }
@ -634,31 +638,87 @@ export class CustomizeView extends LitElement {
renderSidebarIcon(icon) { renderSidebarIcon(icon) {
const icons = { const icons = {
user: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> user: html`<svg
<path d="M19 21V19C19 17.9391 18.5786 16.9217 17.8284 16.1716C17.0783 15.4214 16.0609 15 15 15H9C7.93913 15 6.92172 15.4214 6.17157 16.1716C5.42143 16.9217 5 17.9391 5 19V21"></path> width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<path
d="M19 21V19C19 17.9391 18.5786 16.9217 17.8284 16.1716C17.0783 15.4214 16.0609 15 15 15H9C7.93913 15 6.92172 15.4214 6.17157 16.1716C5.42143 16.9217 5 17.9391 5 19V21"
></path>
<circle cx="12" cy="7" r="4"></circle> <circle cx="12" cy="7" r="4"></circle>
</svg>`, </svg>`,
mic: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> mic: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<path d="M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z"></path> <path d="M12 1a3 3 0 0 0-3 3v8a3 3 0 0 0 6 0V4a3 3 0 0 0-3-3z"></path>
<path d="M19 10v2a7 7 0 0 1-14 0v-2"></path> <path d="M19 10v2a7 7 0 0 1-14 0v-2"></path>
<line x1="12" y1="19" x2="12" y2="23"></line> <line x1="12" y1="19" x2="12" y2="23"></line>
<line x1="8" y1="23" x2="16" y2="23"></line> <line x1="8" y1="23" x2="16" y2="23"></line>
</svg>`, </svg>`,
globe: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> globe: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<circle cx="12" cy="12" r="10"></circle> <circle cx="12" cy="12" r="10"></circle>
<line x1="2" y1="12" x2="22" y2="12"></line> <line x1="2" y1="12" x2="22" y2="12"></line>
<path d="M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z"></path> <path d="M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z"></path>
</svg>`, </svg>`,
display: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> display: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<rect x="2" y="3" width="20" height="14" rx="2" ry="2"></rect> <rect x="2" y="3" width="20" height="14" rx="2" ry="2"></rect>
<line x1="8" y1="21" x2="16" y2="21"></line> <line x1="8" y1="21" x2="16" y2="21"></line>
<line x1="12" y1="17" x2="12" y2="21"></line> <line x1="12" y1="17" x2="12" y2="21"></line>
</svg>`, </svg>`,
camera: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> camera: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<path d="M23 19a2 2 0 0 1-2 2H3a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h4l2-3h6l2 3h4a2 2 0 0 1 2 2z"></path> <path d="M23 19a2 2 0 0 1-2 2H3a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h4l2-3h6l2 3h4a2 2 0 0 1 2 2z"></path>
<circle cx="12" cy="13" r="4"></circle> <circle cx="12" cy="13" r="4"></circle>
</svg>`, </svg>`,
keyboard: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> keyboard: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<rect x="2" y="4" width="20" height="16" rx="2" ry="2"></rect> <rect x="2" y="4" width="20" height="16" rx="2" ry="2"></rect>
<path d="M6 8h.001"></path> <path d="M6 8h.001"></path>
<path d="M10 8h.001"></path> <path d="M10 8h.001"></path>
@ -669,11 +729,29 @@ export class CustomizeView extends LitElement {
<path d="M16 12h.001"></path> <path d="M16 12h.001"></path>
<path d="M7 16h10"></path> <path d="M7 16h10"></path>
</svg>`, </svg>`,
search: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> search: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<circle cx="11" cy="11" r="8"></circle> <circle cx="11" cy="11" r="8"></circle>
<line x1="21" y1="21" x2="16.65" y2="16.65"></line> <line x1="21" y1="21" x2="16.65" y2="16.65"></line>
</svg>`, </svg>`,
cpu: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> cpu: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<rect x="4" y="4" width="16" height="16" rx="2" ry="2"></rect> <rect x="4" y="4" width="16" height="16" rx="2" ry="2"></rect>
<rect x="9" y="9" width="6" height="6"></rect> <rect x="9" y="9" width="6" height="6"></rect>
<line x1="9" y1="1" x2="9" y2="4"></line> <line x1="9" y1="1" x2="9" y2="4"></line>
@ -685,7 +763,16 @@ export class CustomizeView extends LitElement {
<line x1="1" y1="9" x2="4" y2="9"></line> <line x1="1" y1="9" x2="4" y2="9"></line>
<line x1="1" y1="14" x2="4" y2="14"></line> <line x1="1" y1="14" x2="4" y2="14"></line>
</svg>`, </svg>`,
warning: html`<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.7" stroke-linecap="round" stroke-linejoin="round"> warning: html`<svg
width="16"
height="16"
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="1.7"
stroke-linecap="round"
stroke-linejoin="round"
>
<path d="M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z"></path> <path d="M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z"></path>
<line x1="12" y1="9" x2="12" y2="13"></line> <line x1="12" y1="9" x2="12" y2="13"></line>
<line x1="12" y1="17" x2="12.01" y2="17"></line> <line x1="12" y1="17" x2="12.01" y2="17"></line>
@ -701,7 +788,7 @@ export class CustomizeView extends LitElement {
cheatingDaddy.storage.getKeybinds(), cheatingDaddy.storage.getKeybinds(),
cheatingDaddy.storage.getCredentials(), cheatingDaddy.storage.getCredentials(),
cheatingDaddy.storage.getOpenAICredentials(), cheatingDaddy.storage.getOpenAICredentials(),
cheatingDaddy.storage.getOpenAISDKCredentials() cheatingDaddy.storage.getOpenAISDKCredentials(),
]); ]);
this.googleSearchEnabled = prefs.googleSearchEnabled ?? true; this.googleSearchEnabled = prefs.googleSearchEnabled ?? true;
@ -1072,21 +1159,21 @@ export class CustomizeView extends LitElement {
async handleOpenAIApiKeyInput(e) { async handleOpenAIApiKeyInput(e) {
this.openaiApiKey = e.target.value; this.openaiApiKey = e.target.value;
await cheatingDaddy.storage.setOpenAICredentials({ await cheatingDaddy.storage.setOpenAICredentials({
apiKey: e.target.value apiKey: e.target.value,
}); });
} }
async handleOpenAIBaseUrlInput(e) { async handleOpenAIBaseUrlInput(e) {
this.openaiBaseUrl = e.target.value; this.openaiBaseUrl = e.target.value;
await cheatingDaddy.storage.setOpenAICredentials({ await cheatingDaddy.storage.setOpenAICredentials({
baseUrl: e.target.value baseUrl: e.target.value,
}); });
} }
async handleOpenAIModelInput(e) { async handleOpenAIModelInput(e) {
this.openaiModel = e.target.value; this.openaiModel = e.target.value;
await cheatingDaddy.storage.setOpenAICredentials({ await cheatingDaddy.storage.setOpenAICredentials({
model: e.target.value model: e.target.value,
}); });
} }
@ -1094,35 +1181,35 @@ export class CustomizeView extends LitElement {
async handleOpenAISdkApiKeyInput(e) { async handleOpenAISdkApiKeyInput(e) {
this.openaiSdkApiKey = e.target.value; this.openaiSdkApiKey = e.target.value;
await cheatingDaddy.storage.setOpenAISDKCredentials({ await cheatingDaddy.storage.setOpenAISDKCredentials({
apiKey: e.target.value apiKey: e.target.value,
}); });
} }
async handleOpenAISdkBaseUrlInput(e) { async handleOpenAISdkBaseUrlInput(e) {
this.openaiSdkBaseUrl = e.target.value; this.openaiSdkBaseUrl = e.target.value;
await cheatingDaddy.storage.setOpenAISDKCredentials({ await cheatingDaddy.storage.setOpenAISDKCredentials({
baseUrl: e.target.value baseUrl: e.target.value,
}); });
} }
async handleOpenAISdkModelInput(e) { async handleOpenAISdkModelInput(e) {
this.openaiSdkModel = e.target.value; this.openaiSdkModel = e.target.value;
await cheatingDaddy.storage.setOpenAISDKCredentials({ await cheatingDaddy.storage.setOpenAISDKCredentials({
model: e.target.value model: e.target.value,
}); });
} }
async handleOpenAISdkVisionModelInput(e) { async handleOpenAISdkVisionModelInput(e) {
this.openaiSdkVisionModel = e.target.value; this.openaiSdkVisionModel = e.target.value;
await cheatingDaddy.storage.setOpenAISDKCredentials({ await cheatingDaddy.storage.setOpenAISDKCredentials({
visionModel: e.target.value visionModel: e.target.value,
}); });
} }
async handleOpenAISdkWhisperModelInput(e) { async handleOpenAISdkWhisperModelInput(e) {
this.openaiSdkWhisperModel = e.target.value; this.openaiSdkWhisperModel = e.target.value;
await cheatingDaddy.storage.setOpenAISDKCredentials({ await cheatingDaddy.storage.setOpenAISDKCredentials({
whisperModel: e.target.value whisperModel: e.target.value,
}); });
} }
@ -1209,9 +1296,7 @@ export class CustomizeView extends LitElement {
<select class="form-control" .value=${this.selectedProfile} @change=${this.handleProfileSelect}> <select class="form-control" .value=${this.selectedProfile} @change=${this.handleProfileSelect}>
${profiles.map( ${profiles.map(
profile => html` profile => html`
<option value=${profile.value} ?selected=${this.selectedProfile === profile.value}> <option value=${profile.value} ?selected=${this.selectedProfile === profile.value}>${profile.name}</option>
${profile.name}
</option>
` `
)} )}
</select> </select>
@ -1221,15 +1306,12 @@ export class CustomizeView extends LitElement {
<label class="form-label">Custom AI Instructions</label> <label class="form-label">Custom AI Instructions</label>
<textarea <textarea
class="form-control" class="form-control"
placeholder="Add specific instructions for how you want the AI to behave during ${ placeholder="Add specific instructions for how you want the AI to behave during ${profileNames[this.selectedProfile] ||
profileNames[this.selectedProfile] || 'this interaction' 'this interaction'}..."
}..."
.value=${this.customPrompt} .value=${this.customPrompt}
@input=${this.handleCustomPromptInput} @input=${this.handleCustomPromptInput}
></textarea> ></textarea>
<div class="form-description"> <div class="form-description">Personalize the AI's behavior with specific instructions</div>
Personalize the AI's behavior with specific instructions
</div>
</div> </div>
</div> </div>
</div> </div>
@ -1247,9 +1329,7 @@ export class CustomizeView extends LitElement {
<option value="mic_only">Microphone Only (Me)</option> <option value="mic_only">Microphone Only (Me)</option>
<option value="both">Both Speaker & Microphone</option> <option value="both">Both Speaker & Microphone</option>
</select> </select>
<div class="form-description"> <div class="form-description">Choose which audio sources to capture for the AI.</div>
Choose which audio sources to capture for the AI.
</div>
</div> </div>
</div> </div>
`; `;
@ -1270,9 +1350,7 @@ export class CustomizeView extends LitElement {
<select class="form-control" .value=${this.selectedLanguage} @change=${this.handleLanguageSelect}> <select class="form-control" .value=${this.selectedLanguage} @change=${this.handleLanguageSelect}>
${languages.map( ${languages.map(
language => html` language => html`
<option value=${language.value} ?selected=${this.selectedLanguage === language.value}> <option value=${language.value} ?selected=${this.selectedLanguage === language.value}>${language.name}</option>
${language.name}
</option>
` `
)} )}
</select> </select>
@ -1295,17 +1373,9 @@ export class CustomizeView extends LitElement {
<span class="current-selection">${currentTheme?.name || 'Dark'}</span> <span class="current-selection">${currentTheme?.name || 'Dark'}</span>
</label> </label>
<select class="form-control" .value=${this.theme} @change=${this.handleThemeChange}> <select class="form-control" .value=${this.theme} @change=${this.handleThemeChange}>
${themes.map( ${themes.map(theme => html` <option value=${theme.value} ?selected=${this.theme === theme.value}>${theme.name}</option> `)}
theme => html`
<option value=${theme.value} ?selected=${this.theme === theme.value}>
${theme.name}
</option>
`
)}
</select> </select>
<div class="form-description"> <div class="form-description">Choose a color theme for the interface</div>
Choose a color theme for the interface
</div>
</div> </div>
<div class="form-group"> <div class="form-group">
@ -1318,10 +1388,7 @@ export class CustomizeView extends LitElement {
<option value="compact" ?selected=${this.layoutMode === 'compact'}>Compact</option> <option value="compact" ?selected=${this.layoutMode === 'compact'}>Compact</option>
</select> </select>
<div class="form-description"> <div class="form-description">
${this.layoutMode === 'compact' ${this.layoutMode === 'compact' ? 'Smaller window with reduced padding' : 'Standard layout with comfortable spacing'}
? 'Smaller window with reduced padding'
: 'Standard layout with comfortable spacing'
}
</div> </div>
</div> </div>
@ -1379,7 +1446,9 @@ export class CustomizeView extends LitElement {
<div class="form-group"> <div class="form-group">
<label class="form-label"> <label class="form-label">
Image Quality Image Quality
<span class="current-selection">${this.selectedImageQuality.charAt(0).toUpperCase() + this.selectedImageQuality.slice(1)}</span> <span class="current-selection"
>${this.selectedImageQuality.charAt(0).toUpperCase() + this.selectedImageQuality.slice(1)}</span
>
</label> </label>
<select class="form-control" .value=${this.selectedImageQuality} @change=${this.handleImageQualitySelect}> <select class="form-control" .value=${this.selectedImageQuality} @change=${this.handleImageQualitySelect}>
<option value="high" ?selected=${this.selectedImageQuality === 'high'}>High Quality</option> <option value="high" ?selected=${this.selectedImageQuality === 'high'}>High Quality</option>
@ -1391,8 +1460,7 @@ export class CustomizeView extends LitElement {
? 'Best quality, uses more tokens' ? 'Best quality, uses more tokens'
: this.selectedImageQuality === 'medium' : this.selectedImageQuality === 'medium'
? 'Balanced quality and token usage' ? 'Balanced quality and token usage'
: 'Lower quality, uses fewer tokens' : 'Lower quality, uses fewer tokens'}
}
</div> </div>
</div> </div>
</div> </div>
@ -1446,9 +1514,9 @@ export class CustomizeView extends LitElement {
renderAIProviderSection() { renderAIProviderSection() {
const providerNames = { const providerNames = {
'gemini': 'Google Gemini', gemini: 'Google Gemini',
'openai-realtime': 'OpenAI Realtime', 'openai-realtime': 'OpenAI Realtime',
'openai-sdk': 'OpenAI SDK (BotHub, etc.)' 'openai-sdk': 'OpenAI SDK (BotHub, etc.)',
}; };
return html` return html`
@ -1464,12 +1532,11 @@ export class CustomizeView extends LitElement {
<option value="openai-realtime" ?selected=${this.aiProvider === 'openai-realtime'}>OpenAI Realtime API</option> <option value="openai-realtime" ?selected=${this.aiProvider === 'openai-realtime'}>OpenAI Realtime API</option>
<option value="openai-sdk" ?selected=${this.aiProvider === 'openai-sdk'}>OpenAI SDK (BotHub, Azure, etc.)</option> <option value="openai-sdk" ?selected=${this.aiProvider === 'openai-sdk'}>OpenAI SDK (BotHub, Azure, etc.)</option>
</select> </select>
<div class="form-description"> <div class="form-description">Choose which AI provider to use for conversations and screen analysis</div>
Choose which AI provider to use for conversations and screen analysis
</div>
</div> </div>
${this.aiProvider === 'gemini' ? html` ${this.aiProvider === 'gemini'
? html`
<div class="form-group full-width"> <div class="form-group full-width">
<label class="form-label">Gemini API Key</label> <label class="form-label">Gemini API Key</label>
<input <input
@ -1480,10 +1547,15 @@ export class CustomizeView extends LitElement {
@input=${this.handleGeminiApiKeyInput} @input=${this.handleGeminiApiKeyInput}
/> />
<div class="form-description"> <div class="form-description">
Get your API key from <a href="https://aistudio.google.com/app/apikey" target="_blank" style="color: var(--text-color);">Google AI Studio</a> Get your API key from
<a href="https://aistudio.google.com/app/apikey" target="_blank" style="color: var(--text-color);"
>Google AI Studio</a
>
</div> </div>
</div> </div>
` : this.aiProvider === 'openai-realtime' ? html` `
: this.aiProvider === 'openai-realtime'
? html`
<div class="form-group full-width"> <div class="form-group full-width">
<label class="form-label">OpenAI API Key</label> <label class="form-label">OpenAI API Key</label>
<input <input
@ -1494,7 +1566,10 @@ export class CustomizeView extends LitElement {
@input=${this.handleOpenAIApiKeyInput} @input=${this.handleOpenAIApiKeyInput}
/> />
<div class="form-description"> <div class="form-description">
Get your API key from <a href="https://platform.openai.com/api-keys" target="_blank" style="color: var(--text-color);">OpenAI Platform</a> Get your API key from
<a href="https://platform.openai.com/api-keys" target="_blank" style="color: var(--text-color);"
>OpenAI Platform</a
>
</div> </div>
</div> </div>
@ -1507,9 +1582,7 @@ export class CustomizeView extends LitElement {
.value=${this.openaiBaseUrl} .value=${this.openaiBaseUrl}
@input=${this.handleOpenAIBaseUrlInput} @input=${this.handleOpenAIBaseUrlInput}
/> />
<div class="form-description"> <div class="form-description">Override the base URL for OpenAI-compatible APIs</div>
Override the base URL for OpenAI-compatible APIs
</div>
</div> </div>
<div class="form-group full-width"> <div class="form-group full-width">
@ -1521,11 +1594,10 @@ export class CustomizeView extends LitElement {
.value=${this.openaiModel} .value=${this.openaiModel}
@input=${this.handleOpenAIModelInput} @input=${this.handleOpenAIModelInput}
/> />
<div class="form-description"> <div class="form-description">Realtime API model to use</div>
Realtime API model to use
</div> </div>
</div> `
` : html` : html`
<div class="form-group full-width"> <div class="form-group full-width">
<label class="form-label">API Key</label> <label class="form-label">API Key</label>
<input <input
@ -1535,9 +1607,7 @@ export class CustomizeView extends LitElement {
.value=${this.openaiSdkApiKey} .value=${this.openaiSdkApiKey}
@input=${this.handleOpenAISdkApiKeyInput} @input=${this.handleOpenAISdkApiKeyInput}
/> />
<div class="form-description"> <div class="form-description">API key for your provider (BotHub, Azure, OpenRouter, etc.)</div>
API key for your provider (BotHub, Azure, OpenRouter, etc.)
</div>
</div> </div>
<div class="form-group full-width"> <div class="form-group full-width">
@ -1549,9 +1619,7 @@ export class CustomizeView extends LitElement {
.value=${this.openaiSdkBaseUrl} .value=${this.openaiSdkBaseUrl}
@input=${this.handleOpenAISdkBaseUrlInput} @input=${this.handleOpenAISdkBaseUrlInput}
/> />
<div class="form-description"> <div class="form-description">API endpoint URL (e.g., https://bothub.chat/api/v2/openai/v1)</div>
API endpoint URL (e.g., https://bothub.chat/api/v2/openai/v1)
</div>
</div> </div>
<div class="form-row"> <div class="form-row">
@ -1586,13 +1654,14 @@ export class CustomizeView extends LitElement {
.value=${this.openaiSdkWhisperModel} .value=${this.openaiSdkWhisperModel}
@input=${this.handleOpenAISdkWhisperModelInput} @input=${this.handleOpenAISdkWhisperModelInput}
/> />
<div class="form-description"> <div class="form-description">Model for audio transcription</div>
Model for audio transcription
</div>
</div> </div>
`} `}
<div class="form-description full-width" style="margin-top: 12px; padding: 12px; background: var(--bg-secondary); border-left: 2px solid var(--border-default); border-radius: 3px;"> <div
class="form-description full-width"
style="margin-top: 12px; padding: 12px; background: var(--bg-secondary); border-left: 2px solid var(--border-default); border-radius: 3px;"
>
<strong>Note:</strong> You must restart the AI session for provider changes to take effect. <strong>Note:</strong> You must restart the AI session for provider changes to take effect.
</div> </div>
</div> </div>
@ -1628,20 +1697,19 @@ export class CustomizeView extends LitElement {
<div class="form-group"> <div class="form-group">
<label class="form-label" style="color: var(--error-color);">Data Management</label> <label class="form-label" style="color: var(--error-color);">Data Management</label>
<div class="form-description" style="margin-bottom: 12px;"> <div class="form-description" style="margin-bottom: 12px;">
<strong>Warning:</strong> This action will permanently delete all local data including API keys, preferences, and session history. This cannot be undone. <strong>Warning:</strong> This action will permanently delete all local data including API keys, preferences, and session
history. This cannot be undone.
</div> </div>
<button <button class="danger-button" @click=${this.clearLocalData} ?disabled=${this.isClearing}>
class="danger-button"
@click=${this.clearLocalData}
?disabled=${this.isClearing}
>
${this.isClearing ? 'Clearing...' : 'Clear All Local Data'} ${this.isClearing ? 'Clearing...' : 'Clear All Local Data'}
</button> </button>
${this.clearStatusMessage ? html` ${this.clearStatusMessage
? html`
<div class="status-message ${this.clearStatusType === 'success' ? 'status-success' : 'status-error'}"> <div class="status-message ${this.clearStatusType === 'success' ? 'status-success' : 'status-error'}">
${this.clearStatusMessage} ${this.clearStatusMessage}
</div> </div>
` : ''} `
: ''}
</div> </div>
</div> </div>
`; `;
@ -1690,9 +1758,7 @@ export class CustomizeView extends LitElement {
` `
)} )}
</nav> </nav>
<div class="settings-content"> <div class="settings-content">${this.renderSectionContent()}</div>
${this.renderSectionContent()}
</div>
</div> </div>
`; `;
} }

View File

@ -4,7 +4,11 @@ import { resizeLayout } from '../../utils/windowResize.js';
export class HelpView extends LitElement { export class HelpView extends LitElement {
static styles = css` static styles = css`
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
user-select: none; user-select: none;
} }
@ -292,26 +296,61 @@ export class HelpView extends LitElement {
</div> </div>
<div class="community-links"> <div class="community-links">
<div class="community-link" @click=${() => this.handleExternalLinkClick('https://cheatingdaddy.com')}> <div class="community-link" @click=${() => this.handleExternalLinkClick('https://cheatingdaddy.com')}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <svg
<path d="M14 11.9976C14 9.5059 11.683 7 8.85714 7C8.52241 7 7.41904 7.00001 7.14286 7.00001C4.30254 7.00001 2 9.23752 2 11.9976C2 14.376 3.70973 16.3664 6 16.8714C6.36756 16.9525 6.75006 16.9952 7.14286 16.9952"></path> viewBox="0 0 24 24"
<path d="M10 11.9976C10 14.4893 12.317 16.9952 15.1429 16.9952C15.4776 16.9952 16.581 16.9952 16.8571 16.9952C19.6975 16.9952 22 14.7577 22 11.9976C22 9.6192 20.2903 7.62884 18 7.12383C17.6324 7.04278 17.2499 6.99999 16.8571 6.99999"></path> fill="none"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
>
<path
d="M14 11.9976C14 9.5059 11.683 7 8.85714 7C8.52241 7 7.41904 7.00001 7.14286 7.00001C4.30254 7.00001 2 9.23752 2 11.9976C2 14.376 3.70973 16.3664 6 16.8714C6.36756 16.9525 6.75006 16.9952 7.14286 16.9952"
></path>
<path
d="M10 11.9976C10 14.4893 12.317 16.9952 15.1429 16.9952C15.4776 16.9952 16.581 16.9952 16.8571 16.9952C19.6975 16.9952 22 14.7577 22 11.9976C22 9.6192 20.2903 7.62884 18 7.12383C17.6324 7.04278 17.2499 6.99999 16.8571 6.99999"
></path>
</svg> </svg>
Website Website
</div> </div>
<div class="community-link" @click=${() => this.handleExternalLinkClick('https://github.com/sohzm/cheating-daddy')}> <div class="community-link" @click=${() => this.handleExternalLinkClick('https://github.com/sohzm/cheating-daddy')}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <svg
<path d="M16 22.0268V19.1568C16.0375 18.68 15.9731 18.2006 15.811 17.7506C15.6489 17.3006 15.3929 16.8902 15.06 16.5468C18.2 16.1968 21.5 15.0068 21.5 9.54679C21.4997 8.15062 20.9627 6.80799 20 5.79679C20.4558 4.5753 20.4236 3.22514 19.91 2.02679C19.91 2.02679 18.73 1.67679 16 3.50679C13.708 2.88561 11.292 2.88561 8.99999 3.50679C6.26999 1.67679 5.08999 2.02679 5.08999 2.02679C4.57636 3.22514 4.54413 4.5753 4.99999 5.79679C4.03011 6.81549 3.49251 8.17026 3.49999 9.57679C3.49999 14.9968 6.79998 16.1868 9.93998 16.5768C9.61098 16.9168 9.35725 17.3222 9.19529 17.7667C9.03334 18.2112 8.96679 18.6849 8.99999 19.1568V22.0268"></path> viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
>
<path
d="M16 22.0268V19.1568C16.0375 18.68 15.9731 18.2006 15.811 17.7506C15.6489 17.3006 15.3929 16.8902 15.06 16.5468C18.2 16.1968 21.5 15.0068 21.5 9.54679C21.4997 8.15062 20.9627 6.80799 20 5.79679C20.4558 4.5753 20.4236 3.22514 19.91 2.02679C19.91 2.02679 18.73 1.67679 16 3.50679C13.708 2.88561 11.292 2.88561 8.99999 3.50679C6.26999 1.67679 5.08999 2.02679 5.08999 2.02679C4.57636 3.22514 4.54413 4.5753 4.99999 5.79679C4.03011 6.81549 3.49251 8.17026 3.49999 9.57679C3.49999 14.9968 6.79998 16.1868 9.93998 16.5768C9.61098 16.9168 9.35725 17.3222 9.19529 17.7667C9.03334 18.2112 8.96679 18.6849 8.99999 19.1568V22.0268"
></path>
<path d="M9 20.0267C6 20.9999 3.5 20.0267 2 17.0267"></path> <path d="M9 20.0267C6 20.9999 3.5 20.0267 2 17.0267"></path>
</svg> </svg>
GitHub GitHub
</div> </div>
<div class="community-link" @click=${() => this.handleExternalLinkClick('https://discord.gg/GCBdubnXfJ')}> <div class="community-link" @click=${() => this.handleExternalLinkClick('https://discord.gg/GCBdubnXfJ')}>
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <svg
viewBox="0 0 24 24"
fill="none"
stroke="currentColor"
stroke-width="2"
stroke-linecap="round"
stroke-linejoin="round"
>
<path d="M5.5 16C10.5 18.5 13.5 18.5 18.5 16"></path> <path d="M5.5 16C10.5 18.5 13.5 18.5 18.5 16"></path>
<path d="M15.5 17.5L16.5 19.5C16.5 19.5 20.6713 18.1717 22 16C22 15 22.5301 7.85339 19 5.5C17.5 4.5 15 4 15 4L14 6H12"></path> <path
<path d="M8.52832 17.5L7.52832 19.5C7.52832 19.5 3.35699 18.1717 2.02832 16C2.02832 15 1.49823 7.85339 5.02832 5.5C6.52832 4.5 9.02832 4 9.02832 4L10.0283 6H12.0283"></path> d="M15.5 17.5L16.5 19.5C16.5 19.5 20.6713 18.1717 22 16C22 15 22.5301 7.85339 19 5.5C17.5 4.5 15 4 15 4L14 6H12"
<path d="M8.5 14C7.67157 14 7 13.1046 7 12C7 10.8954 7.67157 10 8.5 10C9.32843 10 10 10.8954 10 12C10 13.1046 9.32843 14 8.5 14Z"></path> ></path>
<path d="M15.5 14C14.6716 14 14 13.1046 14 12C14 10.8954 14.6716 10 15.5 10C16.3284 10 17 10.8954 17 12C17 13.1046 16.3284 14 15.5 14Z"></path> <path
d="M8.52832 17.5L7.52832 19.5C7.52832 19.5 3.35699 18.1717 2.02832 16C2.02832 15 1.49823 7.85339 5.02832 5.5C6.52832 4.5 9.02832 4 9.02832 4L10.0283 6H12.0283"
></path>
<path
d="M8.5 14C7.67157 14 7 13.1046 7 12C7 10.8954 7.67157 10 8.5 10C9.32843 10 10 10.8954 10 12C10 13.1046 9.32843 14 8.5 14Z"
></path>
<path
d="M15.5 14C14.6716 14 14 13.1046 14 12C14 10.8954 14.6716 10 15.5 10C16.3284 10 17 10.8954 17 12C17 13.1046 16.3284 14 15.5 14Z"
></path>
</svg> </svg>
Discord Discord
</div> </div>
@ -395,9 +434,7 @@ export class HelpView extends LitElement {
</div> </div>
</div> </div>
</div> </div>
<div class="description" style="margin-top: 12px; text-align: center;"> <div class="description" style="margin-top: 12px; text-align: center;">You can customize these shortcuts in Settings.</div>
You can customize these shortcuts in Settings.
</div>
</div> </div>
<div class="option-group"> <div class="option-group">
@ -469,9 +506,7 @@ export class HelpView extends LitElement {
<div class="description" style="margin-bottom: 12px;"> <div class="description" style="margin-bottom: 12px;">
If you're experiencing issues with audio capture or other features, check the application logs for diagnostic information. If you're experiencing issues with audio capture or other features, check the application logs for diagnostic information.
</div> </div>
<button class="open-logs-btn" @click=${this.openLogsFolder}> <button class="open-logs-btn" @click=${this.openLogsFolder}>📁 Open Logs Folder</button>
📁 Open Logs Folder
</button>
</div> </div>
</div> </div>
`; `;

View File

@ -4,7 +4,11 @@ import { resizeLayout } from '../../utils/windowResize.js';
export class HistoryView extends LitElement { export class HistoryView extends LitElement {
static styles = css` static styles = css`
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
user-select: none; user-select: none;
} }
@ -505,18 +509,22 @@ export class HistoryView extends LitElement {
return html` return html`
<div class="session-context"> <div class="session-context">
${profile ? html` ${profile
? html`
<div class="session-context-row"> <div class="session-context-row">
<span class="context-label">Profile:</span> <span class="context-label">Profile:</span>
<span class="context-value">${profileNames[profile] || profile}</span> <span class="context-value">${profileNames[profile] || profile}</span>
</div> </div>
` : ''} `
${customPrompt ? html` : ''}
${customPrompt
? html`
<div class="session-context-row"> <div class="session-context-row">
<span class="context-label">Custom Prompt:</span> <span class="context-label">Custom Prompt:</span>
<span class="custom-prompt-value">${customPrompt}</span> <span class="custom-prompt-value">${customPrompt}</span>
</div> </div>
` : ''} `
: ''}
</div> </div>
`; `;
} }
@ -559,9 +567,14 @@ export class HistoryView extends LitElement {
return html`<div class="empty-state">No screen analysis data available</div>`; return html`<div class="empty-state">No screen analysis data available</div>`;
} }
return screenAnalysisHistory.map(analysis => html` return screenAnalysisHistory.map(
<div class="message screen"><div class="analysis-meta">${this.formatTimestamp(analysis.timestamp)} ${analysis.model || 'unknown model'}</div>${analysis.response}</div> analysis => html`
`); <div class="message screen">
<div class="analysis-meta">${this.formatTimestamp(analysis.timestamp)} ${analysis.model || 'unknown model'}</div>
${analysis.response}
</div>
`
);
} }
renderConversationView() { renderConversationView() {
@ -604,22 +617,13 @@ export class HistoryView extends LitElement {
</div> </div>
</div> </div>
<div class="view-tabs"> <div class="view-tabs">
<button <button class="view-tab ${this.activeTab === 'conversation' ? 'active' : ''}" @click=${() => this.handleTabClick('conversation')}>
class="view-tab ${this.activeTab === 'conversation' ? 'active' : ''}"
@click=${() => this.handleTabClick('conversation')}
>
Conversation ${hasConversation ? `(${conversationHistory.length})` : ''} Conversation ${hasConversation ? `(${conversationHistory.length})` : ''}
</button> </button>
<button <button class="view-tab ${this.activeTab === 'screen' ? 'active' : ''}" @click=${() => this.handleTabClick('screen')}>
class="view-tab ${this.activeTab === 'screen' ? 'active' : ''}"
@click=${() => this.handleTabClick('screen')}
>
Screen ${hasScreenAnalysis ? `(${screenAnalysisHistory.length})` : ''} Screen ${hasScreenAnalysis ? `(${screenAnalysisHistory.length})` : ''}
</button> </button>
<button <button class="view-tab ${this.activeTab === 'context' ? 'active' : ''}" @click=${() => this.handleTabClick('context')}>
class="view-tab ${this.activeTab === 'context' ? 'active' : ''}"
@click=${() => this.handleTabClick('context')}
>
Context ${hasContext ? '' : '(empty)'} Context ${hasContext ? '' : '(empty)'}
</button> </button>
</div> </div>
@ -638,11 +642,7 @@ export class HistoryView extends LitElement {
return html`<div class="history-container">${this.renderConversationView()}</div>`; return html`<div class="history-container">${this.renderConversationView()}</div>`;
} }
return html` return html` <div class="history-container">${this.renderSessionsList()}</div> `;
<div class="history-container">
${this.renderSessionsList()}
</div>
`;
} }
} }

View File

@ -4,7 +4,11 @@ import { resizeLayout } from '../../utils/windowResize.js';
export class MainView extends LitElement { export class MainView extends LitElement {
static styles = css` static styles = css`
* { * {
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; font-family:
'Inter',
-apple-system,
BlinkMacSystemFont,
sans-serif;
cursor: default; cursor: default;
user-select: none; user-select: none;
} }
@ -54,7 +58,8 @@ export class MainView extends LitElement {
} }
@keyframes blink-red { @keyframes blink-red {
0%, 100% { 0%,
100% {
border-color: var(--border-color); border-color: var(--border-color);
} }
50% { 50% {

View File

@ -512,7 +512,9 @@ export class OnboardingView extends LitElement {
<div class="onboarding-container"> <div class="onboarding-container">
<button class="close-button" @click=${this.handleClose} title="Close"> <button class="close-button" @click=${this.handleClose} title="Close">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor"> <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
<path d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z" /> <path
d="M6.28 5.22a.75.75 0 0 0-1.06 1.06L8.94 10l-3.72 3.72a.75.75 0 1 0 1.06 1.06L10 11.06l3.72 3.72a.75.75 0 1 0 1.06-1.06L11.06 10l3.72-3.72a.75.75 0 0 0-1.06-1.06L10 8.94 6.28 5.22Z"
/>
</svg> </svg>
</button> </button>
<canvas class="gradient-canvas"></canvas> <canvas class="gradient-canvas"></canvas>

View File

@ -8,7 +8,7 @@ const CONFIG_VERSION = 1;
const DEFAULT_CONFIG = { const DEFAULT_CONFIG = {
configVersion: CONFIG_VERSION, configVersion: CONFIG_VERSION,
onboarded: false, onboarded: false,
layout: 'normal' layout: 'normal',
}; };
const DEFAULT_CREDENTIALS = { const DEFAULT_CREDENTIALS = {
@ -22,7 +22,7 @@ const DEFAULT_CREDENTIALS = {
openaiSdkBaseUrl: '', openaiSdkBaseUrl: '',
openaiSdkModel: 'gpt-4o', openaiSdkModel: 'gpt-4o',
openaiSdkVisionModel: 'gpt-4o', openaiSdkVisionModel: 'gpt-4o',
openaiSdkWhisperModel: 'whisper-1' openaiSdkWhisperModel: 'whisper-1',
}; };
const DEFAULT_PREFERENCES = { const DEFAULT_PREFERENCES = {
@ -36,13 +36,13 @@ const DEFAULT_PREFERENCES = {
fontSize: 'medium', fontSize: 'medium',
backgroundTransparency: 0.8, backgroundTransparency: 0.8,
googleSearchEnabled: false, googleSearchEnabled: false,
aiProvider: 'gemini' aiProvider: 'gemini',
}; };
const DEFAULT_KEYBINDS = null; // null means use system defaults const DEFAULT_KEYBINDS = null; // null means use system defaults
const DEFAULT_LIMITS = { const DEFAULT_LIMITS = {
data: [] // Array of { date: 'YYYY-MM-DD', flash: { count: 0 }, flashLite: { count: 0 } } data: [], // Array of { date: 'YYYY-MM-DD', flash: { count: 0 }, flashLite: { count: 0 } }
}; };
// Get the config directory path based on OS // Get the config directory path based on OS
@ -208,7 +208,7 @@ function getOpenAICredentials() {
return { return {
apiKey: creds.openaiApiKey || '', apiKey: creds.openaiApiKey || '',
baseUrl: creds.openaiBaseUrl || '', baseUrl: creds.openaiBaseUrl || '',
model: creds.openaiModel || 'gpt-4o-realtime-preview-2024-12-17' model: creds.openaiModel || 'gpt-4o-realtime-preview-2024-12-17',
}; };
} }
@ -227,7 +227,7 @@ function getOpenAISDKCredentials() {
baseUrl: creds.openaiSdkBaseUrl || '', baseUrl: creds.openaiSdkBaseUrl || '',
model: creds.openaiSdkModel || 'gpt-4o', model: creds.openaiSdkModel || 'gpt-4o',
visionModel: creds.openaiSdkVisionModel || 'gpt-4o', visionModel: creds.openaiSdkVisionModel || 'gpt-4o',
whisperModel: creds.openaiSdkWhisperModel || 'whisper-1' whisperModel: creds.openaiSdkWhisperModel || 'whisper-1',
}; };
} }
@ -301,7 +301,7 @@ function getTodayLimits() {
const newEntry = { const newEntry = {
date: today, date: today,
flash: { count: 0 }, flash: { count: 0 },
flashLite: { count: 0 } flashLite: { count: 0 },
}; };
limits.data.push(newEntry); limits.data.push(newEntry);
setLimits(limits); setLimits(limits);
@ -322,7 +322,7 @@ function incrementLimitCount(model) {
todayEntry = { todayEntry = {
date: today, date: today,
flash: { count: 0 }, flash: { count: 0 },
flashLite: { count: 0 } flashLite: { count: 0 },
}; };
limits.data.push(todayEntry); limits.data.push(todayEntry);
} else { } else {
@ -376,7 +376,7 @@ function saveSession(sessionId, data) {
customPrompt: data.customPrompt || existingSession?.customPrompt || null, customPrompt: data.customPrompt || existingSession?.customPrompt || null,
// Conversation data // Conversation data
conversationHistory: data.conversationHistory || existingSession?.conversationHistory || [], conversationHistory: data.conversationHistory || existingSession?.conversationHistory || [],
screenAnalysisHistory: data.screenAnalysisHistory || existingSession?.screenAnalysisHistory || [] screenAnalysisHistory: data.screenAnalysisHistory || existingSession?.screenAnalysisHistory || [],
}; };
return writeJsonFile(sessionPath, sessionData); return writeJsonFile(sessionPath, sessionData);
} }
@ -393,7 +393,8 @@ function getAllSessions() {
return []; return [];
} }
const files = fs.readdirSync(historyDir) const files = fs
.readdirSync(historyDir)
.filter(f => f.endsWith('.json')) .filter(f => f.endsWith('.json'))
.sort((a, b) => { .sort((a, b) => {
// Sort by timestamp descending (newest first) // Sort by timestamp descending (newest first)
@ -402,7 +403,8 @@ function getAllSessions() {
return tsB - tsA; return tsB - tsA;
}); });
return files.map(file => { return files
.map(file => {
const sessionId = file.replace('.json', ''); const sessionId = file.replace('.json', '');
const data = readJsonFile(path.join(historyDir, file), null); const data = readJsonFile(path.join(historyDir, file), null);
if (data) { if (data) {
@ -413,11 +415,12 @@ function getAllSessions() {
messageCount: data.conversationHistory?.length || 0, messageCount: data.conversationHistory?.length || 0,
screenAnalysisCount: data.screenAnalysisHistory?.length || 0, screenAnalysisCount: data.screenAnalysisHistory?.length || 0,
profile: data.profile || null, profile: data.profile || null,
customPrompt: data.customPrompt || null customPrompt: data.customPrompt || null,
}; };
} }
return null; return null;
}).filter(Boolean); })
.filter(Boolean);
} catch (error) { } catch (error) {
console.error('Error reading sessions:', error.message); console.error('Error reading sessions:', error.message);
return []; return [];
@ -504,5 +507,5 @@ module.exports = {
deleteAllSessions, deleteAllSessions,
// Clear all // Clear all
clearAllData clearAllData,
}; };

View File

@ -52,9 +52,7 @@ function buildContextMessage() {
if (validTurns.length === 0) return null; if (validTurns.length === 0) return null;
const contextLines = validTurns.map(turn => const contextLines = validTurns.map(turn => `[Interviewer]: ${turn.transcription.trim()}\n[Your answer]: ${turn.ai_response.trim()}`);
`[Interviewer]: ${turn.transcription.trim()}\n[Your answer]: ${turn.ai_response.trim()}`
);
return `Session reconnected. Here's the conversation so far:\n\n${contextLines.join('\n\n')}\n\nContinue from here.`; return `Session reconnected. Here's the conversation so far:\n\n${contextLines.join('\n\n')}\n\nContinue from here.`;
} }
@ -74,7 +72,7 @@ function initializeNewSession(profile = null, customPrompt = null) {
sendToRenderer('save-session-context', { sendToRenderer('save-session-context', {
sessionId: currentSessionId, sessionId: currentSessionId,
profile: profile, profile: profile,
customPrompt: customPrompt || '' customPrompt: customPrompt || '',
}); });
} }
} }
@ -110,7 +108,7 @@ function saveScreenAnalysis(prompt, response, model) {
timestamp: Date.now(), timestamp: Date.now(),
prompt: prompt, prompt: prompt,
response: response.trim(), response: response.trim(),
model: model model: model,
}; };
screenAnalysisHistory.push(analysisEntry); screenAnalysisHistory.push(analysisEntry);
@ -122,7 +120,7 @@ function saveScreenAnalysis(prompt, response, model) {
analysis: analysisEntry, analysis: analysisEntry,
fullHistory: screenAnalysisHistory, fullHistory: screenAnalysisHistory,
profile: currentProfile, profile: currentProfile,
customPrompt: currentCustomPrompt customPrompt: currentCustomPrompt,
}); });
} }

View File

@ -65,7 +65,8 @@ function writeLog(level, args) {
try { try {
const timestamp = new Date().toISOString(); const timestamp = new Date().toISOString();
const message = args.map(arg => { const message = args
.map(arg => {
if (typeof arg === 'object') { if (typeof arg === 'object') {
try { try {
return JSON.stringify(arg, null, 2); return JSON.stringify(arg, null, 2);
@ -74,7 +75,8 @@ function writeLog(level, args) {
} }
} }
return String(arg); return String(arg);
}).join(' '); })
.join(' ');
logFile.write(`[${timestamp}] [${level}] ${message}\n`); logFile.write(`[${timestamp}] [${level}] ${message}\n`);
} catch (err) { } catch (err) {

View File

@ -311,7 +311,9 @@ async function sendImageToOpenAI(base64Data, prompt, config) {
const { apiKey, baseUrl, model } = config; const { apiKey, baseUrl, model } = config;
// OpenAI doesn't support images in Realtime API yet, use standard Chat Completions // OpenAI doesn't support images in Realtime API yet, use standard Chat Completions
const apiEndpoint = baseUrl ? `${baseUrl.replace('wss://', 'https://').replace('/v1/realtime', '')}/v1/chat/completions` : 'https://api.openai.com/v1/chat/completions'; const apiEndpoint = baseUrl
? `${baseUrl.replace('wss://', 'https://').replace('/v1/realtime', '')}/v1/chat/completions`
: 'https://api.openai.com/v1/chat/completions';
try { try {
const response = await fetch(apiEndpoint, { const response = await fetch(apiEndpoint, {

View File

@ -22,12 +22,18 @@ const isWindows = process.platform === 'win32';
// Send logs to main process for file logging // Send logs to main process for file logging
function logToMain(level, ...args) { function logToMain(level, ...args) {
const message = args.map(arg => { const message = args
.map(arg => {
if (typeof arg === 'object') { if (typeof arg === 'object') {
try { return JSON.stringify(arg); } catch { return String(arg); } try {
return JSON.stringify(arg);
} catch {
return String(arg);
}
} }
return String(arg); return String(arg);
}).join(' '); })
.join(' ');
ipcRenderer.send('renderer-log', { level, message }); ipcRenderer.send('renderer-log', { level, message });
// Also log to console // Also log to console
@ -130,7 +136,7 @@ const storage = {
async getTodayLimits() { async getTodayLimits() {
const result = await ipcRenderer.invoke('storage:get-today-limits'); const result = await ipcRenderer.invoke('storage:get-today-limits');
return result.success ? result.data : { flash: { count: 0 }, flashLite: { count: 0 } }; return result.success ? result.data : { flash: { count: 0 }, flashLite: { count: 0 } };
} },
}; };
// Cache for preferences to avoid async calls in hot paths // Cache for preferences to avoid async calls in hot paths
@ -335,7 +341,7 @@ async function startCapture(screenshotIntervalSeconds = 5, imageQuality = 'mediu
enabled: t.enabled, enabled: t.enabled,
muted: t.muted, muted: t.muted,
readyState: t.readyState, readyState: t.readyState,
settings: t.getSettings() settings: t.getSettings(),
})), })),
}); });
@ -475,9 +481,12 @@ function setupWindowsLoopbackProcessing() {
// Resume AudioContext if suspended (Chrome policy) // Resume AudioContext if suspended (Chrome policy)
if (audioContext.state === 'suspended') { if (audioContext.state === 'suspended') {
logToMain('warn', 'AudioContext suspended, attempting resume...'); logToMain('warn', 'AudioContext suspended, attempting resume...');
audioContext.resume().then(() => { audioContext
.resume()
.then(() => {
logToMain('info', 'AudioContext resumed successfully'); logToMain('info', 'AudioContext resumed successfully');
}).catch(err => { })
.catch(err => {
logToMain('error', 'Failed to resume AudioContext:', err.message); logToMain('error', 'Failed to resume AudioContext:', err.message);
}); });
} }
@ -524,7 +533,6 @@ function setupWindowsLoopbackProcessing() {
audioProcessor.connect(audioContext.destination); audioProcessor.connect(audioContext.destination);
logToMain('info', 'Windows audio processing pipeline connected'); logToMain('info', 'Windows audio processing pipeline connected');
} catch (err) { } catch (err) {
logToMain('error', 'Error setting up Windows audio:', err.message, err.stack); logToMain('error', 'Error setting up Windows audio:', err.message, err.stack);
cheatingDaddy.setStatus('Audio error: ' + err.message); cheatingDaddy.setStatus('Audio error: ' + err.message);
@ -760,17 +768,7 @@ async function captureRegionFromScreenshot(rect, screenshotDataUrl) {
const cropContext = cropCanvas.getContext('2d'); const cropContext = cropCanvas.getContext('2d');
// Draw only the selected region // Draw only the selected region
cropContext.drawImage( cropContext.drawImage(img, scaledRect.left, scaledRect.top, scaledRect.width, scaledRect.height, 0, 0, scaledRect.width, scaledRect.height);
img,
scaledRect.left,
scaledRect.top,
scaledRect.width,
scaledRect.height,
0,
0,
scaledRect.width,
scaledRect.height
);
// Convert to blob and send // Convert to blob and send
cropCanvas.toBlob( cropCanvas.toBlob(
@ -983,7 +981,7 @@ ipcRenderer.on('save-session-context', async (event, data) => {
try { try {
await storage.saveSession(data.sessionId, { await storage.saveSession(data.sessionId, {
profile: data.profile, profile: data.profile,
customPrompt: data.customPrompt customPrompt: data.customPrompt,
}); });
console.log('Session context saved:', data.sessionId, 'profile:', data.profile); console.log('Session context saved:', data.sessionId, 'profile:', data.profile);
} catch (error) { } catch (error) {
@ -997,7 +995,7 @@ ipcRenderer.on('save-screen-analysis', async (event, data) => {
await storage.saveSession(data.sessionId, { await storage.saveSession(data.sessionId, {
screenAnalysisHistory: data.fullHistory, screenAnalysisHistory: data.fullHistory,
profile: data.profile, profile: data.profile,
customPrompt: data.customPrompt customPrompt: data.customPrompt,
}); });
console.log('Screen analysis saved:', data.sessionId); console.log('Screen analysis saved:', data.sessionId);
} catch (error) { } catch (error) {
@ -1032,60 +1030,102 @@ const theme = {
themes: { themes: {
dark: { dark: {
background: '#1e1e1e', background: '#1e1e1e',
text: '#e0e0e0', textSecondary: '#a0a0a0', textMuted: '#6b6b6b', text: '#e0e0e0',
border: '#333333', accent: '#ffffff', textSecondary: '#a0a0a0',
btnPrimaryBg: '#ffffff', btnPrimaryText: '#000000', btnPrimaryHover: '#e0e0e0', textMuted: '#6b6b6b',
tooltipBg: '#1a1a1a', tooltipText: '#ffffff', border: '#333333',
keyBg: 'rgba(255,255,255,0.1)' accent: '#ffffff',
btnPrimaryBg: '#ffffff',
btnPrimaryText: '#000000',
btnPrimaryHover: '#e0e0e0',
tooltipBg: '#1a1a1a',
tooltipText: '#ffffff',
keyBg: 'rgba(255,255,255,0.1)',
}, },
light: { light: {
background: '#ffffff', background: '#ffffff',
text: '#1a1a1a', textSecondary: '#555555', textMuted: '#888888', text: '#1a1a1a',
border: '#e0e0e0', accent: '#000000', textSecondary: '#555555',
btnPrimaryBg: '#1a1a1a', btnPrimaryText: '#ffffff', btnPrimaryHover: '#333333', textMuted: '#888888',
tooltipBg: '#1a1a1a', tooltipText: '#ffffff', border: '#e0e0e0',
keyBg: 'rgba(0,0,0,0.1)' accent: '#000000',
btnPrimaryBg: '#1a1a1a',
btnPrimaryText: '#ffffff',
btnPrimaryHover: '#333333',
tooltipBg: '#1a1a1a',
tooltipText: '#ffffff',
keyBg: 'rgba(0,0,0,0.1)',
}, },
midnight: { midnight: {
background: '#0d1117', background: '#0d1117',
text: '#c9d1d9', textSecondary: '#8b949e', textMuted: '#6e7681', text: '#c9d1d9',
border: '#30363d', accent: '#58a6ff', textSecondary: '#8b949e',
btnPrimaryBg: '#58a6ff', btnPrimaryText: '#0d1117', btnPrimaryHover: '#79b8ff', textMuted: '#6e7681',
tooltipBg: '#161b22', tooltipText: '#c9d1d9', border: '#30363d',
keyBg: 'rgba(88,166,255,0.15)' accent: '#58a6ff',
btnPrimaryBg: '#58a6ff',
btnPrimaryText: '#0d1117',
btnPrimaryHover: '#79b8ff',
tooltipBg: '#161b22',
tooltipText: '#c9d1d9',
keyBg: 'rgba(88,166,255,0.15)',
}, },
sepia: { sepia: {
background: '#f4ecd8', background: '#f4ecd8',
text: '#5c4b37', textSecondary: '#7a6a56', textMuted: '#998875', text: '#5c4b37',
border: '#d4c8b0', accent: '#8b4513', textSecondary: '#7a6a56',
btnPrimaryBg: '#5c4b37', btnPrimaryText: '#f4ecd8', btnPrimaryHover: '#7a6a56', textMuted: '#998875',
tooltipBg: '#5c4b37', tooltipText: '#f4ecd8', border: '#d4c8b0',
keyBg: 'rgba(92,75,55,0.15)' accent: '#8b4513',
btnPrimaryBg: '#5c4b37',
btnPrimaryText: '#f4ecd8',
btnPrimaryHover: '#7a6a56',
tooltipBg: '#5c4b37',
tooltipText: '#f4ecd8',
keyBg: 'rgba(92,75,55,0.15)',
}, },
nord: { nord: {
background: '#2e3440', background: '#2e3440',
text: '#eceff4', textSecondary: '#d8dee9', textMuted: '#4c566a', text: '#eceff4',
border: '#3b4252', accent: '#88c0d0', textSecondary: '#d8dee9',
btnPrimaryBg: '#88c0d0', btnPrimaryText: '#2e3440', btnPrimaryHover: '#8fbcbb', textMuted: '#4c566a',
tooltipBg: '#3b4252', tooltipText: '#eceff4', border: '#3b4252',
keyBg: 'rgba(136,192,208,0.15)' accent: '#88c0d0',
btnPrimaryBg: '#88c0d0',
btnPrimaryText: '#2e3440',
btnPrimaryHover: '#8fbcbb',
tooltipBg: '#3b4252',
tooltipText: '#eceff4',
keyBg: 'rgba(136,192,208,0.15)',
}, },
dracula: { dracula: {
background: '#282a36', background: '#282a36',
text: '#f8f8f2', textSecondary: '#bd93f9', textMuted: '#6272a4', text: '#f8f8f2',
border: '#44475a', accent: '#ff79c6', textSecondary: '#bd93f9',
btnPrimaryBg: '#ff79c6', btnPrimaryText: '#282a36', btnPrimaryHover: '#ff92d0', textMuted: '#6272a4',
tooltipBg: '#44475a', tooltipText: '#f8f8f2', border: '#44475a',
keyBg: 'rgba(255,121,198,0.15)' accent: '#ff79c6',
btnPrimaryBg: '#ff79c6',
btnPrimaryText: '#282a36',
btnPrimaryHover: '#ff92d0',
tooltipBg: '#44475a',
tooltipText: '#f8f8f2',
keyBg: 'rgba(255,121,198,0.15)',
}, },
abyss: { abyss: {
background: '#0a0a0a', background: '#0a0a0a',
text: '#d4d4d4', textSecondary: '#808080', textMuted: '#505050', text: '#d4d4d4',
border: '#1a1a1a', accent: '#ffffff', textSecondary: '#808080',
btnPrimaryBg: '#ffffff', btnPrimaryText: '#0a0a0a', btnPrimaryHover: '#d4d4d4', textMuted: '#505050',
tooltipBg: '#141414', tooltipText: '#d4d4d4', border: '#1a1a1a',
keyBg: 'rgba(255,255,255,0.08)' accent: '#ffffff',
} btnPrimaryBg: '#ffffff',
btnPrimaryText: '#0a0a0a',
btnPrimaryHover: '#d4d4d4',
tooltipBg: '#141414',
tooltipText: '#d4d4d4',
keyBg: 'rgba(255,255,255,0.08)',
},
}, },
current: 'dark', current: 'dark',
@ -1102,29 +1142,31 @@ const theme = {
sepia: 'Sepia', sepia: 'Sepia',
nord: 'Nord', nord: 'Nord',
dracula: 'Dracula', dracula: 'Dracula',
abyss: 'Abyss' abyss: 'Abyss',
}; };
return Object.keys(this.themes).map(key => ({ return Object.keys(this.themes).map(key => ({
value: key, value: key,
name: names[key] || key, name: names[key] || key,
colors: this.themes[key] colors: this.themes[key],
})); }));
}, },
hexToRgb(hex) { hexToRgb(hex) {
const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex); const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
return result ? { return result
? {
r: parseInt(result[1], 16), r: parseInt(result[1], 16),
g: parseInt(result[2], 16), g: parseInt(result[2], 16),
b: parseInt(result[3], 16) b: parseInt(result[3], 16),
} : { r: 30, g: 30, b: 30 }; }
: { r: 30, g: 30, b: 30 };
}, },
lightenColor(rgb, amount) { lightenColor(rgb, amount) {
return { return {
r: Math.min(255, rgb.r + amount), r: Math.min(255, rgb.r + amount),
g: Math.min(255, rgb.g + amount), g: Math.min(255, rgb.g + amount),
b: Math.min(255, rgb.b + amount) b: Math.min(255, rgb.b + amount),
}; };
}, },
@ -1132,7 +1174,7 @@ const theme = {
return { return {
r: Math.max(0, rgb.r - amount), r: Math.max(0, rgb.r - amount),
g: Math.max(0, rgb.g - amount), g: Math.max(0, rgb.g - amount),
b: Math.max(0, rgb.b - amount) b: Math.max(0, rgb.b - amount),
}; };
}, },
@ -1212,7 +1254,7 @@ const theme = {
async save(themeName) { async save(themeName) {
await storage.updatePreference('theme', themeName); await storage.updatePreference('theme', themeName);
this.apply(themeName); this.apply(themeName);
} },
}; };
// Consolidated cheatingDaddy object - all functions in one place // Consolidated cheatingDaddy object - all functions in one place

View File

@ -553,7 +553,10 @@ function setupWindowIpcHandlers(mainWindow, sendToRenderer, geminiSessionRef) {
const primaryDisplay = screen.getPrimaryDisplay(); const primaryDisplay = screen.getPrimaryDisplay();
// Calculate bounds that cover all displays // Calculate bounds that cover all displays
let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity; let minX = Infinity,
minY = Infinity,
maxX = -Infinity,
maxY = -Infinity;
displays.forEach(display => { displays.forEach(display => {
minX = Math.min(minX, display.bounds.x); minX = Math.min(minX, display.bounds.x);
minY = Math.min(minY, display.bounds.y); minY = Math.min(minY, display.bounds.y);
@ -712,7 +715,7 @@ function setupWindowIpcHandlers(mainWindow, sendToRenderer, geminiSessionRef) {
regionSelectionWindow.loadURL(`data:text/html;charset=utf-8,${encodeURIComponent(htmlContent)}`); regionSelectionWindow.loadURL(`data:text/html;charset=utf-8,${encodeURIComponent(htmlContent)}`);
return new Promise((resolve) => { return new Promise(resolve => {
ipcMain.once('region-selected', (event, rect) => { ipcMain.once('region-selected', (event, rect) => {
if (regionSelectionWindow && !regionSelectionWindow.isDestroyed()) { if (regionSelectionWindow && !regionSelectionWindow.isDestroyed()) {
regionSelectionWindow.close(); regionSelectionWindow.close();