# HG changeset patch # User laurent # Date 1340878041 -7200 # Node ID 6f0e10085df92b9a45647bf41c6c65872cb99388 # Parent cdc6393705ce35f4c795518e2c6ed2754a83fb8d Adding support for file explorer for project files diff -r cdc6393705ce -r 6f0e10085df9 Beremiz.py --- a/Beremiz.py Mon Jun 25 20:05:29 2012 +0200 +++ b/Beremiz.py Thu Jun 28 12:07:21 2012 +0200 @@ -562,7 +562,7 @@ ResourceEditor, ConfigurationEditor, DataTypeEditor))): - return ("confnode", tab.Controler.CTNFullName()) + return ("confnode", tab.Controler.CTNFullName(), tab.GetTagName()) elif (isinstance(tab, TextViewer) and (tab.Controler is None or isinstance(tab.Controler, MiniTextControler))): return ("confnode", None, tab.GetInstancePath()) @@ -941,6 +941,17 @@ else: IDEFrame.OnProjectTreeItemActivated(self, event) + def ProjectTreeItemSelect(self, select_item): + name = self.ProjectTree.GetItemText(select_item) + item_infos = self.ProjectTree.GetPyData(select_item) + if item_infos["type"] == ITEM_CONFNODE: + item_infos["confnode"]._OpenView(onlyopened=True) + elif item_infos["type"] == ITEM_PROJECT: + self.CTR._OpenView(onlyopened=True) + else: + IDEFrame.ProjectTreeItemSelect(self, select_item) + + def SelectProjectTreeItem(self, tagname): if self.ProjectTree is not None: root = self.ProjectTree.GetRootItem() diff -r cdc6393705ce -r 6f0e10085df9 ConfigTreeNode.py --- a/ConfigTreeNode.py Mon Jun 25 20:05:29 2012 +0200 +++ b/ConfigTreeNode.py Thu Jun 28 12:07:21 2012 +0200 @@ -405,15 +405,18 @@ self.BaseParams.setIEC_Channel(res) return res - def _OpenView(self, name=None): + def _OpenView(self, name=None, onlyopened=False): if self.EditorType is not None: + app_frame = self.GetCTRoot().AppFrame if self._View is None: - app_frame = self.GetCTRoot().AppFrame self._View = self.EditorType(app_frame.TabsOpened, self, app_frame) app_frame.EditProjectElement(self._View, self.CTNName()) - + + elif onlyopened: + app_frame.EditProjectElement(self._View, self.CTNName(), onlyopened) + return self._View return None diff -r cdc6393705ce -r 6f0e10085df9 ProjectController.py --- a/ProjectController.py Mon Jun 25 20:05:29 2012 +0200 +++ b/ProjectController.py Thu Jun 28 12:07:21 2012 +0200 @@ -17,6 +17,7 @@ from util.misc import CheckPathPerm, GetClassImporter from util.MiniTextControler import MiniTextControler from util.ProcessLogger import ProcessLogger +from util.FileManagementPanel import FileManagementPanel from PLCControler import PLCControler from TextViewer import TextViewer from plcopen.structures import IEC_KEYWORDS @@ -24,6 +25,7 @@ from util.discovery import DiscoveryDialog from ConfigTreeNode import ConfigTreeNode from ProjectNodeEditor import ProjectNodeEditor +from utils.BitmapLibrary import GetBitmap base_folder = os.path.split(sys.path[0])[0] @@ -932,14 +934,18 @@ _IECRawCodeView = None def _editIECrawcode(self): self._OpenView("IEC raw code") - - def _OpenView(self, name=None): + + _ProjectFilesView = None + def _OpenProjectFiles(self): + self._OpenView("Project files") + + def _OpenView(self, name=None, onlyopened=False): if name == "IEC code": - if self._IEC_code_viewer is None: + if self._IECCodeView is None: plc_file = self._getIECcodepath() self._IECCodeView = TextViewer(self.AppFrame.TabsOpened, "", None, None, instancepath=name) - #self._IECCodeViewr.Enable(False) + #self._IECCodeView.Enable(False) self._IECCodeView.SetTextSyntax("ALL") self._IECCodeView.SetKeywords(IEC_KEYWORDS) try: @@ -947,29 +953,46 @@ except: text = '(* No IEC code have been generated at that time ! *)' self._IECCodeView.SetText(text = text) - self._IECCodeView.SetIcon(self.AppFrame.GenerateBitmap("ST")) - + self._IECCodeView.SetIcon(GetBitmap("ST")) + self.AppFrame.EditProjectElement(self._IECCodeView, name) - + + elif onlyopened: + self.AppFrame.EditProjectElement(self._IECCodeView, name, onlyopened) + return self._IECCodeView elif name == "IEC raw code": - if self.IEC_raw_code_viewer is None: + if self._IECRawCodeView is None: controler = MiniTextControler(self._getIECrawcodepath()) - self.IEC_raw_code_viewer = TextViewer(self.AppFrame.TabsOpened, "", None, controler, instancepath=name) - #self.IEC_raw_code_viewer.Enable(False) - self.IEC_raw_code_viewer.SetTextSyntax("ALL") - self.IEC_raw_code_viewer.SetKeywords(IEC_KEYWORDS) - self.IEC_raw_code_viewer.RefreshView() - self.IEC_raw_code_viewer.SetIcon(self.AppFrame.GenerateBitmap("ST")) + self._IECRawCodeView = TextViewer(self.AppFrame.TabsOpened, "", None, controler, instancepath=name) + #self._IECRawCodeView.Enable(False) + self._IECRawCodeView.SetTextSyntax("ALL") + self._IECRawCodeView.SetKeywords(IEC_KEYWORDS) + self._IECRawCodeView.RefreshView() + self._IECRawCodeView.SetIcon(GetBitmap("ST")) - self.AppFrame.EditProjectElement(self.IEC_raw_code_viewer, name) - - return self.IEC_raw_code_viewer - + self.AppFrame.EditProjectElement(self._IECRawCodeView, name) + + elif onlyopened: + self.AppFrame.EditProjectElement(self._IECRawCodeView, name, onlyopened) + + return self._IECRawCodeView + + elif name == "Project files": + if self._ProjectFilesView is None: + self._ProjectFilesView = FileManagementPanel(self.AppFrame.TabsOpened, self, name, self._getProjectFilesPath(), True) + + self.AppFrame.EditProjectElement(self._ProjectFilesView, name) + + elif onlyopened: + self.AppFrame.EditProjectElement(self._ProjectFilesView, name, onlyopened) + + return self._ProjectFilesView + else: - return ConfigTreeNode._OpenView(self, name) + return ConfigTreeNode._OpenView(self, name, onlyopened) def OnCloseEditor(self, view): ConfigTreeNode.OnCloseEditor(self, view) @@ -977,6 +1000,8 @@ self._IECCodeView = None if self._IECRawCodeView == view: self._IECRawCodeView = None + if self._ProjectFilesView == view: + self._ProjectFilesView = None def _Clean(self): self._CloseView(self._IECCodeView) @@ -1414,16 +1439,6 @@ wx.CallAfter(self.UpdateMethodsFromPLCStatus) - def _ImportProjectFile(self): - dialog = wx.FileDialog(self.AppFrame, _("Choose a file"), os.getcwd(), "", _("All files|*.*"), wx.OPEN) - if dialog.ShowModal() == wx.ID_OK: - filepath = dialog.GetPath() - if os.path.isfile(filepath): - shutil.copy(filepath, self._getProjectFilesPath()) - else: - self.logger.write_error(_("No such file: %s\n") % filepath) - dialog.Destroy() - StatusMethods = [ {"bitmap" : "Build", "name" : _("Build"), @@ -1470,10 +1485,10 @@ "name" : _("Raw IEC code"), "tooltip" : _("Edit raw IEC code added to code generated by PLCGenerator"), "method" : "_editIECrawcode"}, - {"bitmap" : "ImportFile", - "name" : _("Import file"), - "tooltip" : _("Import into project a file to be transfered with PLC"), - "method" : "_ImportProjectFile"}, + {"bitmap" : "ManageFolder", + "name" : _("Project Files"), + "tooltip" : _("Open a file explorer to manage project files"), + "method" : "_OpenProjectFiles"}, ] diff -r cdc6393705ce -r 6f0e10085df9 c_ext/CFileEditor.py --- a/c_ext/CFileEditor.py Mon Jun 25 20:05:29 2012 +0200 +++ b/c_ext/CFileEditor.py Thu Jun 28 12:07:21 2012 +0200 @@ -4,10 +4,10 @@ import wx.grid import wx.stc as stc import wx.lib.buttons -from util.misc import opjimg from controls import CustomGrid, CustomTable from ConfTreeNodeEditor import ConfTreeNodeEditor +from utils.BitmapLibrary import GetBitmap if wx.Platform == '__WXMSW__': faces = { 'times': 'Times New Roman', @@ -790,10 +790,10 @@ for idx, (name, panel_class) in enumerate(CFILE_PARTS): button_id = wx.NewId() button = FoldPanelCaption(id=button_id, name='FoldPanelCaption_%s' % name, - label=name, bitmap=wx.Bitmap(opjimg("CollapsedIconData")), + label=name, bitmap=GetBitmap("CollapsedIconData"), parent=self.ConfNodeEditor, pos=wx.Point(0, 0), size=wx.Size(0, 20), style=wx.NO_BORDER|wx.ALIGN_LEFT) - button.SetBitmapSelected(wx.Bitmap(opjimg("ExpandedIconData"))) + button.SetBitmapSelected(GetBitmap("ExpandedIconData")) button.Bind(wx.EVT_BUTTON, self.GenPanelButtonCallback(name), id=button_id) self.MainSizer.AddWindow(button, 0, border=0, flag=wx.TOP|wx.GROW) diff -r cdc6393705ce -r 6f0e10085df9 canfestival/canfestival.py --- a/canfestival/canfestival.py Mon Jun 25 20:05:29 2012 +0200 +++ b/canfestival/canfestival.py Thu Jun 28 12:07:21 2012 +0200 @@ -255,8 +255,9 @@ def _ShowGeneratedMaster(self): self._OpenView("Generated master") - def _OpenView(self, name=None): + def _OpenView(self, name=None, onlyopened=False): if name == "Generated master": + app_frame = self.GetCTRoot().AppFrame if self._GeneratedMasterView is None: buildpath = self._getBuildPath() # Eventually create build dir @@ -269,16 +270,17 @@ self.GetCTRoot().logger.write_error(_("Error: No Master generated\n")) return - app_frame = self.GetCTRoot().AppFrame - manager = MiniNodeManager(self, masterpath, self.CTNFullName() + ".generated_master") self._GeneratedMasterView = MasterViewer(app_frame.TabsOpened, manager, app_frame) app_frame.EditProjectElement(self._GeneratedMasterView, name) + elif onlyopened: + app_frame.EditProjectElement(self._IECCodeView, name, onlyopened) + return self._GeneratedMasterView else: - ConfigTreeNode._OpenView(self) + ConfigTreeNode._OpenView(self, name, onlyopened) if self._View is not None: self._View.SetBusId(self.GetCurrentLocation()) return self._View diff -r cdc6393705ce -r 6f0e10085df9 images/FOLDER.png Binary file images/FOLDER.png has changed diff -r cdc6393705ce -r 6f0e10085df9 images/LeftCopy.png Binary file images/LeftCopy.png has changed diff -r cdc6393705ce -r 6f0e10085df9 images/ManageFolder.png Binary file images/ManageFolder.png has changed diff -r cdc6393705ce -r 6f0e10085df9 images/RightCopy.png Binary file images/RightCopy.png has changed diff -r cdc6393705ce -r 6f0e10085df9 images/icons.svg --- a/images/icons.svg Mon Jun 25 20:05:29 2012 +0200 +++ b/images/icons.svg Thu Jun 28 12:07:21 2012 +0200 @@ -43,9 +43,9 @@ pagecolor="#ffffff" id="base" showgrid="false" - inkscape:zoom="2.5600001" - inkscape:cx="643.14253" - inkscape:cy="778.71873" + inkscape:zoom="14.481548" + inkscape:cx="571.21601" + inkscape:cy="800.14078" inkscape:window-x="0" inkscape:window-y="24" inkscape:current-layer="svg2" @@ -85902,6 +85902,1777 @@ y1="198.01724" x2="131.52188" y2="41.586746" /> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + %% Build Clean editPLC HMIEditor ImportFile ImportDEF ImportSVG NetworkEdit ShowMaster ExportSlave Run ShowIECcode Stop Unknown %% + sodipodi:role="line">%% Build Clean editPLC HMIEditor ImportFile ManageFolder ImportDEF ImportSVG NetworkEdit ShowMaster ExportSlave Run ShowIECcode Stop Unknown %% @@ -86290,7 +88061,7 @@ inkscape:connector-curvature="0" /> @@ -86567,7 +88338,7 @@ + transform="translate(1922.9892,-430.1329)"> ST @@ -86777,7 +88548,7 @@ id="g1161" transform="matrix(0.5724346,-0.3079575,0.3079575,0.5724346,131.42904,887.47867)" /> %% Add Delete Disabled Enabled HideVars IECCDown IECCUp Maximize Minimize minus plus ShowVars %% + sodipodi:role="line">%% Add Delete Disabled Enabled HideVars IECCDown IECCUp Maximize Minimize minus plus ShowVars LeftCopy RightCopy%% @@ -89490,7 +91261,7 @@ + transform="translate(1727.0897,-400.03854)"> %% Extension Cfile Pyfile wxGlade SVGUI %% + sodipodi:role="line">%% Extension Cfile Pyfile wxGlade SVGUI FOLDER %% + transform="translate(1807.0897,-400.03854)"> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Tree icons + %% tree_folder tree_file %% + + + + + + + + + + + + + + + + + + + + + + + + + diff -r cdc6393705ce -r 6f0e10085df9 images/tree_file.png Binary file images/tree_file.png has changed diff -r cdc6393705ce -r 6f0e10085df9 images/tree_folder.png Binary file images/tree_folder.png has changed diff -r cdc6393705ce -r 6f0e10085df9 util/FileManagementPanel.py --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/util/FileManagementPanel.py Thu Jun 28 12:07:21 2012 +0200 @@ -0,0 +1,369 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +#This file is part of Beremiz, a Integrated Development Environment for +#programming IEC 61131-3 automates supporting plcopen standard and CanFestival. +# +#Copyright (C) 2007: Edouard TISSERANT and Laurent BESSARD +# +#See COPYING file for copyrights details. +# +#This library 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 2.1 of the License, or (at your option) any later version. +# +#This library 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 this library; if not, write to the Free Software +#Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + +import os +import shutil + +import wx + +from controls import EditorPanel +from utils.BitmapLibrary import GetBitmap + +FILTER = _("All files (*.*)|*.*|CSV files (*.csv)|*.csv") + +def sort_folder(x, y): + if x[1] == y[1]: + return cmp(x[0], y[0]) + elif x[1]: + return -1 + else: + return 1 + +def splitpath(path): + head, tail = os.path.split(path) + if head == "": + return [tail] + elif tail == "": + return splitpath(head) + return splitpath(head) + [tail] + +class FolderTree(wx.Panel): + + def __init__(self, parent, folder, filter, editable=True): + wx.Panel.__init__(self, parent, style=wx.TAB_TRAVERSAL) + + main_sizer = wx.BoxSizer(wx.VERTICAL) + + self.Tree = wx.TreeCtrl(self, + style=wx.TR_HAS_BUTTONS| + wx.TR_SINGLE| + wx.SUNKEN_BORDER| + wx.TR_HIDE_ROOT| + wx.TR_LINES_AT_ROOT| + wx.TR_EDIT_LABELS) + self.Bind(wx.EVT_TREE_ITEM_EXPANDED, self.OnTreeItemExpanded, self.Tree) + self.Bind(wx.EVT_TREE_ITEM_COLLAPSED, self.OnTreeItemCollapsed, self.Tree) + self.Bind(wx.EVT_TREE_BEGIN_LABEL_EDIT, self.OnTreeBeginLabelEdit, self.Tree) + self.Bind(wx.EVT_TREE_END_LABEL_EDIT, self.OnTreeEndLabelEdit, self.Tree) + main_sizer.AddWindow(self.Tree, 1, flag=wx.GROW) + + self.Filter = wx.ComboBox(self, style=wx.CB_READONLY) + self.Bind(wx.EVT_COMBOBOX, self.OnFilterChanged, self.Filter) + main_sizer.AddWindow(self.Filter, flag=wx.GROW) + + self.SetSizer(main_sizer) + + self.Folder = folder + self.Editable = editable + + self.TreeImageList = wx.ImageList(16, 16) + self.FOLDER_IMAGE = self.TreeImageList.Add(GetBitmap("tree_folder")) + self.FILE_IMAGE = self.TreeImageList.Add(GetBitmap("tree_file")) + self.Tree.SetImageList(self.TreeImageList) + + self.Filters = {} + filter_parts = filter.split("|") + for idx in xrange(0, len(filter_parts), 2): + if filter_parts[idx + 1] == "*.*": + self.Filters[filter_parts[idx]] = "" + else: + self.Filters[filter_parts[idx]] = filter_parts[idx + 1].replace("*", "") + self.Filter.Append(filter_parts[idx]) + if idx == 0: + self.Filter.SetStringSelection(filter_parts[idx]) + + self.CurrentFilter = self.Filters[self.Filter.GetStringSelection()] + + def _GetFolderChildren(self, folderpath, recursive=True): + items = [] + for filename in os.listdir(folderpath): + if not filename.startswith("."): + filepath = os.path.join(folderpath, filename) + if os.path.isdir(filepath): + if recursive: + children = len(self._GetFolderChildren(filepath, False)) + else: + children = 0 + items.append((filename, True, children)) + elif (self.CurrentFilter == "" or + os.path.splitext(filename)[1] == self.CurrentFilter): + items.append((filename, False, None)) + if recursive: + items.sort(sort_folder) + return items + + def GetTreeCtrl(self): + return self.Tree + + def RefreshTree(self): + root = self.Tree.GetRootItem() + if not root.IsOk(): + root = self.Tree.AddRoot("") + self.GenerateTreeBranch(root, self.Folder) + + def GenerateTreeBranch(self, root, folderpath): + item, item_cookie = self.Tree.GetFirstChild(root) + for idx, (filename, isdir, children) in enumerate(self._GetFolderChildren(folderpath)): + new = False + if not item.IsOk(): + item = self.Tree.AppendItem(root, filename) + if wx.Platform != '__WXMSW__': + item, item_cookie = self.Tree.GetNextChild(root, item_cookie) + new = True + elif self.Tree.GetItemText(item) != filename: + item = self.Tree.InsertItemBefore(root, idx, filename) + new = True + filepath = os.path.join(folderpath, filename) + if isdir: + if new: + self.Tree.SetItemImage(item, self.FOLDER_IMAGE) + if children > 0: + self.Tree.SetItemHasChildren(item) + elif self.Tree.IsExpanded(item): + self.GenerateTreeBranch(item, filepath) + elif new: + self.Tree.SetItemImage(item, self.FILE_IMAGE) + item, item_cookie = self.Tree.GetNextChild(root, item_cookie) + to_delete = [] + while item.IsOk(): + to_delete.append(item) + item, item_cookie = self.Tree.GetNextChild(root, item_cookie) + for item in to_delete: + self.Tree.Delete(item) + + def OnTreeItemExpanded(self, event): + item = event.GetItem() + self.GenerateTreeBranch(item, self.GetPath(item)) + event.Skip() + + def OnTreeItemCollapsed(self, event): + item = event.GetItem() + self.Tree.DeleteChildren(item) + self.Tree.SetItemHasChildren(item) + event.Skip() + + def OnTreeBeginLabelEdit(self, event): + item = event.GetItem() + if self.Editable and not self.Tree.ItemHasChildren(item): + event.Skip() + else: + event.Veto() + + def OnTreeEndLabelEdit(self, event): + event.Veto() + + def OnFilterChanged(self, event): + self.CurrentFilter = self.Filters[self.Filter.GetStringSelection()] + self.RefreshTree() + event.Skip() + + def _SelectItem(self, root, parts): + if len(parts) == 0: + self.Tree.SelectItem(root) + else: + item, item_cookie = self.Tree.GetFirstChild(root) + while item.IsOk(): + if self.Tree.GetItemText(item) == parts[0]: + if (self.Tree.ItemHasChildren(item) and + not self.Tree.IsExpanded(item)): + self.Tree.Expand(item) + wx.CallAfter(self._SelectItem, item, parts[1:]) + else: + self._SelectItem(item, parts[1:]) + return + item, item_cookie = self.Tree.GetNextChild(root, item_cookie) + + def SetPath(self, path): + if path.startswith(self.Folder): + root = self.Tree.GetRootItem() + if root.IsOk(): + relative_path = path.replace(os.path.join(self.Folder, ""), "") + self._SelectItem(root, splitpath(relative_path)) + + def GetPath(self, item=None): + if item is None: + item = self.Tree.GetSelection() + if item.IsOk(): + filepath = self.Tree.GetItemText(item) + parent = self.Tree.GetItemParent(item) + while parent.IsOk() and parent != self.Tree.GetRootItem(): + filepath = os.path.join(self.Tree.GetItemText(parent), filepath) + parent = self.Tree.GetItemParent(parent) + return os.path.join(self.Folder, filepath) + return self.Folder + +class FileManagementPanel(EditorPanel): + + def _init_Editor(self, parent): + self.Editor = wx.Panel(parent) + + main_sizer = wx.BoxSizer(wx.HORIZONTAL) + + left_sizer = wx.BoxSizer(wx.VERTICAL) + main_sizer.AddSizer(left_sizer, 1, border=5, flag=wx.GROW|wx.ALL) + + managed_dir_label = wx.StaticText(self.Editor, label=self.TagName + ":") + left_sizer.AddWindow(managed_dir_label, border=5, flag=wx.GROW|wx.BOTTOM) + + self.ManagedDir = FolderTree(self.Editor, self.Folder, FILTER) + left_sizer.AddWindow(self.ManagedDir, 1, flag=wx.GROW) + + managed_treectrl = self.ManagedDir.GetTreeCtrl() + self.Bind(wx.EVT_TREE_SEL_CHANGED, self.OnTreeItemChanged, managed_treectrl) + if self.EnableDragNDrop: + self.Bind(wx.EVT_TREE_BEGIN_DRAG, self.OnTreeBeginDrag, managed_treectrl) + + button_sizer = wx.BoxSizer(wx.VERTICAL) + main_sizer.AddSizer(button_sizer, border=5, + flag=wx.ALL|wx.ALIGN_CENTER_VERTICAL) + + for idx, (name, bitmap, help) in enumerate([ + ("DeleteButton", "remove_element", _("Remove file from left folder")), + ("LeftCopyButton", "LeftCopy", _("Copy file from right folder to left")), + ("RightCopyButton", "RightCopy", _("copy file from left folder to right"))]): + button = wx.lib.buttons.GenBitmapButton(self.Editor, + bitmap=GetBitmap(bitmap), + size=wx.Size(28, 28), style=wx.NO_BORDER) + button.SetToolTipString(help) + setattr(self, name, button) + if idx > 0: + flag = wx.TOP + else: + flag = 0 + self.Bind(wx.EVT_BUTTON, getattr(self, "On" + name), button) + button_sizer.AddWindow(button, border=20, flag=flag) + + right_sizer = wx.BoxSizer(wx.VERTICAL) + main_sizer.AddSizer(right_sizer, 1, border=5, flag=wx.GROW|wx.ALL) + + if wx.Platform == '__WXMSW__': + system_dir_label = wx.StaticText(self.Editor, label=_("My Computer:")) + else: + system_dir_label = wx.StaticText(self.Editor, label=_("Home Directory:")) + right_sizer.AddWindow(system_dir_label, border=5, flag=wx.GROW|wx.BOTTOM) + + self.SystemDir = FolderTree(self.Editor, self.HomeDirectory, FILTER, False) + right_sizer.AddWindow(self.SystemDir, 1, flag=wx.GROW) + + system_treectrl = self.SystemDir.GetTreeCtrl() + self.Bind(wx.EVT_TREE_SEL_CHANGED, self.OnTreeItemChanged, system_treectrl) + + self.Editor.SetSizer(main_sizer) + + def __init__(self, parent, controler, name, folder, enable_dragndrop=False): + self.Folder = os.path.realpath(folder) + self.EnableDragNDrop = enable_dragndrop + self.HomeDirectory = os.path.expanduser("~") + + EditorPanel.__init__(self, parent, name, None, None) + + self.Controler = controler + + self.SetIcon(GetBitmap("FOLDER")) + + def __del__(self): + self.Controler.OnCloseEditor(self) + + def GetTitle(self): + return self.TagName + + def RefreshView(self): + self.ManagedDir.RefreshTree() + self.SystemDir.RefreshTree() + self.RefreshButtonsState() + + def RefreshButtonsState(self): + managed_filepath = self.ManagedDir.GetPath() + system_filepath = self.SystemDir.GetPath() + + self.DeleteButton.Enable(os.path.isfile(managed_filepath)) + self.LeftCopyButton.Enable(os.path.isfile(system_filepath)) + self.RightCopyButton.Enable(os.path.isfile(managed_filepath)) + + def OnTreeItemChanged(self, event): + self.RefreshButtonsState() + event.Skip() + + def OnDeleteButton(self, event): + filepath = self.ManagedDir.GetPath() + if os.path.isfile(filepath): + folder, filename = os.path.split(filepath) + + dialog = wx.MessageDialog(self, + _("Do you really want to delete the file '%s'?") % filename, + _("Delete File"), wx.YES_NO|wx.ICON_QUESTION) + remove = dialog.ShowModal() == wx.ID_YES + dialog.Destroy() + + if remove: + os.remove(filepath) + self.ManagedDir.RefreshTree() + event.Skip() + + def CopyFile(self, src, dst): + if os.path.isfile(src): + src_folder, src_filename = os.path.split(src) + if os.path.isfile(dst): + dst_folder, dst_filename = os.path.split(dst) + else: + dst_folder = dst + + dst_filepath = os.path.join(dst_folder, src_filename) + if os.path.isfile(dst_filepath): + dialog = wx.MessageDialog(self, + _("The file '%s' already exist.\nDo you want to replace it?") % src_filename, + _("Replace File"), wx.YES_NO|wx.ICON_QUESTION) + copy = dialog.ShowModal() == wx.ID_YES + dialog.Destroy() + else: + copy = True + + if copy: + shutil.copyfile(src, dst_filepath) + return dst_filepath + return None + + def OnLeftCopyButton(self, event): + filepath = self.CopyFile(self.SystemDir.GetPath(), self.ManagedDir.GetPath()) + if filepath is not None: + self.ManagedDir.RefreshTree() + self.ManagedDir.SetPath(filepath) + event.Skip() + + def OnRightCopyButton(self, event): + filepath = self.CopyFile(self.ManagedDir.GetPath(), self.SystemDir.GetPath()) + if filepath is not None: + self.SystemDir.RefreshTree() + self.SystemDir.SetPath(filepath) + event.Skip() + + def OnTreeBeginDrag(self, event): + filepath = self.ManagedDir.GetPath() + if os.path.isfile(filepath): + relative_filepath = filepath.replace(os.path.join(self.Folder, ""), "") + data = wx.TextDataObject(str(("'%s'" % relative_filepath, "Constant"))) + dragSource = wx.DropSource(self) + dragSource.SetData(data) + dragSource.DoDragDrop() + \ No newline at end of file