17 KiB
CLAUDE.md
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
Project Overview
Notepad3 is a Windows-only Win32 desktop text editor (C/C++) built on Scintilla (editing component) and Lexilla (syntax highlighting). It ships with the companion tool MiniPath (file browser, Ctrl+M) and integrates with the external grepWin tool (file search/grep, Ctrl+Shift+F) via pre-built portable executables. Licensed under BSD 3-Clause.
Build Commands
# NuGet restore (required before first build)
nuget restore Notepad3.sln
# Single platform builds
Build\Build_x64.cmd [Release|Debug]
Build\Build_Win32.cmd [Release|Debug]
Build\Build_ARM64.cmd [Release|Debug]
Build\Build_x64_AVX2.cmd [Release|Debug]
# All platforms at once
Build\BuildAll.cmd [Release|Debug]
# MSBuild directly (used by CI)
msbuild Notepad3.sln /m /p:Configuration=Release /p:Platform=x64
# Clean all outputs
Build\Clean.cmd
Default configuration is Release. Build scripts delegate to PowerShell in Build\scripts\.
Versioning
Run Version.ps1 before building to generate src\VersionEx.h from templates in Versions\. Format: Major.YY.Mdd.Build (build number persisted in Versions\build.txt).
Tests
cd test
TestFileVersion.cmd # Verifies built binary version info
TestAhkNotepad3.cmd # AutoHotkey-based GUI tests (requires AutoHotkey)
CI
GitHub Actions (.github/workflows/build.yml) builds all four platforms (Win32, x64, x64_AVX2, ARM64) in Release on windows-2022 runners, triggered on push/PR to master.
Architecture
Core Modules (src\)
| File | Purpose |
|---|---|
| Notepad3.c/h | Entry point (wWinMain), window procedure (MainWndProc), global state structs (Globals, Settings, Settings2, Flags, Paths) |
| Edit.c/h | Text manipulation: find/replace (PCRE2 regex), encoding conversion, clipboard, indentation, sorting, bookmarks, folding, auto-complete |
| Styles.c/h | Scintilla styling, lexer selection, theme management, margin configuration |
| Dialogs.c/h | All dialog boxes, DPI-aware UI interactions, window placement |
| Config/Config.cpp/h | INI file management, settings loading/saving, MRU list |
| Encoding.c/h | Encoding detection and conversion (integrates uchardet) |
| SciCall.h | Type-safe inline wrappers for Scintilla direct function calls (avoids SendMessage overhead) |
| DynStrg.c/h | Custom dynamic wide-string type (HSTRINGW) with automatic buffer management |
| PathLib.c/h | Path manipulation via opaque HPATHL handle |
| TypeDefs.h | Core type definitions (DocPos, DocLn, cpi_enc_t), Windows version targeting, compiler macros |
| MuiLanguage.c/h | Multi-language UI support, language DLL loading |
Window Hierarchy
MainWndProc (Notepad3.c)
+-- Scintilla Edit Control (hwndEdit / IDC_EDIT)
+-- Toolbar (via Rebar control)
+-- Status Bar (16 configurable fields)
Vendored Dependencies
| Directory | Library | Purpose |
|---|---|---|
scintilla\ |
Scintilla 5.5.8 | Editor component (NP3 patches in np3_patches\, docs in doc\) |
lexilla\ |
Lexilla 5.4.6 | Syntax highlighting (NP3 patches in np3_patches\, docs in doc\) |
scintilla\pcre2\ |
PCRE2 10.47 | Regex engine for find/replace (replaced archived Oniguruma) |
src\uchardet\ |
uchardet | Mozilla encoding detection |
src\tinyexpr\ / src\tinyexprcpp\ |
TinyExpr | Expression evaluator (statusbar) |
src\uthash\ |
uthash | Hash table / dynamic array macros |
src\crypto\ |
Rijndael/SHA-256 | AES-256 encryption |
grepWin Integration (grepWin\)
grepWin is an external file search/grep tool — it is not built from source as part of Notepad3. Pre-built portable executables and translation files are stored in the repository:
grepWin\portables\—grepWin-x86_portable.exe,grepWin-x64_portable.exe,LICENSE.txtgrepWin\translations\—*.langtranslation files (e.g.German.lang,French.lang)
At runtime (src\Dialogs.c), Notepad3 searches for grepWin in this order:
Settings2.GrepWinPath(user-configured INI setting)<ModuleDirectory>\grepWin\grepWin-x64_portable.exe(or x86) — portable layout%APPDATA%\Rizonesoft\Notepad3\grepWin\— installed layout
Language mapping (src\MuiLanguage.c): grepWinLangResName[] maps Notepad3 locale names (e.g. de-DE) to grepWin .lang filenames (e.g. German.lang). The language file path is written to grepwin.ini before launching.
Portable build scripts (Build\make_portable_*.cmd) package grepWin into a grepWin\ subdirectory in the archive containing both portable executables, the license, and all *.lang translations.
Syntax Lexers (src\StyleLexers\)
50+ languages, each in a styleLexXXX.c file. All follow the EDITLEXER struct pattern from EditLexer.h:
EDITLEXER lexXXX = {
SCLEX_XXX, // Scintilla lexer ID
"lexerName", // Lexilla lexer name (case-sensitive)
IDS_LEX_XXX_STR, // Resource string ID
L"Config Name", // INI section name
L"ext1; ext2", // Default file extensions
L"", // Extension buffer (runtime)
&KeyWords_XXX, // Keyword lists
{ /* EDITSTYLE array */ }
};
To add a new lexer: create styleLexNEW.c, define the EDITLEXER struct, register in Styles.c lexer array, add localization string IDs to resource files.
Localization (language\)
Resource-based MUI system with 27+ locales. Each locale has a np3_LANG_COUNTRY\ directory with .rc files. Language packs are built as separate DLLs (separate projects in the solution).
Adding String Resources
- Add
#define IDS_MUI_XXX <id>tolanguage\common_res.h(use next available ID in the appropriate range: 13xxx for errors/warnings, 14xxx for info/prompts) - Add the English string to
language\np3_en_us\strings_en_us.rcin the matchingSTRINGTABLEblock - Add the same English text as placeholder to all other 25 locale
strings_*.rcfiles (translators update later) - Use
InfoBoxLng()/MessageBoxLng()withMB_YESNO,MB_ICONWARNING, etc. to display; check result withIsYesOkay() Settings.MuteMessageBeepcontrols whether to useInfoBoxLng(silent) orMessageBoxLng(with sound) — always provide both paths
Configuration / Portable Design
- INI file alongside executable (
Notepad3.ini), no registry usage - No auto-creation: runs with defaults if no INI exists; user explicitly creates via "Save Settings Now"
- Admin redirect:
Notepad3.ini=<path>in[Notepad3]section redirects to per-user path (up to 2 levels) - Key paths:
Paths.IniFile(active writable INI),Paths.IniFileDefault(fallback for recovery) - INI init flow:
FindIniFile()->TestIniFile()->CreateIniFile()->LoadSettings() - MiniPath follows the same portable INI and admin-redirect pattern (
minipath\src\Config.cpp). Redirect targets are auto-created viaCreateIniFileEx(). - New parameters: When adding new
Settings2(or other INI) parameters, always document them as commented entries inBuild\Notepad3.ini
Creating Directories
Use Path_CreateDirectoryEx(hpth) (PathLib wrapper around SHCreateDirectoryExW) to recursively create directory trees. Check result: SUCCEEDED(hr) || (hr == HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS)). See CreateIniFile() in src\Config\Config.cpp for the reference pattern.
Long-Path Support / Win32 File API Wrappers
Notepad3 supports paths exceeding MAX_PATH (260 chars) on systems without the Windows 10 long-path registry opt-in by conditionally prepending the \\?\ prefix. The static function PrependLongPathPrefix() in PathLib.c handles this.
Rule: Never call Win32 file APIs directly with Path_Get(hpth). Use the PathLib wrapper functions instead — they internally apply the copy-prefix-call-release pattern:
| Wrapper | Win32 API |
|---|---|
Path_CreateFile(hpth, ...) |
CreateFileW |
Path_DeleteFile(hpth) |
DeleteFileW |
Path_GetFileAttributes(hpth) |
GetFileAttributesW |
Path_GetFileAttributesEx(hpth, ...) |
GetFileAttributesExW |
Path_SetFileAttributes(hpth, ...) |
SetFileAttributesW |
Path_ReplaceFile(hpth_dest, src) |
ReplaceFileW |
Path_MoveFileEx(src, hpth_dest, ...) |
MoveFileExW |
Path_FindFirstFile(hpth, ...) |
FindFirstFileW |
Path_CreateDirectoryEx(hpth) |
SHCreateDirectoryExW |
Path_IsExistingFile(hpth) |
GetFileAttributesW + check |
Path_IsExistingDirectory(hpth) |
GetFileAttributesW + check |
The prefix is only added when RtlAreLongPathsEnabled() returns false AND the path is ≥ 260 chars. On modern Windows 10+ with the opt-in, these wrappers are effectively pass-through.
Path Comparison
Use Path_StrgComparePath() for comparing file paths — it supports optional normalization and uses CompareStringOrdinal (locale-independent, case-insensitive). For raw wide-string path comparison, use CompareStringOrdinal(s1, -1, s2, -1, TRUE) instead of _wcsicmp or _wcsnicmp which are locale-dependent.
File I/O Flow
FileSave()(src\Notepad3.c) — Main save dispatcher. Handles Save, Save As, Save Copy. CallsFileIO()→EditSaveFile()(src\Edit.c)FileLoad()(src\Notepad3.c) — Main load dispatcher. CallsFileIO()→EditLoadFile()(src\Edit.c)EditSaveFile()supports atomic save (temp file +ReplaceFileW) controlled bySettings2.AtomicFileSave- Error handling:
Globals.dwLastErrorholds the Win32 error code after failed I/O.FileSave()checks specific codes (ERROR_ACCESS_DENIED,ERROR_PATH_NOT_FOUND) before falling back to generic error. - File watching:
InstallFileWatching()usesFindFirstChangeNotificationWon the parent directory. Must be stopped before save (InstallFileWatching(false)) and restarted after (InstallFileWatching(true)).
PCRE2 Regex Engine (scintilla\pcre2\)
PCRE2 10.47 replaced the archived Oniguruma library. The Scintilla integration lives in scintilla\pcre2\scintilla\PCRE2RegExEngine.cxx, compiled with SCI_OWNREGEX to override Scintilla's built-in regex.
Key components:
PCRE2RegExEngine::FindText— Scintilla regex search (pattern matching viapcre2_match)PCRE2RegExEngine::SubstituteByPosition— Regex replacement with group referencesPCRE2RegExEngine::convertReplExpr— Normalizes replacement strings: converts\1-\9to$1-$9, processes escape sequences (\n,\t,\xHH,\uHHHH)PCRE2RegExEngine::translateRegExpr— Translates Scintilla regex extensions:\</\>word boundaries → lookarounds,\uHHHH→\x{HHHH}RegExFind(exported C function) — Standalone regex find used byEditURLDecodeinEdit.c; wrapsSimplePCRE2Engine
Replacement string backreference syntax (both flavors supported for backward compatibility):
$0-$99and\0-\9— numbered group references${name}/${+name}— named group references- Escape sequences:
\n,\t,\r,\\,\xHH,\uHHHH
URL hotspot regex is defined at src\Edit.c:108 (HYPLNK_REGEX_FULL macro). It matches https?://, ftp://, file:///, file://, mailto:, www., ftp. schemes. The trailing group excludes punctuation (.,:?!) so URLs don't absorb sentence-ending characters.
DarkMode (src\DarkMode\)
Windows 10/11 dark mode via IAT (Import Address Table) hooks. Includes stub DLLs for uxtheme and user32.
ARM64 Platform Considerations
Supported platforms: Win32 (x86), x64, x64_AVX2, ARM64. ARM 32-bit is not supported (the Release|ARM solution config maps to Win32).
Architecture detection
Use #if defined(_M_ARM64) or the helper macro NP3_BUILD_ARM64 (defined in src\TypeDefs.h) for ARM64-specific code paths. Important: both ARM64 and x64 define _WIN64, so use _M_ARM64 when you need to distinguish ARM64 from x64.
ARM64 rendering defaults
ARM64 defaults to SC_TECHNOLOGY_DIRECTWRITERETAIN (value 2) instead of SC_TECHNOLOGY_DIRECTWRITE (value 1) to preserve the Direct2D back buffer between frames. This avoids flickering on Qualcomm Adreno GPUs and the Win11 25H2 DWM compositor. The main window also uses WS_EX_COMPOSITED on ARM64 for system-level double-buffering. Users can override via RenderingTechnology in the INI file or the View menu.
ARM64 build configuration
CETCompatmust befalsefor ARM64 (CET is x86/x64 only)TargetMachinemust beMachineARM64in all ARM64 linker sections_WIN64must be defined in preprocessor definitions for all ARM64 configurations- Build fix scripts in
Build\scripts\:FixARM64CETCompat.ps1,FixARM64CrossCompile.ps1,FixARM64OutDir.ps1
GrepWin on ARM64
No native ARM64 grepWin build exists. The ARM64 build uses grepWin-x64_portable.exe which runs via x64 emulation on Windows ARM64. The binary selection in src\Notepad3.c uses #if defined(_M_ARM64) to handle this explicitly.
Theme change flickering prevention
MsgThemeChanged() in src\Notepad3.c wraps all heavy operations (bar recreation, lexer reset, restyling) in WM_SETREDRAW FALSE/TRUE to suppress intermediate repaints and performs a single RedrawWindow() at the end. DarkMode RedrawWindow() calls in ListViewUtil.hpp omit RDW_ERASE to avoid background erase flashes during theme transitions.
Formatting
- LLVM-based
.clang-formatinsrc\— 4-space indentation, Stroustrup brace style, left-aligned pointers, no column limit, no include sorting .editorconfigenforces UTF-8/CRLF for source, 4-space indent for C/C++; Lexilla code uses tabs (preserved from upstream)- String safety via
strsafe.hthroughout; deprecated string functions are disabled
Type Conventions
DocPos/DocPosU/DocLnfor Scintilla document positions and line numbers (not rawint)cpi_enc_tfor encoding identifiersHSTRINGWandHPATHL(opaque handle types) instead of rawWCHAR*buffersNOMINMAXis defined globally — usemin()/max()macros or typed equivalents
Scintilla Interaction
Always use SciCall.h wrappers (e.g. SciCall_GetTextLength()) instead of raw SendMessage(hwnd, SCI_XXX, ...). Add missing wrappers to SciCall.h if needed.
Wrapper macros follow the naming DeclareSciCall{V|R}{0|01|1|2}:
- V = void return, R = has return value
- 0 = no params, 1 = one param (wParam), 2 = two params, 01 = lParam only (wParam=0)
- The
msgargument is the suffix afterSCI_(e.g.UNDOforSCI_UNDO)
DeclareSciCallV0(Undo, UNDO); // SciCall_Undo()
DeclareSciCallR0(GetTextLength, GETTEXTLENGTH, DocPos); // DocPos SciCall_GetTextLength()
DeclareSciCallV1(SetTechnology, SETTECHNOLOGY, int, technology); // SciCall_SetTechnology(int)
DeclareSciCallR1(SupportsFeature, SUPPORTSFEATURE, bool, int, feature); // bool SciCall_SupportsFeature(int)
Global State
Application state is centralized in global structs (Globals, Settings, Settings2, Flags, Paths) defined in Notepad3.c. Access these through their defined interfaces rather than adding new globals.
Undo/Redo Transactions
Use _BEGIN_UNDO_ACTION_ / _END_UNDO_ACTION_ macros (defined in Notepad3.h) to group Scintilla operations into single undo steps. These also handle notification limiting during bulk edits.
WriteAccessBuf — Dangling Pointer Anti-Pattern
NEVER use a pointer obtained from Path_WriteAccessBuf() / StrgWriteAccessBuf() after ANY operation that may reallocate or swap the underlying buffer of the SAME handle. The pointer becomes dangling (use-after-free).
Buffer-invalidating operations:
Path_CanonicalizeEx(h, ...)— callsPath_SwapinternallyPath_Swap(h, ...)/StrgSwap(h, ...)Path_ExpandEnvStrings(h)/ExpandEnvironmentStrgs(h, ...)— may reallocPath_Append(h, ...)/Path_Reset(h, ...)— may reallocStrgCat(h, ...)/StrgInsert(h, ...)/StrgFormat(h, ...)/StrgReset(h, ...)— may reallocPath_NormalizeEx(h, ...)/Path_AbsoluteFromApp(h, ...)/Path_RelativeToApp(h, ...)— may realloc/swap
Safe patterns after invalidation:
- Read-only: use
Path_Get(h)orStrgGet(h)— always returns current buffer - Read-write: re-obtain via
ptr = Path_WriteAccessBuf(h, 0)/ptr = StrgWriteAccessBuf(h, 0)(size 0 = no resize, just returns current pointer)
Python Environment
A Python 3.14 virtual environment is available at .venv\ for scripting tasks (batch file manipulation, locale file updates, code generation, etc.).
# Run a script (from project root, in bash/Cygwin)
.venv/Scripts/python.exe <script.py>
# Install packages
.venv/Scripts/pip.exe install <package>
Use this venv instead of system Python (which may not be installed). Useful for bulk operations across the 26 locale strings_*.rc files.
Key File Paths (Quick Reference)
| Pattern | Purpose |
|---|---|
language/common_res.h |
String resource ID definitions |
language/np3_*/strings_*.rc |
Locale string tables (26 files) |
Build/Notepad3.ini |
Reference INI with documented settings |
src/Notepad3.c:FileSave() |
Save flow entry point |
src/Edit.c:EditSaveFile() |
Actual file write logic |
src/Config/Config.cpp |
INI management, settings load/save |
src/Dialogs.c:SaveFileDlg() |
Save As dialog |