17 Commits

Author SHA1 Message Date
c2202bdfef Remove keyboard smashing detection logic from note saving process which I thought I already did, damn you AI 2025-05-21 11:24:07 +02:00
8daf7ed6bf Improve note creation by trimming whitespace from each line 2025-05-21 11:20:14 +02:00
4339763261 Enhance cleanup dialog with sensitivity slider for problematic entries 2025-05-20 12:44:53 +02:00
621e85c747 Add functionality to identify and clean up problematic entries in notes 2025-05-20 12:39:43 +02:00
d937ae212c Sort search results by date, newest first 2025-05-18 17:00:23 +02:00
b5dad28491 Clear input field after saving note 2025-05-18 13:50:59 +02:00
1e11cd53c9 Do not reset controller when showing 2025-05-18 09:56:04 +02:00
b74dc5b3c6 Remove unused 2025-05-16 15:44:51 +02:00
474cb662e5 Remove fallback to like 2025-05-16 15:41:28 +02:00
a84973def6 Fix cyrillic 2025-05-16 15:32:58 +02:00
fcb8e41cde Add icu support for cyrillic fts 2025-05-16 14:56:22 +02:00
4f54b89689 Add a date to latest 2025-04-24 10:03:43 +02:00
eb2ec79150 Implement local date via intl
We don't want to display utc
2025-04-24 09:58:57 +02:00
e8b9f0ba49 Enable deleting notes by deleting all their content 2025-04-23 20:56:01 +02:00
6c8340d768 Update 2025-04-23 16:32:24 +02:00
db565f4603 Work out a README 2025-04-23 16:16:03 +02:00
b2bff9e5c5 Implement IPC to show up existing instance when re-running 2025-04-23 15:51:55 +02:00
9 changed files with 954 additions and 89 deletions

View File

@@ -1,16 +1,89 @@
# journaler # Journaler
A new Flutter project. ![Journaler Logo](assets/app_icon.ico)
## Getting Started Journaler is a "cross-platform" desktop application designed to encourage regular journaling by providing automated reminders and a clean writing environment.
This project is a starting point for a Flutter application. ## Overview
A few resources to get you started if this is your first Flutter project: Journaler helps you build a consistent journaling habit by periodically reminding you to record your thoughts throughout the day. It runs silently in the system tray and pops up at customizable intervals, making it perfect for:
- [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab) - **Daily reflection and mindfulness practice**
- [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook) - **Tracking your thoughts and activities**
- **Building a journaling habit**
- **Capturing ideas before they fade**
For help getting started with Flutter development, view the ![Journaler Screenshot](docs/screenshots/journaler_main.png)
[online documentation](https://docs.flutter.dev/), which offers tutorials,
samples, guidance on mobile development, and a full API reference. ## Features
- **Automated Reminders**: Customizable popup intervals to remind you to journal
- **System Tray Integration**: Runs silently in the background
- **Scratch Pad**: Quick notes area for temporary thoughts
- **Journal History**: Browse through past entries
- **Full-Text Search**: Find entries containing specific words or phrases
- **Dark/Light Theme**: Automatically follows your system theme
- **Notification Sounds**: Audio cues when it's time to journal
- **Single Instance**: Only one copy of the app runs at a time
## Usage
### Getting Started
1. **First Launch**: When you first launch Journaler, it will start in the system tray
2. **Writing Entries**: The app will automatically pop up at set intervals (default: 20 minutes)
3. **Manual Access**: Click the system tray icon to show the app manually
4. **"Multiple" instances**: Running another instance will make the existing instance show itself
5. **Saving entries**: Entries are saved automatically when the popup is closed either manually or by hitting the escape key
### Main Interface
1. **Current Entry**: Write your thoughts for the current moment
2. **Previous Entry**: View what you wrote last time
3. **Scratch Pad**: Space for quick notes that persist between sessions
4. **Navigation**: Browse through your past entries
5. **Search**: Find specific content in your journal history (May also be invoked by CTRL-F)
![Journaler Interface](docs/screenshots/journaler_annotated.png)
### Settings
You can adjust:
- **Popup Interval**: Change how frequently Journaler reminds you
- **Notification Sound**: Select your preferred audio reminder
- **Volume**: Adjust the sound level of reminders
## Data privacy
All your journal entries are stored securely on your local machine:
- **Windows**: `C:\Users\YourUsername\.journaler\`
No network requests are made, no data sent or received, your data is yours alone.
Data is stored in an SQLite database with a very simple schema that can be fucked with easily.
## Development
This application is built with Flutter. If you want to build from source:
1. Install Flutter (version 3.7.2 or higher)
2. Clone the repository
3. Run `flutter pub get` to install dependencies
4. Use `flutter run` to run the application in debug mode
## Known issues
I don't know whether or not the sound customization works.<br>
In theory only 2 sound files are bundled with the app so, in theory, you may only choose between those two.<br>
But I don't know how flutter handles assets, currently I am not very motivated to figure this out.
Scrolling through long notes is difficult since we're intercepting the scroll event and using it to scroll through notes themselves.<br>
This, again, is not really an issue for me so I'm not very motivated to fix it.
Showing and focusing the window is really annoying for "gamers" and generally whoever is doing anything at the time.<br>
But I have no better ideas.<br>
If we show the window without focusing it, it will still be on top and, if not focused, will be even more annoying because to close it you have to click on it (to focus it) and THEN close it.<br>
If we only play the sound as a reminder we could miss it or simply get used to the sound and learn to ignore it.<br>
I have no ideas for a better solution.

Binary file not shown.

BIN
docs/screenshots/journaler_annotated.png (Stored with Git LFS) Normal file

Binary file not shown.

BIN
docs/screenshots/journaler_main.png (Stored with Git LFS) Normal file

Binary file not shown.

View File

@@ -6,10 +6,23 @@ import 'package:sqflite_common_ffi/sqflite_ffi.dart';
const settingsDir = '.journaler'; const settingsDir = '.journaler';
const dbFileName = 'journaler.db'; const dbFileName = 'journaler.db';
// Add this at the top level
typedef ShowMessageCallback = void Function(String message);
class DB { class DB {
static late Database db; static late Database db;
static bool _hasIcuSupport = false;
static ShowMessageCallback? _showMessage;
static const String _schema = ''' // Add this to register the callback
static void registerMessageCallback(ShowMessageCallback callback) {
_showMessage = callback;
}
// Add this method to check ICU status
static bool get hasIcuSupport => _hasIcuSupport;
static const String _baseSchema = '''
CREATE TABLE IF NOT EXISTS notes ( CREATE TABLE IF NOT EXISTS notes (
id INTEGER PRIMARY KEY AUTOINCREMENT, id INTEGER PRIMARY KEY AUTOINCREMENT,
date TEXT DEFAULT CURRENT_TIMESTAMP, date TEXT DEFAULT CURRENT_TIMESTAMP,
@@ -17,10 +30,7 @@ CREATE TABLE IF NOT EXISTS notes (
); );
CREATE INDEX IF NOT EXISTS idx_notes_date ON notes (date); CREATE INDEX IF NOT EXISTS idx_notes_date ON notes (date);
CREATE UNIQUE INDEX IF NOT EXISTS idx_notes_date_unique ON notes (date); CREATE UNIQUE INDEX IF NOT EXISTS idx_notes_date_unique ON notes (date);
-- Todos are "static", we are only interested in the latest entry
-- ie. they're not really a list but instead a string
-- But we will also keep a history of all todos
-- Because we might as well
CREATE TABLE IF NOT EXISTS scratches ( CREATE TABLE IF NOT EXISTS scratches (
id INTEGER PRIMARY KEY AUTOINCREMENT, id INTEGER PRIMARY KEY AUTOINCREMENT,
date TEXT DEFAULT CURRENT_TIMESTAMP, date TEXT DEFAULT CURRENT_TIMESTAMP,
@@ -33,15 +43,31 @@ CREATE TABLE IF NOT EXISTS settings (
key TEXT PRIMARY KEY NOT NULL, key TEXT PRIMARY KEY NOT NULL,
value TEXT NOT NULL value TEXT NOT NULL
); );
''';
-- Create virtual FTS5 table for searching notes content static const String _ftsSchemaWithIcu = '''
-- Create virtual FTS5 table with Unicode tokenizer
CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts USING fts5( CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts USING fts5(
content, content,
date, date,
content='notes', content='notes',
content_rowid='id' content_rowid='id',
tokenize='unicode61 remove_diacritics 2 tokenchars "\u0401\u0451\u0410-\u044f"'
); );
''';
static const String _ftsSchemaBasic = '''
-- Create virtual FTS5 table with basic tokenizer
CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts USING fts5(
content,
date,
content='notes',
content_rowid='id',
tokenize='unicode61'
);
''';
static const String _ftsTriggers = '''
-- Trigger to keep FTS table in sync with notes table when inserting -- Trigger to keep FTS table in sync with notes table when inserting
CREATE TRIGGER IF NOT EXISTS notes_ai AFTER INSERT ON notes BEGIN CREATE TRIGGER IF NOT EXISTS notes_ai AFTER INSERT ON notes BEGIN
INSERT INTO notes_fts(rowid, content, date) VALUES (new.id, new.content, new.date); INSERT INTO notes_fts(rowid, content, date) VALUES (new.id, new.content, new.date);
@@ -58,6 +84,124 @@ CREATE TRIGGER IF NOT EXISTS notes_au AFTER UPDATE ON notes BEGIN
END; END;
'''; ''';
// Add this method to check ICU status with visible feedback
static Future<bool> checkAndShowIcuStatus() async {
final status = _hasIcuSupport;
final message =
status
? 'ICU support is enabled - Full Unicode search available'
: 'ICU support is not available - Unicode search will be limited\n'
'Looking for sqlite3_icu.dll in the application directory';
debugPrint(message);
_showMessage?.call(message);
return status;
}
static Future<bool> _checkIcuSupport(Database db) async {
try {
debugPrint(
'\n================== CHECKING UNICODE SUPPORT ==================',
);
// Test Unicode support with a simple query
try {
debugPrint('Testing Unicode support...');
// Test with some Cyrillic characters
await db.rawQuery("SELECT 'тест' LIKE '%ест%'");
await db.rawQuery("SELECT 'ТЕСТ' LIKE '%ЕСТ%'");
final message = 'Unicode support is available';
debugPrint(message);
_showMessage?.call(message);
return true;
} catch (e, stackTrace) {
final message = 'Unicode support test failed';
debugPrint('$message:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
// Try to get SQLite version and compile options for debugging
try {
final version = await db.rawQuery('SELECT sqlite_version()');
final compileOpts = await db.rawQuery('PRAGMA compile_options');
debugPrint('SQLite version: $version');
debugPrint('SQLite compile options: $compileOpts');
} catch (e) {
debugPrint('Could not get SQLite info: $e');
}
_showMessage?.call(message);
return false;
}
} catch (e, stackTrace) {
final message = 'Failed to test Unicode support';
debugPrint('$message:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
_showMessage?.call(message);
return false;
} finally {
debugPrint('=====================================================\n');
}
}
static Future<void> _recreateFtsTable(Database db, bool useIcu) async {
debugPrint('Updating FTS table configuration...');
try {
// Start a transaction to ensure data safety
await db.transaction((txn) async {
// First, create a temporary table with the new configuration
final tempTableName = 'notes_fts_temp';
final schema = useIcu ? _ftsSchemaWithIcu : _ftsSchemaBasic;
// Create temp table with new configuration
await txn.execute(schema.replaceAll('notes_fts', tempTableName));
// Copy data from old FTS table if it exists
try {
debugPrint('Copying existing FTS data to temporary table...');
await txn.execute('''
INSERT INTO $tempTableName(rowid, content, date)
SELECT rowid, content, date FROM notes_fts
''');
debugPrint('Data copied successfully');
} catch (e) {
debugPrint('No existing FTS data to copy: $e');
}
// Drop old triggers
debugPrint('Updating triggers...');
await txn.execute('DROP TRIGGER IF EXISTS notes_ai');
await txn.execute('DROP TRIGGER IF EXISTS notes_ad');
await txn.execute('DROP TRIGGER IF EXISTS notes_au');
// Drop old FTS table
await txn.execute('DROP TABLE IF EXISTS notes_fts');
// Rename temp table to final name
await txn.execute('ALTER TABLE $tempTableName RENAME TO notes_fts');
// Create new triggers
await txn.execute(_ftsTriggers);
// Rebuild FTS index from notes table to ensure consistency
debugPrint('Rebuilding FTS index from notes table...');
await txn.execute('''
INSERT OR REPLACE INTO notes_fts(rowid, content, date)
SELECT id, content, date FROM notes
''');
debugPrint('FTS table update completed successfully');
});
} catch (e, stackTrace) {
debugPrint('Error updating FTS table:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
rethrow;
}
}
static Future<String> _getDatabasePath() async { static Future<String> _getDatabasePath() async {
debugPrint('Attempting to get database path...'); debugPrint('Attempting to get database path...');
if (Platform.isWindows || Platform.isLinux) { if (Platform.isWindows || Platform.isLinux) {
@@ -88,26 +232,100 @@ END;
static Future<void> init() async { static Future<void> init() async {
debugPrint('Starting database initialization...'); debugPrint('Starting database initialization...');
// Initialize SQLite FFI
sqfliteFfiInit(); sqfliteFfiInit();
final databaseFactory = databaseFactoryFfi;
// Create a temporary database to check version
try {
debugPrint(
'\n================== SQLITE VERSION CHECK ==================',
);
final tempDb = await databaseFactory.openDatabase(
':memory:',
options: OpenDatabaseOptions(
version: 1,
onCreate: (db, version) async {
final results = await db.rawQuery('SELECT sqlite_version()');
debugPrint('SQLite version: ${results.first.values.first}');
final compileOpts = await db.rawQuery('PRAGMA compile_options');
debugPrint('SQLite compile options:');
for (var opt in compileOpts) {
debugPrint(' ${opt.values.first}');
}
},
),
);
await tempDb.close();
debugPrint('=====================================================\n');
} catch (e, stackTrace) {
debugPrint('Error checking SQLite version:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
}
await databaseFactory.setDatabasesPath(
await databaseFactory.getDatabasesPath(),
);
final dbPath = await _getDatabasePath(); final dbPath = await _getDatabasePath();
debugPrint('Database path: $dbPath'); debugPrint('Database path: $dbPath');
try { try {
db = await databaseFactoryFfi.openDatabase( db = await databaseFactory.openDatabase(
dbPath, dbPath,
options: OpenDatabaseOptions( options: OpenDatabaseOptions(
version: 1, version: 2,
onConfigure: (db) async {
debugPrint('Configuring database...');
await db.execute('PRAGMA foreign_keys = ON');
debugPrint('Database configured');
},
onCreate: (db, version) async { onCreate: (db, version) async {
debugPrint('Creating database schema...'); debugPrint('Creating database schema...');
await db.execute(_schema); await db.execute(_baseSchema);
// Check for Unicode support on first creation
_hasIcuSupport = await _checkIcuSupport(db);
await _recreateFtsTable(db, _hasIcuSupport);
debugPrint('Database schema created successfully'); debugPrint('Database schema created successfully');
}, },
onOpen: (db) async {
debugPrint('Database opened, checking Unicode support...');
try {
_hasIcuSupport = await _checkIcuSupport(db);
debugPrint('Unicode support check completed: $_hasIcuSupport');
} catch (e, stackTrace) {
debugPrint('Error during Unicode support check:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
rethrow;
}
},
onUpgrade: (db, oldVersion, newVersion) async {
debugPrint('Upgrading database from $oldVersion to $newVersion');
if (oldVersion < 2) {
// Check for Unicode support during upgrade
_hasIcuSupport = await _checkIcuSupport(db);
await _recreateFtsTable(db, _hasIcuSupport);
}
},
), ),
); );
debugPrint('Database opened and initialized');
} catch (e) { // Store Unicode support status in settings for future reference
debugPrint('Failed to initialize database: $e'); await setSetting('has_icu_support', _hasIcuSupport.toString());
debugPrint(
'Database opened and initialized (Unicode support: $_hasIcuSupport)',
);
} catch (e, stackTrace) {
debugPrint('Failed to initialize database:');
debugPrint('Error: $e');
debugPrint('Stack trace: $stackTrace');
rethrow; rethrow;
} }
} }
@@ -141,7 +359,6 @@ END;
return []; return [];
} }
// Process the query for partial word matching
// Split into individual terms, filter empty ones // Split into individual terms, filter empty ones
List<String> terms = List<String> terms =
query query
@@ -154,16 +371,16 @@ END;
return []; return [];
} }
// Add wildcards to each term for prefix matching (e.g., "fuck*" will match "fucked") // Process terms for FTS5 query using proper tokenization
// Join terms with AND for all-term matching (results must contain ALL terms)
String ftsQuery = terms String ftsQuery = terms
.map((term) { .map((term) {
// Remove any special characters that might break the query // Remove dangerous characters but preserve Unicode
String sanitizedTerm = term.replaceAll(RegExp(r'[^\w]'), ''); String sanitizedTerm = term.replaceAll(RegExp(r'''['"]'''), '');
if (sanitizedTerm.isEmpty) return ''; if (sanitizedTerm.isEmpty) return '';
// Add wildcard for stemming/prefix matching // Use proper FTS5 syntax: each word becomes a separate token with prefix matching
return '$sanitizedTerm*'; List<String> words = sanitizedTerm.split(RegExp(r'\s+'));
return words.map((word) => '$word*').join(' OR ');
}) })
.where((term) => term.isNotEmpty) .where((term) => term.isNotEmpty)
.join(' AND '); .join(' AND ');
@@ -175,24 +392,25 @@ END;
debugPrint('FTS query: "$ftsQuery"'); debugPrint('FTS query: "$ftsQuery"');
// Execute the FTS query with AND logic // Execute the FTS query
final List<Map<String, dynamic>> results = await db.rawQuery( final List<Map<String, dynamic>> results = await db.rawQuery(
''' '''
SELECT n.id, n.date, n.content, snippet(notes_fts, 0, '<b>', '</b>', '...', 20) as snippet SELECT n.id, n.date, n.content,
snippet(notes_fts, -1, '<b>', '</b>', '...', 64) as snippet
FROM notes_fts FROM notes_fts
JOIN notes n ON notes_fts.rowid = n.id JOIN notes n ON notes_fts.rowid = n.id
WHERE notes_fts MATCH ? WHERE notes_fts MATCH ?
ORDER BY n.date DESC ORDER BY rank
LIMIT 100 LIMIT 100
''', ''',
[ftsQuery], [ftsQuery],
); );
debugPrint('Search query "$ftsQuery" returned ${results.length} results'); debugPrint('Search returned ${results.length} results');
return results; return results;
} catch (e) { } catch (e, stackTrace) {
debugPrint('Search failed: $e'); debugPrint('Search failed: $e');
// Return empty results rather than crashing on malformed queries debugPrint('Stack trace: $stackTrace');
return []; return [];
} }
} }

View File

@@ -1,4 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:io';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:journaler/db.dart'; import 'package:journaler/db.dart';
@@ -8,6 +9,8 @@ import 'package:window_manager/window_manager.dart';
import 'package:audioplayers/audioplayers.dart'; import 'package:audioplayers/audioplayers.dart';
import 'package:flutter/gestures.dart'; import 'package:flutter/gestures.dart';
import 'dart:math'; import 'dart:math';
import 'package:path/path.dart' as path;
import 'package:ps_list/ps_list.dart';
// TODO: Sound does not play when ran from a different workdir? Weird // TODO: Sound does not play when ran from a different workdir? Weird
// TODO: Fix saving the same scratch over and over again // TODO: Fix saving the same scratch over and over again
@@ -16,9 +19,53 @@ import 'dart:math';
const Duration _defaultPopupInterval = Duration(minutes: 20); const Duration _defaultPopupInterval = Duration(minutes: 20);
const String _defaultNotificationSound = 'MeetTheSniper.mp3'; const String _defaultNotificationSound = 'MeetTheSniper.mp3';
// Flag to indicate if there's a pending show request from a secondary instance
bool _pendingShowRequest = false;
// Global navigator key to access the navigator state from anywhere
final GlobalKey<NavigatorState> navigatorKey = GlobalKey<NavigatorState>();
/// Paths for IPC file in system temp directory
class AppFiles {
static String get ipcFilePath =>
path.join(Directory.systemTemp.path, 'journaler_ipc.txt');
}
void main() async { void main() async {
await DB.init();
WidgetsFlutterBinding.ensureInitialized(); WidgetsFlutterBinding.ensureInitialized();
final ipcFile = File(AppFiles.ipcFilePath);
if (await alreadyRunning()) {
await signalPrimaryInstanceAndExit(ipcFile);
} else {
await runPrimaryInstance(ipcFile);
}
}
Future<bool> alreadyRunning() async {
final processes = await PSList.getRunningProcesses();
final executable = Platform.resolvedExecutable;
final executableName = path.basename(executable);
final journalers =
processes.where((process) => process == executableName).toList();
debugPrint("Journalers: $journalers");
return journalers.length > 1;
}
/// Run the primary instance of the application
Future<void> runPrimaryInstance(File ipcFile) async {
debugPrint("Starting as primary instance");
// Create or clear the IPC file
if (!await ipcFile.exists()) {
await ipcFile.create();
}
// Start a watcher for the IPC file to detect signals from other instances
startIpcWatcher(ipcFile);
// Initialize the app
await DB.init();
await windowManager.ensureInitialized(); await windowManager.ensureInitialized();
WindowOptions windowOptions = const WindowOptions( WindowOptions windowOptions = const WindowOptions(
@@ -34,6 +81,114 @@ void main() async {
}); });
runApp(const JournalerApp()); runApp(const JournalerApp());
// Set up termination handlers to clean up resources
setupTerminationHandlers(ipcFile);
}
/// Handle termination signals by cleaning up IPC file
void setupTerminationHandlers(File ipcFile) {
// Handle normal termination
ProcessSignal.sigterm.watch().listen((_) async {
await cleanupFiles(ipcFile);
exit(0);
});
// Handle Windows and Linux CTRL+C termination
if (Platform.isWindows || Platform.isLinux) {
ProcessSignal.sigint.watch().listen((_) async {
await cleanupFiles(ipcFile);
exit(0);
});
}
}
/// Clean up IPC file
Future<void> cleanupFiles(File ipcFile) async {
try {
if (await ipcFile.exists()) {
await ipcFile.delete();
}
debugPrint("Removed IPC file on app termination");
} catch (e) {
debugPrint("Error cleaning up files: $e");
}
}
/// Signal the primary instance to show itself, then exit
Future<void> signalPrimaryInstanceAndExit(File ipcFile) async {
debugPrint(
"Another instance is already running. Signaling it to show window.",
);
try {
// Create the IPC file if it doesn't exist
if (!await ipcFile.exists()) {
await ipcFile.create();
}
// Write a 'show' command with a timestamp
final timestamp = DateTime.now().millisecondsSinceEpoch;
await ipcFile.writeAsString('show:$timestamp');
debugPrint("Signal sent to primary instance via IPC file");
// Give the primary instance a moment to process the signal
await Future.delayed(const Duration(seconds: 2));
} catch (e) {
debugPrint("Error signaling primary instance: $e");
}
// Exit this instance
debugPrint("Exiting secondary instance");
exit(0);
}
/// Start watching the IPC file for signals from other instances
void startIpcWatcher(File ipcFile) {
// Initial check of the file content
checkIpcFileContent(ipcFile);
// Set up a file watcher
final watcher = ipcFile.watch();
watcher.listen((event) {
if (event.type == FileSystemEvent.modify) {
checkIpcFileContent(ipcFile);
}
});
// Also set up a periodic check as a fallback
Timer.periodic(const Duration(milliseconds: 200), (_) {
checkIpcFileContent(ipcFile);
});
debugPrint("IPC file watcher started");
}
/// Check the IPC file content for commands
Future<void> checkIpcFileContent(File ipcFile) async {
try {
if (await ipcFile.exists()) {
final content = await ipcFile.readAsString();
if (content.isNotEmpty && content.startsWith('show:')) {
// Clear the file immediately to avoid processing the same command multiple times
await ipcFile.writeAsString('');
// Process the show command
if (MainPageState.instance != null) {
debugPrint("Received show command - making window visible");
MainPageState.instance!.showWindow();
} else {
debugPrint(
"MainPageState not initialized yet - setting pending flag",
);
_pendingShowRequest = true;
}
}
}
} catch (e) {
debugPrint("Error processing IPC file: $e");
}
} }
class JournalerApp extends StatelessWidget { class JournalerApp extends StatelessWidget {
@@ -114,6 +269,7 @@ class JournalerApp extends StatelessWidget {
themeMode: ThemeMode.system, themeMode: ThemeMode.system,
home: const MainPage(), home: const MainPage(),
debugShowCheckedModeBanner: false, debugShowCheckedModeBanner: false,
navigatorKey: navigatorKey,
); );
} }
} }
@@ -126,6 +282,9 @@ class MainPage extends StatefulWidget {
} }
class MainPageState extends State<MainPage> with WindowListener { class MainPageState extends State<MainPage> with WindowListener {
// Static reference to the current instance
static MainPageState? instance;
final SystemTray _systemTray = SystemTray(); final SystemTray _systemTray = SystemTray();
final Menu _menu = Menu(); final Menu _menu = Menu();
final AudioPlayer _audioPlayer = AudioPlayer(); final AudioPlayer _audioPlayer = AudioPlayer();
@@ -157,16 +316,31 @@ class MainPageState extends State<MainPage> with WindowListener {
@override @override
void initState() { void initState() {
super.initState(); super.initState();
// Store reference to this instance
instance = this;
windowManager.addListener(this); windowManager.addListener(this);
_initSystemTray(); _initSystemTray();
_loadData(); _loadData();
_loadVolume(); _loadVolume();
windowManager.setPreventClose(true); windowManager.setPreventClose(true);
_setWindowConfig(); _setWindowConfig();
// Check if there's a pending show request from another instance
WidgetsBinding.instance.addPostFrameCallback((_) {
if (_pendingShowRequest) {
debugPrint("Processing pending show request during initialization");
_pendingShowRequest = false;
showWindow();
}
});
} }
@override @override
void dispose() { void dispose() {
// Clear reference to this instance
if (instance == this) {
instance = null;
}
windowManager.removeListener(this); windowManager.removeListener(this);
_popupTimer?.cancel(); _popupTimer?.cancel();
_debounceTimer?.cancel(); _debounceTimer?.cancel();
@@ -183,8 +357,9 @@ class MainPageState extends State<MainPage> with WindowListener {
} }
@override @override
void onWindowClose() { void onWindowClose() async {
_saveData(); // Save data when window is closed
await _saveData();
windowManager.hide(); windowManager.hide();
} }
@@ -210,7 +385,7 @@ class MainPageState extends State<MainPage> with WindowListener {
_systemTray.registerSystemTrayEventHandler((eventName) { _systemTray.registerSystemTrayEventHandler((eventName) {
debugPrint("System Tray Event: $eventName"); debugPrint("System Tray Event: $eventName");
if (eventName == kSystemTrayEventClick) { if (eventName == kSystemTrayEventClick) {
_showWindow(); showWindow();
} else if (eventName == kSystemTrayEventRightClick) { } else if (eventName == kSystemTrayEventRightClick) {
_systemTray.popUpContextMenu(); _systemTray.popUpContextMenu();
} }
@@ -220,26 +395,56 @@ class MainPageState extends State<MainPage> with WindowListener {
void _startPopupTimer() { void _startPopupTimer() {
_popupTimer?.cancel(); _popupTimer?.cancel();
_popupTimer = Timer.periodic(_currentPopupInterval, (timer) { _popupTimer = Timer.periodic(_currentPopupInterval, (timer) {
_showWindow(); showWindow();
}); });
debugPrint( debugPrint(
"Popup timer started with interval: ${_currentPopupInterval.inMinutes} minutes", "Popup timer started with interval: ${_currentPopupInterval.inMinutes} minutes",
); );
} }
Future<void> _showWindow() async { Future<void> showWindow() async {
_loadData(); _loadData();
bool wasVisible = await windowManager.isVisible(); try {
if (!wasVisible) { bool wasVisible = await windowManager.isVisible();
await windowManager.setSize(const Size(1600, 900)); debugPrint("Current window visibility: $wasVisible");
await windowManager.center();
await windowManager.show(); if (!wasVisible) {
await windowManager.focus(); // First make sure the window has the right size and position
_currentEntryFocusNode.requestFocus(); await windowManager.setSize(const Size(1600, 900));
await _playSound(); await windowManager.center();
} else {
await windowManager.focus(); // Now show and focus
_currentEntryFocusNode.requestFocus(); await windowManager.show();
await Future.delayed(const Duration(milliseconds: 100)); // Short delay
await windowManager.focus();
// Set input focus
_currentEntryFocusNode.requestFocus();
// Play notification sound
await _playSound();
debugPrint("Window made visible and focused");
} else {
// Already visible, just focus
await windowManager.focus();
_currentEntryFocusNode.requestFocus();
debugPrint("Window already visible, just focused");
}
// Verify the window is now visible
bool isNowVisible = await windowManager.isVisible();
debugPrint("Window visibility after show attempt: $isNowVisible");
if (!isNowVisible) {
debugPrint(
"WARNING: Window still not visible after show attempt, trying again",
);
await windowManager.show();
await windowManager.focus();
}
} catch (e) {
debugPrint("Error showing window: $e");
} }
} }
@@ -288,6 +493,12 @@ class MainPageState extends State<MainPage> with WindowListener {
Future<void> _goToPreviousNote() async { Future<void> _goToPreviousNote() async {
if (!_canGoPrevious || _currentlyDisplayedNote == null) return; if (!_canGoPrevious || _currentlyDisplayedNote == null) return;
// Save the current note content before navigating away
if (_currentlyDisplayedNote != null) {
_currentlyDisplayedNote!.content = _previousEntryController.text;
await updateNote(_currentlyDisplayedNote!);
}
final prevNote = await getPreviousNote(_currentlyDisplayedNote!.date); final prevNote = await getPreviousNote(_currentlyDisplayedNote!.date);
if (prevNote != null) { if (prevNote != null) {
setState(() { setState(() {
@@ -301,6 +512,12 @@ class MainPageState extends State<MainPage> with WindowListener {
Future<void> _goToNextNote() async { Future<void> _goToNextNote() async {
if (!_canGoNext || _currentlyDisplayedNote == null) return; if (!_canGoNext || _currentlyDisplayedNote == null) return;
// Save the current note content before navigating away
if (_currentlyDisplayedNote != null) {
_currentlyDisplayedNote!.content = _previousEntryController.text;
await updateNote(_currentlyDisplayedNote!);
}
final nextNote = await getNextNote(_currentlyDisplayedNote!.date); final nextNote = await getNextNote(_currentlyDisplayedNote!.date);
if (nextNote != null) { if (nextNote != null) {
setState(() { setState(() {
@@ -333,11 +550,10 @@ class MainPageState extends State<MainPage> with WindowListener {
final scratch = await getLatestScratch(); final scratch = await getLatestScratch();
_scratchController.text = scratch?.content ?? ""; _scratchController.text = scratch?.content ?? "";
_currentEntryController.text = "";
await _checkNavigation(); await _checkNavigation();
debugPrint("Data loaded."); debugPrint("Data loaded");
} }
// Load volume setting from database // Load volume setting from database
@@ -359,18 +575,41 @@ class MainPageState extends State<MainPage> with WindowListener {
debugPrint("Volume saved: $_volume"); debugPrint("Volume saved: $_volume");
} }
void _saveData() async { Future<void> _saveData() async {
String previousEntry = _previousEntryController.text; String previousEntry = _previousEntryController.text;
String currentEntry = _currentEntryController.text; String currentEntry = _currentEntryController.text;
String scratchContent = _scratchController.text; String scratchContent = _scratchController.text;
String intervalStr = _intervalController.text; String intervalStr = _intervalController.text;
String soundStr = _soundController.text; String soundStr = _soundController.text;
await createNote(currentEntry); // Handle current entry
if (currentEntry.isNotEmpty) {
await createNote(currentEntry);
_currentEntryController.clear(); // Clear the input field after saving
}
// Handle scratch pad
await createScratch(scratchContent); await createScratch(scratchContent);
if (previousNote != null) {
previousNote!.content = previousEntry; // Handle previous/currently displayed note
await updateNote(previousNote!); if (_currentlyDisplayedNote != null) {
_currentlyDisplayedNote!.content = previousEntry;
await updateNote(_currentlyDisplayedNote!);
// If the note was deleted (due to being empty), update the UI state
if (previousEntry.isEmpty) {
// Check if we need to navigate to another note
Note? nextNote = await getLatestNote();
setState(() {
_currentlyDisplayedNote = nextNote;
if (nextNote != null) {
_previousEntryController.text = nextNote.content;
} else {
_previousEntryController.text = "";
}
});
await _checkNavigation();
}
} }
int newIntervalMinutes = int newIntervalMinutes =
@@ -510,9 +749,20 @@ class MainPageState extends State<MainPage> with WindowListener {
try { try {
final results = await searchNotes(trimmedQuery); final results = await searchNotes(trimmedQuery);
// Filter out empty notes (which may exist in the search index but were deleted)
final filteredResults =
results
.where((note) => note.content.isNotEmpty)
.toList();
// Sort by date, newest first
filteredResults.sort(
(a, b) => b.date.compareTo(a.date),
);
// Important: update the dialog state after search completes // Important: update the dialog state after search completes
dialogSetState(() { dialogSetState(() {
_searchResults = results; _searchResults = filteredResults;
_isSearching = false; _isSearching = false;
}); });
} catch (e) { } catch (e) {
@@ -555,7 +805,7 @@ class MainPageState extends State<MainPage> with WindowListener {
vertical: 2, vertical: 2,
), // Tighter padding ), // Tighter padding
title: Text( title: Text(
note.date, note.displayDate,
style: const TextStyle( style: const TextStyle(
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
fontSize: fontSize:
@@ -581,9 +831,20 @@ class MainPageState extends State<MainPage> with WindowListener {
), // Smaller font for content ), // Smaller font for content
), ),
isThreeLine: true, isThreeLine: true,
onTap: () { onTap: () async {
// Save current note if needed
if (_currentlyDisplayedNote !=
null) {
_currentlyDisplayedNote!.content =
_previousEntryController.text;
await updateNote(
_currentlyDisplayedNote!,
);
}
// Navigate to the selected note
Navigator.of(context).pop(); Navigator.of(context).pop();
this.setState(() { setState(() {
_currentlyDisplayedNote = note; _currentlyDisplayedNote = note;
_previousEntryController.text = _previousEntryController.text =
note.content; note.content;
@@ -649,6 +910,207 @@ class MainPageState extends State<MainPage> with WindowListener {
); );
} }
// Show cleanup dialog
void _showCleanupDialog() async {
double sensitivity = 0.7; // Default 70%
final problematicEntries = await findProblematicEntries(
maxCharPercentage: sensitivity,
);
if (!mounted) return;
showDialog(
context: context,
builder: (BuildContext context) {
return StatefulBuilder(
builder: (context, dialogSetState) {
return AlertDialog(
title: const Text('Cleanup Problematic Entries'),
content: SizedBox(
width: MediaQuery.of(context).size.width * 0.7,
height: MediaQuery.of(context).size.height * 0.7,
child: Column(
children: [
Row(
children: [
const Text('Sensitivity: '),
Expanded(
child: Slider(
value: sensitivity,
min: 0.3,
max: 0.9,
divisions: 12,
label: '${(sensitivity * 100).toInt()}%',
onChanged: (value) async {
dialogSetState(() {
sensitivity =
(value * 100).round() /
100; // Round to 2 decimal places
});
// Refresh results with new sensitivity
final newResults = await findProblematicEntries(
maxCharPercentage: sensitivity,
);
dialogSetState(() {
problematicEntries.clear();
problematicEntries.addAll(newResults);
});
},
),
),
Text('${(sensitivity * 100).toInt()}%'),
],
),
Text(
'Found ${problematicEntries.length} potentially problematic entries',
style: const TextStyle(fontWeight: FontWeight.bold),
),
const SizedBox(height: 16),
Expanded(
child:
problematicEntries.isEmpty
? const Center(
child: Text('No problematic entries found!'),
)
: ListView.builder(
itemCount: problematicEntries.length,
itemBuilder: (context, index) {
final note = problematicEntries[index];
return Card(
margin: const EdgeInsets.only(bottom: 8),
child: ListTile(
title: Text(
note.displayDate,
style: const TextStyle(
fontWeight: FontWeight.bold,
),
),
subtitle: Column(
crossAxisAlignment:
CrossAxisAlignment.start,
children: [
Text(
'Reason: ${note.problemReason}',
style: TextStyle(
color: Colors.red[700],
fontWeight: FontWeight.bold,
),
),
const SizedBox(height: 4),
Text(note.content),
],
),
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
IconButton(
icon: const Icon(Icons.visibility),
tooltip: 'View in main window',
onPressed: () {
// Close the dialog
Navigator.of(context).pop();
// Navigate to the note in main window
setState(() {
_currentlyDisplayedNote = note;
_previousEntryController.text =
note.content;
});
_checkNavigation();
},
),
IconButton(
icon: const Icon(Icons.delete),
color: Colors.red,
tooltip:
'Delete (hold Shift to skip confirmation)',
onPressed: () async {
// Check if shift is pressed
final isShiftPressed =
HardwareKeyboard
.instance
.isShiftPressed;
bool shouldDelete =
isShiftPressed;
if (!isShiftPressed) {
// Show confirmation dialog
shouldDelete =
await showDialog<bool>(
context: context,
builder:
(
context,
) => AlertDialog(
title: const Text(
'Delete Entry?',
),
content: const Text(
'Are you sure you want to delete this entry? This action cannot be undone.',
),
actions: [
TextButton(
onPressed:
() => Navigator.of(
context,
).pop(
false,
),
child:
const Text(
'Cancel',
),
),
TextButton(
onPressed:
() => Navigator.of(
context,
).pop(true),
child:
const Text(
'Delete',
),
),
],
),
) ??
false;
}
if (shouldDelete) {
await deleteNote(note.date);
dialogSetState(() {
problematicEntries.removeAt(
index,
);
});
}
},
),
],
),
),
);
},
),
),
],
),
),
actions: [
TextButton(
onPressed: () {
Navigator.of(context).pop();
},
child: const Text('Close'),
),
],
);
},
);
},
);
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
// Wrap Scaffold with RawKeyboardListener as workaround for Escape key // Wrap Scaffold with RawKeyboardListener as workaround for Escape key
@@ -680,6 +1142,12 @@ class MainPageState extends State<MainPage> with WindowListener {
appBar: AppBar( appBar: AppBar(
title: const Text('Journaler'), title: const Text('Journaler'),
actions: <Widget>[ actions: <Widget>[
// Add cleanup button
IconButton(
icon: const Icon(Icons.cleaning_services),
tooltip: 'Cleanup Problematic Entries',
onPressed: _showCleanupDialog,
),
// Add search button // Add search button
IconButton( IconButton(
icon: const Icon(Icons.search), icon: const Icon(Icons.search),
@@ -795,10 +1263,10 @@ class MainPageState extends State<MainPage> with WindowListener {
children: [ children: [
Expanded( Expanded(
child: Text( child: Text(
_currentlyDisplayedNote?.date == _currentlyDisplayedNote?.displayDate ==
previousNote?.date previousNote?.displayDate
? 'Previous Entry (Latest)' ? 'Previous Entry: ${_currentlyDisplayedNote?.displayDate ?? 'N/A'}'
: 'Entry: ${_currentlyDisplayedNote?.date ?? 'N/A'}', : 'Entry: ${_currentlyDisplayedNote?.displayDate ?? 'N/A'}',
style: TextStyle( style: TextStyle(
fontSize: 18, fontSize: 18,
color: Colors.grey, color: Colors.grey,
@@ -821,25 +1289,23 @@ class MainPageState extends State<MainPage> with WindowListener {
Expanded( Expanded(
child: TextField( child: TextField(
controller: _previousEntryController, controller: _previousEntryController,
readOnly: readOnly: false, // Always allow editing
_currentlyDisplayedNote?.date !=
previousNote?.date,
maxLines: null, maxLines: null,
expands: true, expands: true,
style: Theme.of(context).textTheme.bodyMedium, style: Theme.of(context).textTheme.bodyMedium,
decoration: InputDecoration( decoration: InputDecoration(
hintText: hintText:
_currentlyDisplayedNote?.date != _currentlyDisplayedNote?.displayDate !=
previousNote?.date previousNote?.displayDate
? 'Viewing note from ${_currentlyDisplayedNote?.date} (Read-Only)' ? 'Viewing note from ${_currentlyDisplayedNote?.displayDate} (Editable)'
: 'Latest Note', : 'Latest Note: ${_currentlyDisplayedNote?.displayDate ?? 'N/A'}',
border: const OutlineInputBorder(), border: const OutlineInputBorder(),
filled: filled:
_currentlyDisplayedNote?.date != _currentlyDisplayedNote?.displayDate !=
previousNote?.date, previousNote?.displayDate,
fillColor: fillColor:
_currentlyDisplayedNote?.date != _currentlyDisplayedNote?.displayDate !=
previousNote?.date previousNote?.displayDate
? Colors.grey.withOpacity(0.1) ? Colors.grey.withOpacity(0.1)
: null, : null,
), ),

View File

@@ -1,12 +1,25 @@
import 'package:journaler/db.dart'; import 'package:journaler/db.dart';
import 'package:intl/intl.dart';
class Note { class Note {
final String date; final String date;
late final String displayDate;
String content; String content;
String? String? snippet;
snippet; // Optional field to hold highlighted snippets for search results bool isProblematic;
String problemReason;
Note({required this.date, required this.content, this.snippet}); Note({
required this.date,
required this.content,
this.snippet,
this.isProblematic = false,
this.problemReason = '',
}) {
final dtUtc = DateFormat('yyyy-MM-dd HH:mm:ss').parse(date, true);
final dtLocal = dtUtc.toLocal();
displayDate = DateFormat('yyyy-MM-dd HH:mm:ss').format(dtLocal);
}
} }
class Scratch { class Scratch {
@@ -30,16 +43,33 @@ Future<Note?> getLatestNote() async {
} }
Future<void> createNote(String content) async { Future<void> createNote(String content) async {
if (content.isEmpty) { // Trim each line, sometimes we fuck up by doing a lil "foobar "
return; // Maybe I should also look for \s{2,}...
final lines = content.split('\n');
final trimmedLines = <String>[];
for (final line in lines) {
final trimmedContent = line.trim().replaceAll(RegExp(r'\s{2,}'), ' ');
if (trimmedContent.isEmpty) {
continue;
}
trimmedLines.add(trimmedContent);
} }
await DB.db.insert('notes', {'content': content}); final trimmedContent = trimmedLines.join('\n');
await DB.db.insert('notes', {'content': trimmedContent});
} }
Future<void> updateNote(Note note) async { Future<void> updateNote(Note note) async {
// Trim the content to avoid saving just whitespace
final trimmedContent = note.content.trim();
if (trimmedContent.isEmpty) {
// Delete the note if content is empty
await DB.db.delete('notes', where: 'date = ?', whereArgs: [note.date]);
return;
}
await DB.db.update( await DB.db.update(
'notes', 'notes',
{'content': note.content}, {'content': trimmedContent},
where: 'date = ?', where: 'date = ?',
whereArgs: [note.date], whereArgs: [note.date],
); );
@@ -61,7 +91,9 @@ Future<Scratch?> getLatestScratch() async {
} }
Future<void> createScratch(String content) async { Future<void> createScratch(String content) async {
await DB.db.insert('scratches', {'content': content}); // Trim content but allow empty scratch notes (they might be intentionally cleared)
final trimmedContent = content.trim();
await DB.db.insert('scratches', {'content': trimmedContent});
} }
// Get the note immediately older than the given date // Get the note immediately older than the given date
@@ -103,6 +135,17 @@ Future<Note?> getNextNote(String currentDate) async {
return getLatestNote(); return getLatestNote();
} }
/// Delete a note by its date
Future<bool> deleteNote(String date) async {
final result = await DB.db.delete(
'notes',
where: 'date = ?',
whereArgs: [date],
);
return result > 0; // Return true if a note was deleted
}
// Search notes using full-text search // Search notes using full-text search
Future<List<Note>> searchNotes(String query) async { Future<List<Note>> searchNotes(String query) async {
if (query.isEmpty) { if (query.isEmpty) {
@@ -124,3 +167,42 @@ Future<List<Note>> searchNotes(String query) async {
) )
.toList(); .toList();
} }
// Find potentially problematic entries based on character distribution
Future<List<Note>> findProblematicEntries({
double maxCharPercentage = 0.7,
}) async {
// Simple SQLite query that counts character occurrences using replace
final List<Map<String, dynamic>> results = await DB.db.rawQuery(
'''
WITH char_counts AS (
SELECT
id,
date,
content,
substr(content, 1, 1) as char,
(length(content) - length(replace(content, substr(content, 1, 1), ''))) as char_count,
length(content) as total_length,
cast(length(content) - length(replace(content, substr(content, 1, 1), '')) as float) / length(content) as percentage
FROM notes
)
SELECT *
FROM char_counts
WHERE percentage > ?
ORDER BY date DESC
''',
[maxCharPercentage],
);
return results
.map(
(row) => Note(
date: row['date'] as String,
content: row['content'] as String,
isProblematic: true,
problemReason:
'Character "${row['char']}" makes up ${(row['percentage'] * 100).toStringAsFixed(1)}% of the content',
),
)
.toList();
}

View File

@@ -224,6 +224,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "4.5.4" version: "4.5.4"
intl:
dependency: "direct main"
description:
name: intl
sha256: "3df61194eb431efc39c4ceba583b95633a403f46c9fd341e550ce0bfa50e9aa5"
url: "https://pub.dev"
source: hosted
version: "0.20.2"
json_annotation: json_annotation:
dependency: transitive dependency: transitive
description: description:
@@ -289,7 +297,7 @@ packages:
source: hosted source: hosted
version: "1.16.0" version: "1.16.0"
path: path:
dependency: transitive dependency: "direct main"
description: description:
name: path name: path
sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5"
@@ -376,6 +384,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "6.0.2" version: "6.0.2"
ps_list:
dependency: "direct main"
description:
name: ps_list
sha256: "19d32f6c643313cf4f5101bb144b8978b9ba3dc42c9a01b247e8ed90581bc0ab"
url: "https://pub.dev"
source: hosted
version: "0.0.5"
screen_retriever: screen_retriever:
dependency: transitive dependency: transitive
description: description:

View File

@@ -38,6 +38,9 @@ dependencies:
window_manager: ^0.4.3 window_manager: ^0.4.3
audioplayers: ^6.4.0 audioplayers: ^6.4.0
sqflite_common_ffi: ^2.3.5 sqflite_common_ffi: ^2.3.5
path: ^1.8.0
ps_list: ^0.0.5
intl: ^0.20.2
dev_dependencies: dev_dependencies:
flutter_test: flutter_test:
@@ -66,6 +69,7 @@ flutter:
assets: assets:
- assets/ # Include the main assets directory for the icon - assets/ # Include the main assets directory for the icon
- assets/sounds/ - assets/sounds/
- assets/windows/sqlite3_icu.dll
# - images/a_dot_burr.jpeg # - images/a_dot_burr.jpeg
# - images/a_dot_ham.jpeg # - images/a_dot_ham.jpeg