import QtQuick 2.14 import QtQuick.Controls 2.14 import QtQuick.Layouts 1.14 import Qt.labs.settings 1.0 import StatusQ 0.1 // https://github.com/status-im/status-desktop/issues/10218 import StatusQ.Core.Theme 0.1 import Storybook 1.0 ApplicationWindow { id: root width: 1450 height: 840 visible: true property string currentPage title: "%1 – %2".arg(currentPage).arg(Qt.application.displayName) palette.window: Theme.palette.statusAppLayout.backgroundColor palette.text: Theme.palette.directColor1 palette.windowText: Theme.palette.directColor1 palette.base: Theme.palette.indirectColor1 font.pixelSize: 13 QtObject { id: d function activateInspection(item) { inspectionWindow.inspect(item) inspectionWindow.show() inspectionWindow.requestActivate() } function performInspection() { // Find the items to inspect on the current page const getItems = typeName => InspectionUtils.findItemsByTypeName( viewLoader.item, typeName) const items = [ ...getItems(root.currentPage), ...getItems("Custom" + root.currentPage) ] // Find lowest commont ancestor of found items const lca = InspectionUtils.lowestCommonAncestor( items, viewLoader.item) // Inspect lca if (lca) { activateInspection(lca.parent.contentItem === lca ? lca.parent : lca) return } // Look for the item for inspection on the Overlay, skip items // without contentItem which can be, for example, instance of // Overlay.modal or Overlay.modeless const overlayChildren = root.Overlay.overlay.children for (let i = 0; i < overlayChildren.length; i++) { const item = overlayChildren[i] if (item.contentItem) { activateInspection(item) return } } nothingToInspectDialog.open() } } PagesModel { id: pagesModel } FigmaLinksSource { id: figmaLinksSource filePath: "figma.json" } FigmaDecoratorModel { id: figmaModel sourceModel: pagesModel figmaLinks: figmaLinksSource.figmaLinks } HotReloader { id: reloader loader: viewLoader enabled: hotReloaderControls.enabled onReloaded: hotReloaderControls.notifyReload() } SplitView { anchors.fill: parent ColumnLayout { SplitView.preferredWidth: 270 Pane { Layout.fillWidth: true Layout.fillHeight: true ColumnLayout { width: parent.width height: parent.height Button { Layout.fillWidth: true text: "Settings" onClicked: settingsPopup.open() } CheckBox { id: darkModeCheckBox Layout.fillWidth: true text: "Dark mode" StatusLightTheme { id: lightTheme } StatusDarkTheme { id: darkTheme } Binding { target: Theme property: "palette" value: darkModeCheckBox.checked ? darkTheme : lightTheme } } HotReloaderControls { id: hotReloaderControls Layout.fillWidth: true onForceReloadClicked: reloader.forceReload() } MenuSeparator { Layout.fillWidth: true } FilteredPagesList { Layout.fillWidth: true Layout.fillHeight: true currentPage: root.currentPage model: pagesModel onPageSelected: root.currentPage = page } } } Button { Layout.fillWidth: true text: "Open pages directory" onClicked: Qt.openUrlExternally(Qt.resolvedUrl(pagesFolder)) } } Page { SplitView.fillWidth: true Loader { id: viewLoader anchors.fill: parent clip: true source: `pages/${root.currentPage}Page.qml` asynchronous: settingsLayout.loadAsynchronously visible: status === Loader.Ready // force reload when `asynchronous` changes onAsynchronousChanged: { active = false active = true } } BusyIndicator { anchors.centerIn: parent visible: viewLoader.status === Loader.Loading } Label { anchors.centerIn: parent visible: viewLoader.status === Loader.Error text: "Loading page failed" } footer: PageToolBar { id: pageToolBar componentName: root.currentPage figmaPagesCount: currentPageModelItem.object ? currentPageModelItem.object.figma.count : 0 Instantiator { id: currentPageModelItem model: SingleItemProxyModel { sourceModel: figmaModel roleName: "title" value: root.currentPage } delegate: QtObject { readonly property string title: model.title readonly property var figma: model.figma } } onFigmaPreviewClicked: { if (!settingsLayout.figmaToken) { noFigmaTokenDialog.open() return } figmaWindow.createObject(root, { figmaModel: currentPageModelItem.object.figma, pageTitle: currentPageModelItem.object.title }) } onInspectClicked: d.performInspection() } } } Dialog { id: settingsPopup anchors.centerIn: Overlay.overlay width: 420 modal: true header: Pane { background: null Label { text: "Settings" } } SettingsLayout { id: settingsLayout width: parent.width } } Dialog { id: noFigmaTokenDialog anchors.centerIn: Overlay.overlay title: "Figma token not set" standardButtons: Dialog.Ok Label { text: "Please set Figma personal token in \"Settings\"" } } FigmaLinksCache { id: figmaImageLinksCache figmaToken: settingsLayout.figmaToken } InspectionWindow { id: inspectionWindow } Dialog { id: nothingToInspectDialog anchors.centerIn: Overlay.overlay width: contentItem.implicitWidth + leftPadding + rightPadding title: "No items to inspect found" standardButtons: Dialog.Ok modal: true contentItem: Label { text: ` Tips: • For inline components use naming convention of adding "Custom" at the begining (like Custom${root.currentPage}) • For popups set closePolicy to "Popup.NoAutoClose"" ` } } Component { id: figmaWindow FigmaPreviewWindow { property string pageTitle property alias figmaModel: figmaImagesProxyModel.sourceModel title: pageTitle + " - Figma" model: FigmaImagesProxyModel { id: figmaImagesProxyModel figmaLinksCache: figmaImageLinksCache } onRemoveLinksRequested: figmaLinksSource.remove(pageTitle, indexes) onAppendLinksRequested: figmaLinksSource.append(pageTitle, links) onClosing: Qt.callLater(destroy) } } Settings { property alias currentPage: root.currentPage property alias loadAsynchronously: settingsLayout.loadAsynchronously property alias darkMode: darkModeCheckBox.checked property alias hotReloading: hotReloaderControls.enabled property alias figmaToken: settingsLayout.figmaToken } Shortcut { sequence: "Ctrl+Shift+I" context: Qt.ApplicationShortcut onActivated: d.performInspection() } }