Browse Source

continue project model implementation

cl-refactor
arkpar 10 years ago
parent
commit
a4a2e8c252
  1. 2
      mix/MixApplication.cpp
  2. 2
      mix/qml.qrc
  3. 13
      mix/qml/CodeEditorModel.qml
  4. 17
      mix/qml/CodeEditorView.qml
  5. 47
      mix/qml/ProjectList.qml
  6. 130
      mix/qml/ProjectModel.qml
  7. 142
      mix/qml/js/ProjectModel.js
  8. 29
      mix/qml/main.qml

2
mix/MixApplication.cpp

@ -33,7 +33,7 @@ MixApplication::MixApplication(int _argc, char* _argv[]):
QApplication(_argc, _argv), m_engine(new QQmlApplicationEngine()), m_appContext(new AppContext(m_engine.get())) QApplication(_argc, _argv), m_engine(new QQmlApplicationEngine()), m_appContext(new AppContext(m_engine.get()))
{ {
qmlRegisterType<CodeEditorExtensionManager>("CodeEditorExtensionManager", 1, 0, "CodeEditorExtensionManager"); qmlRegisterType<CodeEditorExtensionManager>("CodeEditorExtensionManager", 1, 0, "CodeEditorExtensionManager");
QObject::connect(this, SIGNAL(lastWindowClosed()), context(), SLOT(quitApplication())); //use to kill ApplicationContext and other stuff //QObject::connect(this, SIGNAL(lastWindowClosed()), context(), SLOT(quitApplication())); //use to kill ApplicationContext and other stuff
m_engine->load(QUrl("qrc:/qml/main.qml")); m_engine->load(QUrl("qrc:/qml/main.qml"));
//m_engine->load(QUrl("qrc:/qml/ProjectModel.qml")); //m_engine->load(QUrl("qrc:/qml/ProjectModel.qml"));
m_appContext->loadProject(); m_appContext->loadProject();

2
mix/qml.qrc

@ -15,8 +15,8 @@
<file>qml/js/Debugger.js</file> <file>qml/js/Debugger.js</file>
<file>qml/NewProjectDialog.qml</file> <file>qml/NewProjectDialog.qml</file>
<file>qml/ProjectModel.qml</file> <file>qml/ProjectModel.qml</file>
<file>qml/CodeEditorModel.qml</file>
<file>qml/CodeEditor.qml</file> <file>qml/CodeEditor.qml</file>
<file>qml/CodeEditorView.qml</file> <file>qml/CodeEditorView.qml</file>
<file>qml/js/ProjectModel.js</file>
</qresource> </qresource>
</RCC> </RCC>

13
mix/qml/CodeEditorModel.qml

@ -1,13 +0,0 @@
pragma Singleton
import QtQuick 2.0
import QtQuick.Window 2.0
import QtQuick.Layouts 1.0
import QtQuick.Controls 1.0
import QtQuick.Dialogs 1.1
Item {
id: codeEditorModel
property var codeDocuments: []
}

17
mix/qml/CodeEditorView.qml

@ -11,7 +11,7 @@ Item {
function getDocumentText(documentId) { function getDocumentText(documentId) {
for (i = 0; i < editorListModel.count; i++) { for (i = 0; i < editorListModel.count; i++) {
if (editorListModel.get(i).documentId === documentId) { if (editorListModel.get(i).documentId === documentId) {
return editors.itemAt(i).getDocumentText(); return editors.itemAt(i).getText();
} }
} }
return ""; return "";
@ -41,9 +41,20 @@ Item {
Connections { Connections {
target: ProjectModel target: ProjectModel
onDocumentOpen: { onDocumentOpened: {
openDocument(document); openDocument(document);
} }
onProjectSaved: {
for (var i = 0; i < editorListModel.count; i++)
fileIo.writeFile(editorListModel.get(i).path, editors.itemAt(i).item.getText());
}
onProjectClosed: {
for (var i = 0; i < editorListModel.count; i++) {
editors.itemAt(i).visible = false;
}
editorListModel.clear();
currentDocumentId = "";
}
} }
CodeEditor { CodeEditor {
@ -58,7 +69,7 @@ Item {
asynchronous: true asynchronous: true
anchors.fill: parent anchors.fill: parent
sourceComponent: codeEditor sourceComponent: codeEditor
visible: (currentDocumentId === editorListModel.get(index).documentId) visible: (index >= 0 && index < editorListModel.count && currentDocumentId === editorListModel.get(index).documentId)
onVisibleChanged: { onVisibleChanged: {
loadIfNotLoaded() loadIfNotLoaded()
} }

47
mix/qml/ProjectList.qml

@ -5,19 +5,36 @@ import QtQuick.Controls 1.0
import org.ethereum.qml.ProjectModel 1.0 import org.ethereum.qml.ProjectModel 1.0
Item { Item {
ListView {
id: projectList ColumnLayout {
model: ProjectModel.listModel
anchors.fill: parent anchors.fill: parent
delegate: renderDelegate Text {
highlight: Rectangle { Layout.fillWidth: true
color: "lightsteelblue"; color: "blue"
text: ProjectModel.projectData ? ProjectModel.projectData.title : ""
horizontalAlignment: Text.AlignHCenter
visible: !ProjectModel.isEmpty;
} }
highlightFollowsCurrentItem: true ListView {
focus: true Layout.fillWidth: true
clip: true Layout.fillHeight: true
} id: projectList
model: ProjectModel.listModel
delegate: renderDelegate
highlight: Rectangle {
color: "lightsteelblue";
}
highlightFollowsCurrentItem: true
focus: true
clip: true
onCurrentIndexChanged: {
if (currentIndex >= 0 && currentIndex < ProjectModel.listModel.count)
ProjectModel.openDocument(ProjectModel.listModel.get(currentIndex).documentId);
}
}
}
Component { Component {
id: renderDelegate id: renderDelegate
Item { Item {
@ -42,10 +59,16 @@ Item {
onClicked:{ onClicked:{
projectList.currentIndex = index; projectList.currentIndex = index;
ProjectModel.documentOpen(ProjectModel.listModel.get(index)); }
}
Connections {
target: ProjectModel
onProjectLoaded: {
projectList.currentIndex = 0;
} }
} }
} }
} }
} }

130
mix/qml/ProjectModel.qml

@ -7,119 +7,33 @@ import QtQuick.Controls 1.0
import QtQuick.Dialogs 1.1 import QtQuick.Dialogs 1.1
import Qt.labs.settings 1.0 import Qt.labs.settings 1.0
import "js/ProjectModel.js" as ProjectModelCode
Item { Item {
id: projectModel id: projectModel
signal projectClosed signal projectClosed
signal projectLoaded signal projectLoaded
signal documentOpen(var document) signal documentOpened(var document)
signal projectSaved
property bool isEmpty: (projectFile === "") property bool isEmpty: (projectData === null)
readonly property string projectFileName: ".mix" readonly property string projectFileName: ".mix"
property bool haveUnsavedChanges: false property bool haveUnsavedChanges: false
property string projectFile: "" property string projectPath: ""
property var projectData: null property var projectData: null
property var listModel: projectListModel property var listModel: projectListModel
function saveAll() { //interface
saveProject(); function saveAll() { ProjectModelCode.saveAll(); }
} function createProject() { ProjectModelCode.createProject(); }
function browseProject() { ProjectModelCode.browseProject(); }
function createProject() { function closeProject() { ProjectModelCode.closeProject(); }
closeProject(); function saveProject() { ProjectModelCode.saveProject(); }
newProjectDialog.open(); function loadProject(path) { ProjectModelCode.loadProject(path); }
} function addExistingFile() { ProjectModelCode.addExistingFile(); }
function openDocument(documentId) { ProjectModelCode.openDocument(documentId); }
function browseProject() {
openProjectFileDialog.open();
}
function closeProject() {
if (!isEmpty) {
console.log("closing project");
if (haveUnsavedChanges)
saveMessageDialog.open();
else
doCloseProject();
}
}
function saveProject() {
if (!isEmpty) {
var json = JSON.stringify(projectData);
fileIo.writeFile(projectFile, json);
}
}
function loadProject(path) {
closeProject();
console.log("loading project at " + path);
var json = fileIo.readFile(path);
projectData = JSON.parse(json);
projectFile = path;
if (!projectData.files)
projectData.files = [];
for(var i = 0; i < projectData.files.length; i++) {
var p = projectData.files[i];
addFile(p);
}
projectSettings.lastProjectPath = projectFile;
projectLoaded();
}
function addExistingFile() {
addExistingFileDialog().open();
}
function addProjectFiles(files) {
for(var i = 0; i < files.length; i++)
addFile(files[i]);
}
function addFile(file) {
var p = file;
var fileData = {
contract: false,
path: p,
name: p.substring(p.lastIndexOf("/") + 1, p.length),
documentId: p,
isText: true,
isContract: p.substring(p.length - 4, p.length) === ".sol",
};
projectListModel.append(fileData);
}
function doCloseProject() {
projectListModel.clear();
projectFile = "";
projectData = null;
projectClosed();
}
function doCreateProject(title, path) {
closeProject();
console.log("creating project " + title + " at " + path);
if (path[path.length - 1] !== "/")
path += "/";
var dirPath = path + title;
fileIo.makeDir(dirPath);
var projectFile = dirPath + "/" + projectFileName;
var indexFile = dirPath + "/index.html";
var contractsFile = dirPath + "/contracts.sol";
var projectData = {
files: [ indexFile, contractsFile ]
};
fileIo.writeFile(indexFile, "<html></html>");
fileIo.writeFile(contractsFile, "contract MyContract {}");
var json = JSON.stringify(projectData);
fileIo.writeFile(projectFile, json);
loadProject(projectFile);
}
Component.onCompleted: { Component.onCompleted: {
if (projectSettings.lastProjectPath) if (projectSettings.lastProjectPath)
@ -132,7 +46,7 @@ Item {
onAccepted: { onAccepted: {
var title = newProjectDialog.projectTitle; var title = newProjectDialog.projectTitle;
var path = newProjectDialog.projectPath; var path = newProjectDialog.projectPath;
doCreateProject(title, path); ProjectModelCode.doCreateProject(title, path);
} }
} }
@ -144,10 +58,10 @@ Item {
icon: StandardIcon.Question icon: StandardIcon.Question
onAccepted: { onAccepted: {
projectModel.saveAll(); projectModel.saveAll();
projectModel.doCloseProject(); ProjectModelCode.doCloseProject();
} }
onRejected: { onRejected: {
projectModel.doCloseProject(); ProjectModelCode.doCloseProject();
} }
} }
@ -167,8 +81,8 @@ Item {
selectFolder: true selectFolder: true
onAccepted: { onAccepted: {
var path = openProjectFileDialog.fileUrl.toString(); var path = openProjectFileDialog.fileUrl.toString();
path += "/" + projectFileName; path += "/";
loadProject(path); loadProject(path);
} }
} }
@ -179,9 +93,7 @@ Item {
selectFolder: false selectFolder: false
onAccepted: { onAccepted: {
var paths = openProjectFileDialog.fileUrls; var paths = openProjectFileDialog.fileUrls;
addProjectFiles(paths); ProjectModelCode.doAddExistingFiles(paths);
} }
} }
} }

142
mix/qml/js/ProjectModel.js

@ -0,0 +1,142 @@
/*
This file is part of cpp-ethereum.
cpp-ethereum is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
cpp-ethereum is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
*/
/** @file ProjectModel.js
* @author Arkadiy Paronyan arkadiy@ethdev.com
* @date 2015
* Ethereum IDE client.
*/
function saveAll() {
saveProject();
projectSaved();
}
function createProject() {
newProjectDialog.open();
}
function browseProject() {
openProjectFileDialog.open();
}
function closeProject() {
if (!isEmpty) {
if (haveUnsavedChanges)
saveMessageDialog.open();
else
doCloseProject();
}
}
function saveProject() {
if (!isEmpty) {
var json = JSON.stringify(projectData);
var projectFile = projectPath + projectFileName;
fileIo.writeFile(projectFile, json);
}
}
function loadProject(path) {
closeProject();
console.log("loading project at " + path);
var projectFile = path + projectFileName;
var json = fileIo.readFile(projectFile);
projectData = JSON.parse(json);
if (!projectData.title) {
var parts = path.split("/");
projectData.title = parts[parts.length - 2];
}
projectPath = path;
if (!projectData.files)
projectData.files = [];
for(var i = 0; i < projectData.files.length; i++) {
addFile(projectData.files[i]);
}
projectSettings.lastProjectPath = path;
projectLoaded();
}
function addExistingFile() {
addExistingFileDialog().open();
}
function addProjectFiles(files) {
for(var i = 0; i < files.length; i++)
addFile(files[i]);
}
function addFile(fileName) {
var p = projectPath + fileName;
var fileData = {
contract: false,
path: p,
name: fileName,
documentId: fileName,
isText: true,
isContract: fileName.substring(fileName.length - 4, fileName.length) === ".sol",
};
projectListModel.append(fileData);
}
function openDocument(documentId) {
for (var i = 0; i < projectListModel.count; i++)
if (projectListModel.get(i).documentId === documentId)
documentOpened(projectListModel.get(i));
}
function doCloseProject() {
console.log("closing project");
projectListModel.clear();
projectPath = "";
projectData = null;
projectClosed();
}
function doCreateProject(title, path) {
closeProject();
console.log("creating project " + title + " at " + path);
if (path[path.length - 1] !== "/")
path += "/";
var dirPath = path + title + "/";
fileIo.makeDir(dirPath);
var projectFile = dirPath + projectFileName;
var indexFile = dirPath + "index.html";
var contractsFile = dirPath + "contracts.sol";
var projectData = {
title: title,
files: [ "contracts.sol", "index.html" ]
};
fileIo.writeFile(indexFile, "<html></html>");
fileIo.writeFile(contractsFile, "contract MyContract {}");
var json = JSON.stringify(projectData);
fileIo.writeFile(projectFile, json);
loadProject(dirPath);
}
function doAddExistingFiles(files) {
for(var i = 0; i < files.length; i++) {
var sourcePath = files[i];
var sourceFileName = sourcePath.substring(sourcePath.lastIndexOf("/") + 1, sourcePath.length);
var destPath = projectPath + sourceFileName;
fileIo.copyFile(sourcePath, destPath);
addFile(sourceFileName);
}
}

29
mix/qml/main.qml

@ -21,14 +21,12 @@ ApplicationWindow {
title: qsTr("File") title: qsTr("File")
MenuItem { action: createProjectAction } MenuItem { action: createProjectAction }
MenuItem { action: openProjectAction } MenuItem { action: openProjectAction }
MenuItem { action: saveAllFilesAction }
MenuItem { action: addExistingFileAction } MenuItem { action: addExistingFileAction }
MenuItem { action: addNewJsFileAction } MenuItem { action: addNewJsFileAction }
MenuItem { action: addNewHtmlFileAction } MenuItem { action: addNewHtmlFileAction }
MenuItem { action: addNewContractAction } MenuItem { action: addNewContractAction }
MenuItem { MenuItem { action: exitAppAction }
text: qsTr("Exit")
onTriggered: Qt.quit();
}
} }
Menu { Menu {
title: qsTr("Debug") title: qsTr("Debug")
@ -54,6 +52,13 @@ ApplicationWindow {
id: messageDialog id: messageDialog
} }
Action {
id: exitAppAction
text: qsTr("Exit")
shortcut: "Ctrl+Q"
onTriggered: Qt.quit();
}
Action { Action {
id: debugRunAction id: debugRunAction
text: "&Run" text: "&Run"
@ -116,4 +121,20 @@ ApplicationWindow {
enabled: !ProjectModel.isEmpty enabled: !ProjectModel.isEmpty
onTriggered: ProjectModel.addExistingFile(); onTriggered: ProjectModel.addExistingFile();
} }
Action {
id: saveAllFilesAction
text: qsTr("Save all")
shortcut: "Ctrl+S"
enabled: !ProjectModel.isEmpty
onTriggered: ProjectModel.saveAll();
}
Action {
id: closeProjectAction
text: qsTr("Close project")
shortcut: "Ctrl+W"
enabled: !ProjectModel.isEmpty
onTriggered: ProjectModel.closeProject();
}
} }

Loading…
Cancel
Save