# Agent Guide for Telegram Desktop This guide defines repository-wide instructions for coding agents working with the Telegram Desktop codebase. ## Build System Structure The build system expects this directory layout: ```text L:\Telegram\ # BuildPath L:\Telegram\tdesktop\ # Repository (you work here) L:\Telegram\Libraries\ # 32-bit dependencies (Linux/macOS) L:\Telegram\win64\Libraries\ # 64-bit dependencies (Windows) L:\Telegram\ThirdParty\ # Build tools (NuGet, Python, etc.) ``` Dependencies are located relative to the repository: `../Libraries`, `../win64/Libraries`, or `../ThirdParty`. ## Build Configuration ### Build Commands **From repository root, run:** ```bash cmake --build out --config Debug --target Telegram ``` That's it. The `out/` directory is already configured. The executable will be at `out/Debug/Telegram.exe`. **Important:** When running cmake from a shell that doesn't support `cd`, use quoted absolute paths: ```bash cmake --build "l:\Telegram\tx64\out" --config Debug --target Telegram ``` **Never build Release** - it's extremely heavy and not needed for testing changes. ## Platform-Specific Requirements ### Windows - Requires Visual Studio 2022 - Must run from appropriate Native Tools Command Prompt: - "x64 Native Tools Command Prompt" for `win64` - "x86 Native Tools Command Prompt" for `win` - "ARM64 Native Tools Command Prompt" for `winarm` - Dependencies: `../win64/Libraries` (64-bit) or `../Libraries` (32-bit) ### macOS - Requires Xcode - Dependencies: `../Libraries/local/Qt-*` - Set `QT` environment variable: `export QT=6.8` ### Linux - Build dependencies in `../Libraries` - Set `QT` environment variable if needed ## Key Files - **`Telegram/build/version`** - Version information - **`out/`** - Build output directory ## Troubleshooting ### "Libraries not found" Ensure the repository is in `L:\Telegram\tdesktop`. The build system requires `../win64/Libraries` to exist. ### Build fails with "wrong command prompt" On Windows, use the correct Visual Studio Native Tools Command Prompt matching your target (x64/x86/ARM64). ### Build fails with PDB or EXE access errors **⚠️ CRITICAL: DO NOT RETRY THE BUILD. STOP AND WAIT FOR USER.** If the build fails with ANY of these errors: - `fatal error C1041: cannot open program database` - `cannot open output file 'Telegram.exe'` - `LNK1104: cannot open file` - Any "access denied" or "file in use" error **STOP IMMEDIATELY.** These errors mean files are locked by a running process (Telegram.exe or debugger). **What to do:** 1. Do NOT attempt another build - it will fail the same way 2. Do NOT try to delete files - they are locked 3. Do NOT try any workarounds or fixes 4. IMMEDIATELY inform the user: > "Build failed - files are locked. Please close Telegram.exe (and any debugger) so I can rebuild." **Then WAIT for user confirmation before attempting any build.** Retrying builds wastes time and context. The ONLY fix is for the user to close the running process. ## Best Practices 1. **Always use Debug builds** - Release builds are extremely heavy 2. **Don't build Release configuration** - it's too heavy for testing ## Text File Format - On Windows, keep project text files with CRLF line endings. - Do not save source, header, build/config, style, or localization files as UTF-8 with BOM. Use UTF-8 without BOM. - When rewriting project text files for normalization, preserve file content otherwise and do not introduce a BOM. ## Local Storage Serialization Both app-level (`Core::Settings`) and session-level (`Main::SessionSettings`) use sequential binary serialization via `QDataStream`. Key rules: - New fields must ALWAYS be appended at the **end** of the stream, never inserted in the middle - Reading new fields must be guarded with `!stream.atEnd()` and provide a meaningful default/fallback - Inserting in the middle breaks reading of data saved by older versions (the new read code consumes bytes that belong to subsequent fields) - For simple flags and values, prefer using the generic KV prefs facility (`writePref` / `readPref`) instead of adding to the binary stream -- this avoids serialization ordering issues entirely --- # Development Guidelines ## Coding Style **Do NOT write comments in code:** This is important! Do not write single-line comments that describe what the next line does - they are bloat. Comments are allowed ONLY to describe complex algorithms in detail, when the explanation requires at least 4-5 lines. Self-documenting code with clear variable and function names is preferred. ```cpp // BAD - don't do this: // Get the user's name auto name = user->name(); // Check if premium if (user->isPremium()) { // GOOD - no comments needed, code is self-explanatory: auto name = user->name(); if (user->isPremium()) { // ACCEPTABLE - complex algorithm explanation (4+ lines): // The algorithm works by first collecting all visible messages // in the viewport, then calculating their intersection with // the clip rectangle. Messages are grouped by date headers, // and we need to account for sticky headers that may overlap // with the first message in each group. ``` **Style and formatting rules** are in `REVIEW.md` — see that file for empty-line-before-closing-brace, operator placement in multi-line expressions, if-with-initializer, and other mechanical style rules. **Use `auto` for type deduction:** Prefer `auto` (or `const auto`, `const auto &`) instead of explicit types: ```cpp // Prefer this: auto currentTitle = tr::lng_settings_title(tr::now); auto nameProducer = GetNameProducer(); // Instead of this: QString currentTitle = tr::lng_settings_title(tr::now); rpl::producer nameProducer = GetNameProducer(); ``` ## API Usage ### API Schema Files API definitions use [TL Language](https://core.telegram.org/mtproto/TL): 1. **`Telegram/SourceFiles/mtproto/scheme/mtproto.tl`** - MTProto protocol (encryption, auth, etc.) 2. **`Telegram/SourceFiles/mtproto/scheme/api.tl`** - Telegram API (messages, users, chats, etc.) ### Making API Requests Standard pattern using `api()`, generated `MTP...` types, and callbacks: ```cpp api().request(MTPnamespace_MethodName( MTP_flags(flags_value), MTP_inputPeer(peer), MTP_string(messageText), MTP_long(randomId), MTP_vector() )).done([=](const MTPResponseType &result) { // Handle successful response // Multiple constructors - use .match() or check type: result.match([&](const MTPDuser &data) { // use data.vfirst_name().v }, [&](const MTPDuserEmpty &data) { // handle empty user }); // Single constructor - use .data() shortcut: const auto &data = result.data(); // use data.vmessages().v }).fail([=](const MTP::Error &error) { // Handle API error if (error.type() == u"FLOOD_WAIT_X"_q) { // Handle flood wait } }).handleFloodErrors().send(); ``` **Key points:** - Always refer to `api.tl` for method signatures and return types - Use generated `MTP...` types for parameters (`MTP_int`, `MTP_string`, etc.) - For multiple constructors, use `.match()` or check `.type()` against `mtpc_` constants then call `.c_constructorName()`: ```cpp // Using match: result.match([&](const MTPDuser &data) { ... }, [&](const MTPDuserEmpty &data) { ... }); // Or explicit type check: if (result.type() == mtpc_user) { const auto &data = result.c_user(); // asserts on type mismatch } ``` - For single constructors, use `.data()` shortcut - Include `.handleFloodErrors()` before `.send()` in rare cases where you want special case flood error handling ## UI Styling ### Style Files UI styles are defined in `.style` files using custom syntax: ```style using "ui/basic.style"; using "ui/widgets/widgets.style"; MyButtonStyle { textPadding: margins; icon: icon; height: pixels; } defaultButton: MyButtonStyle { textPadding: margins(10px, 15px, 10px, 15px); icon: icon{{ "gui/icons/search", iconColor }}; height: 30px; } primaryButton: MyButtonStyle(defaultButton) { icon: icon{{ "gui/icons/check", iconColor }}; } ``` **Built-in types:** - `int` - Integer numbers (e.g., `maxLines: 3;`) - `bool` - Boolean values (e.g., `useShadow: true;`) - `pixels` - Pixel values with `px` suffix (e.g., `10px`) - `color` - Named colors from `ui/colors.palette` - `icon` - Inline icon definition: `icon{{ "path/stem", color }}` - `margins` - Four values: `margins(top, right, bottom, left)` - `size` - Two values: `size(width, height)` - `point` - Two values: `point(x, y)` - `align` - Alignment: `align(center)`, `align(left)` - `font` - Font: `font(14px semibold)` - `double` - Floating point **Multi-part icons** (layers drawn bottom-up): ```style myComplexIcon: icon{ { "gui/icons/background", iconBgColor }, { "gui/icons/foreground", iconFgColor } }; ``` **Borders** are typically separate fields, not a single property: ```style chatInput { border: 1px; // width borderFg: defaultInputFieldBorder; // color } ``` **Never hardcode sizes in code:** The app supports different interface scale options. Style `px` values are automatically scaled at runtime, but raw integer constants in code are not. Never use hardcoded numbers for margins, paddings, spacing, sizes, coordinates, or any other dimensional values. Always define them in `.style` files and reference via `st::`. ```cpp // BAD - breaks at non-100% interface scale: p.drawText(10, 20, text); widget->setFixedHeight(48); auto margin = 8; auto iconSize = QSize(24, 24); // GOOD - define in .style file and reference: p.drawText(st::myWidgetTextLeft, st::myWidgetTextTop, text); widget->setFixedHeight(st::myWidgetHeight); auto margin = st::myWidgetMargin; auto iconSize = st::myWidgetIconSize; ``` **Duration constants**: Animation durations should NOT go in `.style` files, this is a legacy approach. Prefer `constexpr auto kName = crl::time(N)` in an anonymous namespace in the relevant `.cpp` file. ### Usage in Code ```cpp #include "styles/style_widgets.h" // Access style members int height = st::primaryButton.height; const style::icon &icon = st::primaryButton.icon; style::margins padding = st::primaryButton.textPadding; // Use in painting void MyWidget::paintEvent(QPaintEvent *e) { Painter p(this); p.fillRect(rect(), st::chatInput.backgroundColor); } ``` ## Localization ### String Definitions Strings are defined in `Telegram/Resources/langs/lang.strings`: ``` "lng_settings_title" = "Settings"; "lng_confirm_delete_item" = "Are you sure you want to delete {item_name}?"; "lng_files_selected#one" = "{count} file selected"; "lng_files_selected#other" = "{count} files selected"; ``` ### Usage in Code **Immediate (current value):** ```cpp auto currentTitle = tr::lng_settings_title(tr::now); auto currentConfirmation = tr::lng_confirm_delete_item( tr::now, lt_item_name, currentItemName); auto filesText = tr::lng_files_selected(tr::now, lt_count, count); ``` **Reactive (rpl::producer):** ```cpp auto titleProducer = tr::lng_settings_title(); auto confirmationProducer = tr::lng_confirm_delete_item( lt_item_name, std::move(itemNameProducer)); auto filesTextProducer = tr::lng_files_selected( lt_count, countProducer | tr::to_count()); ``` **Key points:** - Pass `tr::now` as first argument for immediate `QString` - Omit `tr::now` for reactive `rpl::producer` - Placeholders use `lt_tag_name, value` pattern - For `{count}`: immediate uses `int`, reactive uses `rpl::producer` with `| tr::to_count()` - Move producers with `std::move` when passing to placeholders - Rich text projectors — these `tr::` helpers serve double duty: as the **last argument** (projector) they set the return type to `TextWithEntities`, and as **placeholder values** they wrap individual substitutions in formatting. Always prefer them over `Ui::Text::Bold()`, `Ui::Text::RichLangValue`, etc. — see REVIEW.md for the full mapping. - `tr::marked` — basic projection, converts `QString` to `TextWithEntities` - `tr::rich` — interprets `**bold**`/`__italic__` markup in the string - `tr::bold`, `tr::italic`, `tr::underline` — wrap text in that formatting - `tr::link` — wrap as a clickable link - `tr::url(u"https://..."_q)` — returns a projection that converts text to a link pointing to the given URL; can be passed to `rpl::map` or directly to a `tr::lng_...` call ```cpp // As last argument (projector): auto title = tr::lng_export_progress_title(tr::now, tr::bold); auto text = tr::lng_proxy_incorrect_secret(tr::now, tr::rich); // As placeholder value wrapper + projector: auto desc = tr::lng_some_key( tr::now, lt_name, tr::bold(userName), lt_group, tr::bold(groupName), tr::rich); // Nested tr::lng as placeholder: auto linked = tr::lng_settings_birthday_contacts( lt_link, tr::lng_settings_birthday_contacts_link(tr::url(link)), tr::marked); ``` ## RPL (Reactive Programming Library) ### Core Concepts **Producers** represent streams of values over time: ```cpp auto intProducer = rpl::single(123); // Emits single value auto lifetime = rpl::lifetime(); // Manages subscription lifetime ``` ### Starting Pipelines ```cpp std::move(counter) | rpl::on_next([=](int value) { qDebug() << "Received: " << value; }, lifetime); // Without lifetime parameter - MUST store returned lifetime: auto subscriptionLifetime = std::move(counter) | rpl::on_next([=](int value) { // process value }); ``` ### Transforming Producers ```cpp auto strings = std::move(ints) | rpl::map([](int value) { return QString::number(value * 2); }); auto evenInts = std::move(ints) | rpl::filter([](int value) { return (value % 2 == 0); }); ``` ### Combining Producers **`rpl::combine`** - combines latest values (lambdas receive unpacked arguments): ```cpp auto combined = rpl::combine(countProducer, textProducer); std::move(combined) | rpl::on_next([=](int count, const QString &text) { qDebug() << "Count=" << count << ", Text=" << text; }, lifetime); ``` **`rpl::merge`** - merges producers of same type: ```cpp auto merged = rpl::merge(sourceA, sourceB); std::move(merged) | rpl::on_next([=](QString &&value) { qDebug() << "Merged value: " << value; }, lifetime); ``` **Other pipeline starters** — besides `rpl::on_next`, there are: - `rpl::on_error([=](Error &&e) { ... }, lifetime)` — handle errors - `rpl::on_done([=] { ... }, lifetime)` — handle stream completion - `rpl::on_next_error_done(nextCb, errorCb, doneCb, lifetime)` — handle all three The `Error` template parameter defaults to `rpl::no_error`: `rpl::producer`. **Key points:** - Explicitly `std::move` producers when starting pipelines - Pass `rpl::lifetime` to `on_...` methods or store returned lifetime - Use `rpl::duplicate(producer)` to reuse a producer multiple times - Combined producers automatically unpack tuples in lambdas (works with `rpl::map`, `rpl::filter`, and `rpl::on_next`)