AUI Framework  develop
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
examples/app/notes/src/main.cpp
Note
This Source File belongs to Notes App Example. Please follow the link for example explanation.
/*
* AUI Framework - Declarative UI toolkit for modern C++20
* Copyright (C) 2020-2025 Alex2772 and Contributors
*
* SPDX-License-Identifier: MPL-2.0
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include <range/v3/all.hpp>
#include <AUI/Platform/AWindow.h>
#include <AUI/Platform/Entry.h>
#include <AUI/Util/UIBuildingHelpers.h>
#include <AUI/View/AScrollArea.h>
#include <AUI/View/ASplitter.h>
#include <AUI/View/ATextArea.h>
#include <AUI/Model/AListModel.h>
#include "AUI/View/AButton.h"
#include "AUI/View/ADrawableView.h"
#include "AUI/Json/Conversion.h"
#include "AUI/IO/AFileInputStream.h"
#include "AUI/Reflect/for_each_field.h"
#include "AUI/Platform/AMessageBox.h"
#include <AUI/View/AForEachUI.h>
static constexpr auto LOG_TAG = "Notes";
using namespace declarative;
using namespace ass;
struct Note {
    AProperty<AString> title;
    AProperty<AString> content;
};
AJSON_FIELDS(Note, AJSON_FIELDS_ENTRY(title) AJSON_FIELDS_ENTRY(content))
class TitleTextArea : public ATextArea {
public:
    using ATextArea::ATextArea;
    void onCharEntered(char16_t c) override {
        if (c == '\r') {
            AWindow::current()->focusNextView();
            return;
        }
        ATextArea::onCharEntered(c);
    }
};
_<AView> notePreview(const _<Note>& note) {
    struct StringOneLinePreview {
        AString operator()(const AString& s) const {
            if (s.empty()) {
                return "Empty";
            }
            return s.restrictLength(100, "").replacedAll('\n', ' ');
        }
    };
    return Vertical {
        Label {} with_style { FontSize { 10_pt }, ATextOverflow::ELLIPSIS } &
            note->title.readProjected(StringOneLinePreview {}),
        Label {} with_style {
                  ATextOverflow::ELLIPSIS,
                  Opacity { 0.7f },
                } &
            note->content.readProjected(StringOneLinePreview {}),
    } with_style {
        Padding { 4_dp, 8_dp },
        BorderRadius { 8_dp },
        Margin { 4_dp, 8_dp },
    };
}
_<AView> noteEditor(const _<Note>& note) {
    if (note == nullptr) {
        return Centered { Label { "No note selected" } };
    }
    return AScrollArea::Builder().withContents(
        Vertical {
          _new<TitleTextArea>("Untitled") let {
                  it->setCustomStyle({ FontSize { 14_pt }, Expanding { 1, 0 } });
                  AObject::biConnect(note->title, it->text());
                  if (note->content->empty()) {
                      it->focus();
                  }
              },
          _new<ATextArea>("Text") with_style { Expanding() } && note->content,
        } with_style {
          Padding { 8_dp, 16_dp },
        });
}
class MainWindow : public AWindow {
public:
    MainWindow() : AWindow("Notes") {
        setExtraStylesheet(AStylesheet {
          {
            c(".plain_bg"),
            BackgroundSolid { AColor::WHITE },
          },
          {
            t<AWindow>(),
            Padding { 0 },
          },
        });
        load();
        connect(mNotes.changed, me::markDirty);
        setContents(Vertical {
          ASplitter::Horizontal()
                  .withItems({
                    Vertical {
                      Centered {
                        Horizontal {
                          Button { Icon { ":img/save.svg" }, Label { "Save" } }.connect(&AView::clicked, me::save) &
                              mDirty > &AView::setEnabled,
                          Button { Icon { ":img/new.svg" }, Label { "New Note" } }.connect(
                              &AView::clicked, me::newNote),
                        },
                      },
                      AScrollArea::Builder()
                          .withContents(
                          AUI_DECLARATIVE_FOR(note, *mNotes, AVerticalLayout) {
                              observeChangesForDirty(note);
                              return notePreview(note) let {
                                  connect(it->clicked, [this, note] { mCurrentNote = note; });
                                  it& mCurrentNote > [note](AView& view, const _<Note>& currentNote) {
                                      ALOG_DEBUG(LOG_TAG) << "currentNote == note " << currentNote << " == " << note;
                                      view.setAssName(".plain_bg", currentNote == note);
                                  };
                              };
                          })
                          .build(),
                    } with_style { MinSize { 200_dp } },
                    Vertical::Expanding {
                      Centered {
                        Button { Icon { ":img/trash.svg" }, Label { "Delete" } }.connect(
                            &AView::clicked, me::deleteCurrentNote) &
                            mCurrentNote.readProjected([](const _<Note>& n) {
                                return n != nullptr;
                            }) > &AView::setEnabled,
                      },
                      CustomLayout::Expanding {} & mCurrentNote.readProjected(noteEditor),
                    }<< ".plain_bg" with_style { MinSize { 200_dp } },
                  })
                  .build() with_style { Expanding() },
        });
        if (mNotes->empty()) {
            newNote();
        }
    }
    ~MainWindow() {
        if (mDirty) {
            save();
        }
    }
    void load() {
        try {
            if (!"notes.json"_path.isRegularFileExists()) {
                return;
            }
            aui::from_json(AJson::fromStream(AFileInputStream("notes.json")), mNotes);
        } catch (const AException& e) {
            ALogger::info(LOG_TAG) << "Can't load notes: " << e;
        }
    }
    void save() {
        AFileOutputStream("notes.json") << aui::to_json(*mNotes);
        mDirty = false;
    }
    void newNote() {
        auto note = aui::ptr::manage(new Note { .title = "Untitled" });
        mNotes.writeScope()->push_back(note);
        mCurrentNote = std::move(note);
    }
    void deleteCurrentNote() {
        if (mCurrentNote == nullptr) {
            return;
        }
        if (AMessageBox::show(
                this, "Do you really want to delete this note?",
                "{}\n\nThis operation is irreversible!"_format((*mCurrentNote)->title), AMessageBox::Icon::NONE,
                AMessageBox::Button::OK_CANCEL) != AMessageBox::ResultButton::OK) {
            return;
        }
        auto it = ranges::find(*mNotes, *mCurrentNote);
        it = mNotes.writeScope()->erase(it);
        mCurrentNote = it != mNotes->end() ? *it : nullptr;
    }
    void markDirty() {
        mDirty = true;
    }
    void observeChangesForDirty(const _<Note>& note) {
                *note,
                aui::lambda_overloaded {
                        [&](auto& field) {},
                        [&](APropertyReadable auto& field) {
                            ALOG_DEBUG(LOG_TAG) << "Observing for changes " << &field;
                            AObject::connect(field.changed, me::markDirty);
                        },
                });
    }
private:
    AProperty<AVector<_<Note>>> mNotes;
    AProperty<_<Note>> mCurrentNote;
    AProperty<bool> mDirty = false;
};
    auto w = _new<MainWindow>();
    w->show();
    return 0;
}
Definition AScrollArea.h:130
Multiline text input area.
Definition ATextArea.h:48
void setContents(const _< AViewContainer > &container)
Moves (like via std::move) all children and layout of the specified container to this container.
void setExtraStylesheet(_< AStylesheet > extraStylesheet)
Definition AView.h:238
emits clicked
Left mouse button clicked.
Definition AView.h:933
Represents a window in the underlying windowing system.
Definition AWindow.h:45
static AWindowBase * current()
An std::weak_ptr with AUI extensions.
Definition SharedPtrTypes.h:179
class_of c
Selects views that are of the specified classes.
Definition class_of.h:84
static void biConnect(PropertySource &&propertySource, PropertyDestination &&propertyDestination)
Connects source property to the destination property and opposite (bidirectionally).
Definition AObject.h:156
constexpr void for_each_field_value(Clazz &&clazz, F &&callback)
Calls callback for each field value of a clazz.
Definition for_each_field.h:38
static decltype(auto) connect(const Signal &signal, Object *object, Function &&function)
Connects signal to the slot of the specified object.
Definition AObject.h:86
#define let
Performs multiple operations on a single object without repeating its name (in place) This function c...
Definition kAUI.h:262
#define AUI_DECLARATIVE_FOR(value, model, layout)
ranged-for-loop style wrapped for AForEachUI.
Definition AForEachUI.h:402
#define with_style
Allows to define a style to the view right in place.
Definition kAUI.h:287
#define AUI_ENTRY
Application entry point.
Definition Entry.h:90
API_AUI_VIEWS ResultButton show(AWindow *parent, const AString &title, const AString &message, Icon icon=Icon::NONE, Button b=Button::OK)
Displays a message box, blocking the caller thread until the user dismisses the message.
@ OK
Indicates the user clicked the OK button.
Definition AMessageBox.h:113
Button
Specifies button(s) to be displayed.
Definition AMessageBox.h:79
@ OK_CANCEL
Display OK and Cancel buttons.
Definition AMessageBox.h:88
Icon
Specifies icon to be displayed.
Definition AMessageBox.h:51
@ NONE
No icon is specified.
Definition AMessageBox.h:58
Basic easy-to-use property implementation containing T.
Definition AProperty.h:30
Controls border radius.
Definition BorderRadius.h:28
Controls the expanding of AView.
Definition Expanding.h:26
Controls the font size of AView.
Definition FontSize.h:27
Controls view's margins.
Definition Margin.h:29
Controls the opacity of AView.
Definition Opacity.h:28
Controls the padding of AView.
Definition Padding.h:29
static _< T > manage(T *raw)
Delegates memory management of the raw pointer T* raw to the shared pointer, which is returned.
Definition SharedPtrTypes.h:424