Files
php-qml/examples/todo/qml/Main.qml
magdev 15f9aa032e Phase 3 sub-commit 3: examples/todo POC app, built via the makers
Standalone Composer/CMake project under examples/todo/ derived from the
skeleton, demonstrating every Phase 3 architectural primitive in a
non-trivial app. All cross-side wiring is maker-generated; no
handwritten bridge glue.

Generated and customised:

  - src/Entity/Todo.php — make:bridge:resource Todo (UUIDv7 id)
  - src/Controller/TodoController.php — make:bridge:resource Todo (CRUD)
  - src/Controller/MarkAllDoneController.php — make:bridge:command
    MarkAllDone, body filled in to flip done=true on every row
  - qml/TodoList.qml — make:bridge:resource Todo (starter ListView)
  - qml/TodoWindow.qml — make:bridge:window Todo, body customised to
    embed a read-only mirror of the same ReactiveListModel

The Phase 1 ping demo is dropped from this app — it doesn't fit the
todo flow and nothing in Main.qml references it.

Main.qml is the real list UI:

  - Add input + button (POST /api/todos with optimistic-friendly key).
  - Per-row CheckBox + delete button (PATCH/DELETE via
    todoModel.invoke() with `pending` role driving opacity).
  - "Mark all done" button (POST /api/mark-all-done).
  - "Open second window" button (Component { TodoWindow {} } pattern).

Build / run delegated to the same Makefile shape as the skeleton, with
SCRIPT_DIR/QT_BIN updated for the renamed binary (build/qml/todo).
composer.json's path repo points at ../../../framework/php (one level
deeper than the skeleton's path repo).

Verified end-to-end with offscreen QPA: POST/PATCH/DELETE on /api/todos
all round-trip, /api/mark-all-done flips every row, Mercure dual-
publishes on every change. Clean shutdown.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 15:22:36 +02:00

213 lines
7.4 KiB
QML
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import QtQuick
import QtQuick.Controls
import QtQuick.Layouts
import QtQuick.Window
import PhpQml.Bridge
import Todo // local module — picks up TodoList.qml + TodoWindow.qml
ApplicationWindow {
id: window
width: 720
height: 560
visible: true
title: "php-qml — Todo example"
// Single shared model means a second window sees the same data.
// (Each ReactiveListModel instance has its own MercureClient
// subscription, but the underlying server state is the same.)
property alias rest: rest
property alias todoModel: todoModel
RestClient {
id: rest
baseUrl: BackendConnection.url
token: BackendConnection.token
}
ReactiveListModel {
id: todoModel
baseUrl: BackendConnection.url
token: BackendConnection.token
source: "/api/todos"
topic: "app://model/todo"
onCommandFailed: function(key, status, problem) {
log.append("× failed " + status + ": " + JSON.stringify(problem))
}
onCommandTimedOut: function(key) {
log.append("× timed out " + key)
}
}
Component {
id: secondWindowCmp
TodoWindow {}
}
AppShell {
anchors.fill: parent
ColumnLayout {
anchors.fill: parent
anchors.margins: 16
spacing: 12
// ── Header / actions ────────────────────────────────────
RowLayout {
Layout.fillWidth: true
spacing: 8
Label {
text: "Todos"
font.pixelSize: 18
font.bold: true
}
Item { Layout.fillWidth: true }
Button {
text: "Mark all done"
onClicked: {
rest.post("/api/mark-all-done").then(function(r) {
log.append("→ mark-all-done")
}).catch(function(e) {
log.append("× mark-all-done " + e.status)
})
}
}
Button {
text: "Open second window"
onClicked: {
const w = secondWindowCmp.createObject()
if (w) w.show()
}
}
}
// ── Add input ───────────────────────────────────────────
RowLayout {
Layout.fillWidth: true
spacing: 8
TextField {
id: newTodoField
Layout.fillWidth: true
placeholderText: "What needs doing?"
onAccepted: addBtn.clicked()
}
Button {
id: addBtn
text: "Add"
enabled: newTodoField.text.trim().length > 0
&& BackendConnection.connectionState === BackendConnection.Online
onClicked: {
const title = newTodoField.text.trim()
if (!title) return
rest.post("/api/todos", {title: title, done: false})
.then(function(r) { log.append("→ added '" + title + "'") })
.catch(function(e) { log.append("× add failed " + e.status) })
newTodoField.clear()
}
}
}
// ── Todo list ───────────────────────────────────────────
Frame {
Layout.fillWidth: true
Layout.fillHeight: true
padding: 0
ListView {
id: todoView
anchors.fill: parent
clip: true
model: todoModel
spacing: 1
delegate: ItemDelegate {
required property string id
required property string title
required property bool done
required property bool pending
width: ListView.view.width
opacity: pending ? 0.5 : 1.0
contentItem: RowLayout {
spacing: 8
CheckBox {
checked: done
enabled: !pending
onToggled: {
todoModel.invoke(
"PATCH",
"/api/todos/" + id,
{ done: checked },
{ op: "upsert", id: id, data: { id: id, title: title, done: checked } }
)
}
}
Label {
Layout.fillWidth: true
text: title
font.strikeout: done
elide: Text.ElideRight
}
Button {
text: "×"
flat: true
enabled: !pending
onClicked: {
todoModel.invoke(
"DELETE",
"/api/todos/" + id,
null,
{ op: "delete", id: id }
)
}
}
}
}
Label {
anchors.centerIn: parent
visible: todoView.count === 0 && todoModel.ready
text: "No todos yet — add one above."
opacity: 0.6
}
}
}
// ── Status / log ────────────────────────────────────────
Label {
text: BackendConnection.url
color: "#888"
font.pixelSize: 11
elide: Text.ElideRight
Layout.fillWidth: true
}
Frame {
Layout.fillWidth: true
Layout.preferredHeight: 100
padding: 0
ScrollView {
anchors.fill: parent
TextArea {
id: log
readOnly: true
wrapMode: TextArea.Wrap
font.family: "monospace"
font.pixelSize: 11
}
}
}
}
}
Connections {
target: SingleInstance
function onLaunchArgsReceived(args) {
window.requestActivate()
}
}
}