From e9bde6e5624618a7a4ae5c4f918187c7ee789943 Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Tue, 8 Dec 2015 15:57:37 +0300 Subject: [PATCH] new file creation --- dlangide_msvc.visualdproj | 1 + src/dlangide/ui/commands.d | 4 +- src/dlangide/ui/frame.d | 53 ++++++ src/dlangide/ui/newfile.d | 313 +++++++++++++++++++++++++++++++ src/dlangide/ui/newproject.d | 4 + src/dlangide/ui/wspanel.d | 30 ++- src/dlangide/workspace/project.d | 80 ++++++++ 7 files changed, 478 insertions(+), 7 deletions(-) create mode 100644 src/dlangide/ui/newfile.d diff --git a/dlangide_msvc.visualdproj b/dlangide_msvc.visualdproj index 87ceee2..aaa86a9 100644 --- a/dlangide_msvc.visualdproj +++ b/dlangide_msvc.visualdproj @@ -446,6 +446,7 @@ + diff --git a/src/dlangide/ui/commands.d b/src/dlangide/ui/commands.d index 160a0e8..163df66 100644 --- a/src/dlangide/ui/commands.d +++ b/src/dlangide/ui/commands.d @@ -37,7 +37,7 @@ enum IDEActions : int { WindowCloseAllDocuments, CreateNewWorkspace, AddToCurrentWorkspace, - ProjectFolderAddItem, + //ProjectFolderAddItem, ProjectFolderRemoveItem, ProjectFolderOpenItem, ProjectFolderRenameItem, @@ -53,7 +53,7 @@ __gshared static this() { } -const Action ACTION_PROJECT_FOLDER_ADD_ITEM = new Action(IDEActions.ProjectFolderAddItem, "MENU_PROJECT_FOLDER_ADD_ITEM"c); +//const Action ACTION_PROJECT_FOLDER_ADD_ITEM = new Action(IDEActions.ProjectFolderAddItem, "MENU_PROJECT_FOLDER_ADD_ITEM"c); const Action ACTION_PROJECT_FOLDER_OPEN_ITEM = new Action(IDEActions.ProjectFolderOpenItem, "MENU_PROJECT_FOLDER_OPEN_ITEM"c); const Action ACTION_PROJECT_FOLDER_REMOVE_ITEM = new Action(IDEActions.ProjectFolderRemoveItem, "MENU_PROJECT_FOLDER_REMOVE_ITEM"c); const Action ACTION_PROJECT_FOLDER_RENAME_ITEM = new Action(IDEActions.ProjectFolderRenameItem, "MENU_PROJECT_FOLDER_RENAME_ITEM"c); diff --git a/src/dlangide/ui/frame.d b/src/dlangide/ui/frame.d index 6e1621f..aac9946 100644 --- a/src/dlangide/ui/frame.d +++ b/src/dlangide/ui/frame.d @@ -20,6 +20,7 @@ import dlangui.core.files; import dlangide.ui.commands; import dlangide.ui.wspanel; import dlangide.ui.outputpanel; +import dlangide.ui.newfile; import dlangide.ui.newproject; import dlangide.ui.dsourceedit; import dlangide.ui.homescreen; @@ -332,6 +333,7 @@ class IDEFrame : AppFrame { // Create workspace docked panel _wsPanel = new WorkspacePanel("workspace"); _wsPanel.sourceFileSelectionListener = &onSourceFileSelected; + _wsPanel.workspaceActionListener = &handleAction; _wsPanel.dockAlignment = DockAlignment.Left; _dockHost.addDockedWindow(_wsPanel); @@ -633,6 +635,9 @@ class IDEFrame : AppFrame { case IDEActions.FileNewProject: createNewProject(false); return true; + case IDEActions.FileNew: + addProjectItem(a.objectParam); + return true; default: return super.handleAction(a); } @@ -640,6 +645,54 @@ class IDEFrame : AppFrame { return false; } + @property ProjectSourceFile currentEditorSourceFile() { + TabItem tab = _tabs.selectedTab; + if (tab) { + return cast(ProjectSourceFile)tab.objectParam; + } + return null; + } + + void addProjectItem(const Object obj) { + if (currentWorkspace is null) + return; + Project project; + ProjectFolder folder; + if (cast(Project)obj) { + project = cast(Project)obj; + } else if (cast(ProjectFolder)obj) { + folder = cast(ProjectFolder)obj; + project = folder.project; + } else if (cast(ProjectSourceFile)obj) { + ProjectSourceFile srcfile = cast(ProjectSourceFile)obj; + folder = cast(ProjectFolder)srcfile.parent; + project = srcfile.project; + } else { + ProjectSourceFile srcfile = currentEditorSourceFile; + if (srcfile) { + folder = cast(ProjectFolder)srcfile.parent; + project = srcfile.project; + } + } + if (project && folder && project.workspace is currentWorkspace) { + NewFileDlg dlg = new NewFileDlg(this, project, folder); + dlg.dialogResult = delegate(Dialog dlg, const Action result) { + if (result.id == ACTION_FILE_NEW_SOURCE_FILE.id) { + FileCreationResult res = cast(FileCreationResult)result.objectParam; + if (res) { + //res.project.reload(); + res.project.refresh(); + refreshWorkspace(); + if (isSupportedSourceTextFileFormat(res.filename)) { + openSourceFile(res.filename, null, true); + } + } + } + }; + dlg.show(); + } + } + void createNewProject(bool newWorkspace) { if (currentWorkspace is null) newWorkspace = true; diff --git a/src/dlangide/ui/newfile.d b/src/dlangide/ui/newfile.d new file mode 100644 index 0000000..533d82a --- /dev/null +++ b/src/dlangide/ui/newfile.d @@ -0,0 +1,313 @@ +module dlangide.ui.newfile; + +import dlangui.core.types; +import dlangui.core.i18n; +import dlangui.platforms.common.platform; +import dlangui.dialogs.dialog; +import dlangui.dialogs.filedlg; +import dlangui.widgets.widget; +import dlangui.widgets.layouts; +import dlangui.widgets.editors; +import dlangui.widgets.controls; +import dlangui.widgets.lists; +import dlangui.dml.parser; +import dlangui.core.stdaction; +import dlangui.core.files; +import dlangide.workspace.project; +import dlangide.workspace.workspace; +import dlangide.ui.commands; +import dlangide.ui.frame; + +import std.path; +import std.file; +import std.array : empty; +import std.algorithm : startsWith, endsWith; + +class FileCreationResult { + Project project; + string filename; + this(Project project, string filename) { + this.project = project; + this.filename = filename; + } +} + +class NewFileDlg : Dialog { + IDEFrame _ide; + Project _project; + ProjectFolder _folder; + string[] _sourcePaths; + this(IDEFrame parent, Project currentProject, ProjectFolder folder) { + super(UIString("New source file"d), parent.window, + DialogFlag.Modal | DialogFlag.Resizable | DialogFlag.Popup, 500, 400); + _ide = parent; + _icon = "dlangui-logo1"; + this._project = currentProject; + this._folder = folder; + _location = folder ? folder.filename : currentProject.dir; + _sourcePaths = currentProject.sourcePaths; + if (_sourcePaths.length) + _location = _sourcePaths[0]; + if (folder) + _location = folder.filename; + } + /// override to implement creation of dialog controls + override void init() { + super.init(); + initTemplates(); + Widget content; + try { + content = parseML(q{ + VerticalLayout { + id: vlayout + padding: Rect { 5, 5, 5, 5 } + layoutWidth: fill; layoutHeight: fill + HorizontalLayout { + layoutWidth: fill; layoutHeight: fill + VerticalLayout { + margins: 5 + layoutWidth: wrap; layoutHeight: fill + TextWidget { text: "Project template" } + StringListWidget { + id: projectTemplateList + layoutWidth: wrap; layoutHeight: fill + } + } + VerticalLayout { + margins: 5 + layoutWidth: fill; layoutHeight: fill + TextWidget { text: "Template description" } + EditBox { + id: templateDescription; readOnly: true + layoutWidth: fill; layoutHeight: fill + } + } + } + TableLayout { + margins: 5 + colCount: 2 + layoutWidth: fill; layoutHeight: wrap + TextWidget { text: "Name" } + EditLine { id: edName; text: "newfile"; layoutWidth: fill } + TextWidget { text: "Location" } + DirEditLine { id: edLocation; layoutWidth: fill } + TextWidget { text: "Module name" } + EditLine { id: edModuleName; text: ""; layoutWidth: fill; readOnly: true } + TextWidget { text: "File path" } + EditLine { id: edFilePath; text: ""; layoutWidth: fill; readOnly: true } + } + TextWidget { id: statusText; text: ""; layoutWidth: fill; textColor: #FF0000 } + } + }); + } catch (Exception e) { + Log.e("Exceptin while parsing DML", e); + throw e; + } + + + _projectTemplateList = content.childById!StringListWidget("projectTemplateList"); + _templateDescription = content.childById!EditBox("templateDescription"); + _edFileName = content.childById!EditLine("edName"); + _edFilePath = content.childById!EditLine("edFilePath"); + _edModuleName = content.childById!EditLine("edModuleName"); + _edLocation = content.childById!DirEditLine("edLocation"); + _edLocation.text = toUTF32(_location); + _statusText = content.childById!TextWidget("statusText"); + + _edLocation.filetypeIcons[".d"] = "text-d"; + _edLocation.filetypeIcons["dub.json"] = "project-d"; + _edLocation.filetypeIcons["package.json"] = "project-d"; + _edLocation.filetypeIcons[".dlangidews"] = "project-development"; + _edLocation.addFilter(FileFilterEntry(UIString("DlangIDE files"d), "*.dlangidews;*.d;*.dd;*.di;*.ddoc;*.dh;*.json;*.xml;*.ini")); + _edLocation.caption = "Select directory"d; + + // fill templates + dstring[] names; + foreach(t; _templates) + names ~= t.name; + _projectTemplateList.items = names; + _projectTemplateList.selectedItemIndex = 0; + + templateSelected(0); + + // listeners + _edLocation.contentChange = delegate (EditableContent source) { + _location = toUTF8(source.text); + validate(); + }; + + _edFileName.contentChange = delegate (EditableContent source) { + _fileName = toUTF8(source.text); + validate(); + }; + + _projectTemplateList.itemSelected = delegate (Widget source, int itemIndex) { + templateSelected(itemIndex); + return true; + }; + _projectTemplateList.itemClick = delegate (Widget source, int itemIndex) { + templateSelected(itemIndex); + return true; + }; + + addChild(content); + addChild(createButtonsPanel([ACTION_FILE_NEW_SOURCE_FILE, ACTION_CANCEL], 0, 0)); + + } + + StringListWidget _projectTemplateList; + EditBox _templateDescription; + DirEditLine _edLocation; + EditLine _edFileName; + EditLine _edModuleName; + EditLine _edFilePath; + TextWidget _statusText; + + string _fileName = "newfile"; + string _location; + string _moduleName; + string _packageName; + string _fullPathName; + + int _currentTemplateIndex = -1; + ProjectTemplate _currentTemplate; + ProjectTemplate[] _templates; + + static bool isSubdirOf(string path, string basePath) { + if (path.equal(basePath)) + return true; + if (path.length > basePath.length + 1 && path.startsWith(basePath)) { + char ch = path[basePath.length]; + return ch == '/' || ch == '\\'; + } + return false; + } + + bool findSource(string path, ref string sourceFolderPath, ref string relativePath) { + foreach(dir; _sourcePaths) { + if (isSubdirOf(path, dir)) { + sourceFolderPath = dir; + relativePath = path[sourceFolderPath.length .. $]; + if (relativePath.length > 0 && (relativePath[0] == '\\' || relativePath[0] == '/')) + relativePath = relativePath[1 .. $]; + return true; + } + } + return false; + } + + bool setError(dstring msg) { + _statusText.text = msg; + return msg.empty; + } + + bool validate() { + string filename = _fileName; + string fullFileName = filename; + if (!_currentTemplate.fileExtension.empty && filename.endsWith(_currentTemplate.fileExtension)) + filename = filename[0 .. $ - _currentTemplate.fileExtension.length]; + else + fullFileName = fullFileName ~ _currentTemplate.fileExtension; + _fullPathName = buildNormalizedPath(_location, fullFileName); + _edFilePath.text = toUTF32(_fullPathName); + if (!isValidFileName(filename)) + return setError("Invalid file name"); + if (!exists(_location) || !isDir(_location)) + return setError("Location directory does not exist"); + + if (_currentTemplate.isModule) { + string sourcePath, relativePath; + if (!findSource(_location, sourcePath, relativePath)) + return setError("Location is outside of source path"); + if (!isValidModuleName(filename)) + return setError("Invalid file name"); + _moduleName = filename; + char[] buf; + foreach(ch; relativePath) { + if (ch == '/' || ch == '\\') + buf ~= '.'; + else + buf ~= ch; + } + _packageName = buf.dup; + string m = !_packageName.empty ? _packageName ~ '.' ~ _moduleName : _moduleName; + _edModuleName.text = toUTF32(m); + } else { + string projectPath = _project.dir; + if (!isSubdirOf(_location, projectPath)) + return setError("Location is outside of project path"); + _edModuleName.text = ""; + _moduleName = ""; + _packageName = ""; + } + return true; + } + + private FileCreationResult _result; + bool createItem() { + try { + if (_currentTemplate.isModule) { + string txt = "module " ~ _packageName ~ ";\n\n" ~ _currentTemplate.srccode; + write(_fullPathName, txt); + } else { + write(_fullPathName, _currentTemplate.srccode); + } + } catch (Exception e) { + Log.e("Cannot create file", e); + return setError("Cannot create file"); + } + _result = new FileCreationResult(_project, _fullPathName); + return true; + } + + override void close(const Action action) { + Action newaction = action.clone(); + if (action.id == IDEActions.FileNew) { + if (!validate()) { + window.showMessageBox(UIString("Error"d), UIString("Invalid parameters")); + return; + } + if (!createItem()) { + window.showMessageBox(UIString("Error"d), UIString("Failed to create project item")); + return; + } + newaction.objectParam = _result; + } + super.close(newaction); + } + + protected void templateSelected(int index) { + if (_currentTemplateIndex == index) + return; + _currentTemplateIndex = index; + _currentTemplate = _templates[index]; + _templateDescription.text = _currentTemplate.description; + //updateDirLayout(); + validate(); + } + + void initTemplates() { + _templates ~= new ProjectTemplate("Empty module"d, "Empty D module file."d, ".d", + "\n", true); + _templates ~= new ProjectTemplate("Text file"d, "Empty text file."d, ".txt", + "\n", true); + _templates ~= new ProjectTemplate("JSON file"d, "Empty json file."d, ".json", + "{\n}\n", true); + } +} + +class ProjectTemplate { + dstring name; + dstring description; + string fileExtension; + string srccode; + bool isModule; + this(dstring name, dstring description, string fileExtension, string srccode, bool isModule) { + this.name = name; + this.description = description; + this.fileExtension = fileExtension; + this.srccode = srccode; + this.isModule = isModule; + } +} + diff --git a/src/dlangide/ui/newproject.d b/src/dlangide/ui/newproject.d index 54cab16..88c7dc6 100644 --- a/src/dlangide/ui/newproject.d +++ b/src/dlangide/ui/newproject.d @@ -284,6 +284,10 @@ class NewProjectDlg : Dialog { if (!exists(_location) || !isDir(_location)) { return setError("Invalid location"); } + if (!isValidProjectName(_projectName)) + return setError("Invalid project name"); + if (!isValidProjectName(_workspaceName)) + return setError("Invalid workspace name"); return setError(""); } diff --git a/src/dlangide/ui/wspanel.d b/src/dlangide/ui/wspanel.d index ff0d382..8fa8828 100644 --- a/src/dlangide/ui/wspanel.d +++ b/src/dlangide/ui/wspanel.d @@ -17,12 +17,17 @@ interface SourceFileSelectionHandler { bool onSourceFileSelected(ProjectSourceFile file, bool activate); } +interface WorkspaceActionHandler { + bool onWorkspaceAction(const Action a); +} + class WorkspacePanel : DockWindow { protected Workspace _workspace; protected TreeWidget _tree; /// handle source file selection change Signal!SourceFileSelectionHandler sourceFileSelectionListener; + Signal!WorkspaceActionHandler workspaceActionListener; this(string id) { super(id); @@ -70,17 +75,17 @@ class WorkspacePanel : DockWindow { _tree.popupMenu = &onTreeItemPopupMenu; _workspacePopupMenu = new MenuItem(); - _workspacePopupMenu.add(ACTION_PROJECT_FOLDER_ADD_ITEM); + _workspacePopupMenu.add(ACTION_FILE_NEW_SOURCE_FILE.clone()); _projectPopupMenu = new MenuItem(); - _projectPopupMenu.add(ACTION_PROJECT_FOLDER_ADD_ITEM, ACTION_PROJECT_FOLDER_OPEN_ITEM, + _projectPopupMenu.add(ACTION_FILE_NEW_SOURCE_FILE, ACTION_PROJECT_FOLDER_OPEN_ITEM, ACTION_PROJECT_FOLDER_REMOVE_ITEM); _folderPopupMenu = new MenuItem(); - _folderPopupMenu.add(ACTION_PROJECT_FOLDER_ADD_ITEM, ACTION_PROJECT_FOLDER_OPEN_ITEM, + _folderPopupMenu.add(ACTION_FILE_NEW_SOURCE_FILE, ACTION_PROJECT_FOLDER_OPEN_ITEM, ACTION_PROJECT_FOLDER_REMOVE_ITEM, ACTION_PROJECT_FOLDER_RENAME_ITEM); _filePopupMenu = new MenuItem(); - _filePopupMenu.add(ACTION_PROJECT_FOLDER_ADD_ITEM, ACTION_PROJECT_FOLDER_OPEN_ITEM, + _filePopupMenu.add(ACTION_FILE_NEW_SOURCE_FILE, ACTION_PROJECT_FOLDER_OPEN_ITEM, ACTION_PROJECT_FOLDER_REMOVE_ITEM, ACTION_PROJECT_FOLDER_RENAME_ITEM); return _tree; } @@ -108,7 +113,15 @@ class WorkspacePanel : DockWindow { menu = _workspacePopupMenu; } if (menu && menu.subitemCount) { - menu.onMenuItem = &onPopupMenuItem; + for (int i = 0; i < menu.subitemCount; i++) { + Action a = menu.subitem(i).action.clone(); + a.objectParam = selectedItem.objectParam; + menu.subitem(i).action = a; + //menu.subitem(i).menuItemAction = &handleAction; + } + //menu.onMenuItem = &onPopupMenuItem; + //menu.menuItemClick = &onPopupMenuItem; + menu.menuItemAction = &handleAction; menu.updateActionState(this); return menu; } @@ -161,4 +174,11 @@ class WorkspacePanel : DockWindow { _workspace = w; reloadItems(); } + + /// override to handle specific actions + override bool handleAction(const Action a) { + if (workspaceActionListener.assigned) + return workspaceActionListener(a); + return false; + } } diff --git a/src/dlangide/workspace/project.d b/src/dlangide/workspace/project.d index de101f5..dff05aa 100644 --- a/src/dlangide/workspace/project.d +++ b/src/dlangide/workspace/project.d @@ -10,6 +10,7 @@ import std.json; import std.utf; import std.algorithm; import std.process; +import std.array; /// return true if filename matches rules for workspace file names bool isProjectFile(string filename) { @@ -74,6 +75,9 @@ class ProjectItem { ProjectItem child(int index) { return null; } + + void refresh() { + } } /// Project folder @@ -99,9 +103,30 @@ class ProjectFolder : ProjectItem { item._parent = this; item._project = _project; } + ProjectItem childByPathName(string path) { + for (int i = 0; i < _children.count; i++) { + if (_children[i].filename.equal(path)) + return _children[i]; + } + return null; + } + ProjectItem childByName(dstring s) { + for (int i = 0; i < _children.count; i++) { + if (_children[i].name.equal(s)) + return _children[i]; + } + return null; + } + bool loadDir(string path) { string src = relativeToAbsolutePath(path); if (exists(src) && isDir(src)) { + ProjectFolder existing = cast(ProjectFolder)childByPathName(src); + if (existing) { + if (existing.isFolder) + existing.loadItems(); + return true; + } ProjectFolder dir = new ProjectFolder(src); addChild(dir); Log.d(" added project folder ", src); @@ -110,9 +135,13 @@ class ProjectFolder : ProjectItem { } return false; } + bool loadFile(string path) { string src = relativeToAbsolutePath(path); if (exists(src) && isFile(src)) { + ProjectItem existing = childByPathName(src); + if (existing) + return true; ProjectSourceFile f = new ProjectSourceFile(src); addChild(f); Log.d(" added project file ", src); @@ -120,21 +149,36 @@ class ProjectFolder : ProjectItem { } return false; } + void loadItems() { + bool[string] loaded; foreach(e; dirEntries(_filename, SpanMode.shallow)) { string fn = baseName(e.name); if (e.isDir) { loadDir(fn); + loaded[fn] = true; } else if (e.isFile) { loadFile(fn); + loaded[fn] = true; + } + } + // removing non-reloaded items + for (int i = _children.count - 1; i >= 0; i--) { + if (!(toUTF8(_children[i].name) in loaded)) { + _children.remove(i); } } } + string relativeToAbsolutePath(string path) { if (isAbsolute(path)) return path; return buildNormalizedPath(_filename, path); } + + override void refresh() { + loadItems(); + } } /// Project source file @@ -408,6 +452,10 @@ class Project : WorkspaceItem { return folder; } + void refresh() { + _items.refresh(); + } + void findMainSourceFile() { string n = toUTF8(name); string[] mainnames = ["app.d", "main.d", n ~ ".d"]; @@ -575,3 +623,35 @@ class DubPackageFinder { } } +bool isValidProjectName(string s) { + if (s.empty) + return false; + for (int i = 0; i < s.length; i++) { + char ch = s[i]; + if (ch != '_' && ch != '-' && (ch < '0' || ch > '9') && (ch < 'a' || ch > 'z') && (ch < 'A' || ch > 'Z')) + return false; + } + return true; +} + +bool isValidModuleName(string s) { + if (s.empty) + return false; + for (int i = 0; i < s.length; i++) { + char ch = s[i]; + if (ch != '_' && (ch < '0' || ch > '9') && (ch < 'a' || ch > 'z') && (ch < 'A' || ch > 'Z')) + return false; + } + return true; +} + +bool isValidFileName(string s) { + if (s.empty) + return false; + for (int i = 0; i < s.length; i++) { + char ch = s[i]; + if (ch != '_' && ch != '.' && ch != '-' && (ch < '0' || ch > '9') && (ch < 'a' || ch > 'z') && (ch < 'A' || ch > 'Z')) + return false; + } + return true; +}