Show More
@@ -22,28 +22,50 b' from IPython.external.qt import QtGui,QtCore' | |||||
22 | from IPython.core.magic import magic_escapes |
|
22 | from IPython.core.magic import magic_escapes | |
23 |
|
23 | |||
24 | class MagicHelper(QtGui.QDockWidget): |
|
24 | class MagicHelper(QtGui.QDockWidget): | |
|
25 | """MagicHelper - dockable widget for convenient search and running of | |||
|
26 | magic command for IPython QtConsole. | |||
|
27 | """ | |||
|
28 | ||||
|
29 | #--------------------------------------------------------------------------- | |||
|
30 | # signals | |||
|
31 | #--------------------------------------------------------------------------- | |||
25 |
|
32 | |||
26 | pasteRequested = QtCore.pyqtSignal(str, name = 'pasteRequested') |
|
33 | pasteRequested = QtCore.pyqtSignal(str, name = 'pasteRequested') | |
|
34 | """This signal is emitted when user wants to paste selected magic | |||
|
35 | command into the command line. | |||
|
36 | """ | |||
|
37 | ||||
27 | runRequested = QtCore.pyqtSignal(str, name = 'runRequested') |
|
38 | runRequested = QtCore.pyqtSignal(str, name = 'runRequested') | |
|
39 | """This signal is emitted when user wants to execute selected magic command | |||
|
40 | """ | |||
|
41 | ||||
28 | readyForUpdate = QtCore.pyqtSignal(name = 'readyForUpdate') |
|
42 | readyForUpdate = QtCore.pyqtSignal(name = 'readyForUpdate') | |
|
43 | """This signal is emitted when MagicHelper is ready to be populated. | |||
|
44 | Since kernel querying mechanisms are out of scope of this class, | |||
|
45 | it expects its owner to invoke MagicHelper.populate_magic_helper() | |||
|
46 | as a reaction on this event. | |||
|
47 | """ | |||
29 |
|
48 | |||
30 | #--------------------------------------------------------------------------- |
|
49 | #--------------------------------------------------------------------------- | |
31 | # 'object' interface |
|
50 | # constructor | |
32 | #--------------------------------------------------------------------------- |
|
51 | #--------------------------------------------------------------------------- | |
33 |
|
52 | |||
34 | def __init__(self, name, parent): |
|
53 | def __init__(self, name, parent): | |
35 |
|
||||
36 | super(MagicHelper, self).__init__(name, parent) |
|
54 | super(MagicHelper, self).__init__(name, parent) | |
37 |
|
55 | |||
38 | self.data = None |
|
56 | self.data = None | |
39 |
|
57 | |||
40 | class MinListWidget(QtGui.QListWidget): |
|
58 | class MinListWidget(QtGui.QListWidget): | |
|
59 | """Temp class to overide the default QListWidget size hint | |||
|
60 | in order to make MagicHelper narrow | |||
|
61 | """ | |||
41 | def sizeHint(self): |
|
62 | def sizeHint(self): | |
42 | s = QtCore.QSize() |
|
63 | s = QtCore.QSize() | |
43 | s.setHeight(super(MinListWidget,self).sizeHint().height()) |
|
64 | s.setHeight(super(MinListWidget,self).sizeHint().height()) | |
44 | s.setWidth(self.sizeHintForColumn(0)) |
|
65 | s.setWidth(self.sizeHintForColumn(0)) | |
45 | return s |
|
66 | return s | |
46 |
|
67 | |||
|
68 | # construct content | |||
47 | self.frame = QtGui.QFrame() |
|
69 | self.frame = QtGui.QFrame() | |
48 | self.search_label = QtGui.QLabel("Search:") |
|
70 | self.search_label = QtGui.QLabel("Search:") | |
49 | self.search_line = QtGui.QLineEdit() |
|
71 | self.search_line = QtGui.QLineEdit() | |
@@ -52,6 +74,7 b' class MagicHelper(QtGui.QDockWidget):' | |||||
52 | self.paste_button = QtGui.QPushButton("Paste") |
|
74 | self.paste_button = QtGui.QPushButton("Paste") | |
53 | self.run_button = QtGui.QPushButton("Run") |
|
75 | self.run_button = QtGui.QPushButton("Run") | |
54 |
|
76 | |||
|
77 | # layout all the widgets | |||
55 | main_layout = QtGui.QVBoxLayout() |
|
78 | main_layout = QtGui.QVBoxLayout() | |
56 | search_layout = QtGui.QHBoxLayout() |
|
79 | search_layout = QtGui.QHBoxLayout() | |
57 | search_layout.addWidget(self.search_label) |
|
80 | search_layout.addWidget(self.search_label) | |
@@ -67,7 +90,8 b' class MagicHelper(QtGui.QDockWidget):' | |||||
67 | self.frame.setLayout(main_layout) |
|
90 | self.frame.setLayout(main_layout) | |
68 | self.setWidget(self.frame) |
|
91 | self.setWidget(self.frame) | |
69 |
|
92 | |||
70 | self.visibilityChanged[bool].connect( self.update_magic_helper ) |
|
93 | # connect all the relevant signals to handlers | |
|
94 | self.visibilityChanged[bool].connect( self._update_magic_helper ) | |||
71 | self.search_class.activated[int].connect( |
|
95 | self.search_class.activated[int].connect( | |
72 | self.class_selected |
|
96 | self.class_selected | |
73 | ) |
|
97 | ) | |
@@ -84,15 +108,29 b' class MagicHelper(QtGui.QDockWidget):' | |||||
84 | self.run_requested |
|
108 | self.run_requested | |
85 | ) |
|
109 | ) | |
86 |
|
110 | |||
87 | def update_magic_helper(self, visible): |
|
111 | #--------------------------------------------------------------------------- | |
|
112 | # implementation | |||
|
113 | #--------------------------------------------------------------------------- | |||
|
114 | ||||
|
115 | def _update_magic_helper(self, visible): | |||
|
116 | """Start update sequence. | |||
|
117 | This method is called when MagicHelper becomes visible. It clears | |||
|
118 | the content and emits readyForUpdate signal. The owner of the | |||
|
119 | instance is expected to invoke populate_magic_helper() when magic | |||
|
120 | info is available. | |||
|
121 | """ | |||
88 | if not visible or self.data != None: |
|
122 | if not visible or self.data != None: | |
89 | return |
|
123 | return | |
90 | self.data = {} |
|
124 | self.data = {} | |
91 | self.search_class.clear() |
|
125 | self.search_class.clear() | |
92 | self.search_class.addItem("Populating...") |
|
126 | self.search_class.addItem("Populating...") | |
|
127 | self.search_list.clear() | |||
93 | self.readyForUpdate.emit() |
|
128 | self.readyForUpdate.emit() | |
94 |
|
129 | |||
95 | def populate_magic_helper(self, data): |
|
130 | def populate_magic_helper(self, data): | |
|
131 | """Expects data returned by lsmagics query from kernel. | |||
|
132 | Populates the search_class and search_list with relevant items. | |||
|
133 | """ | |||
96 | self.search_class.clear() |
|
134 | self.search_class.clear() | |
97 | self.search_list.clear() |
|
135 | self.search_list.clear() | |
98 |
|
136 | |||
@@ -115,17 +153,24 b' class MagicHelper(QtGui.QDockWidget):' | |||||
115 | self.filter_magic_helper('.', 'any') |
|
153 | self.filter_magic_helper('.', 'any') | |
116 |
|
154 | |||
117 | def class_selected(self, index): |
|
155 | def class_selected(self, index): | |
|
156 | """Handle search_class selection changes | |||
|
157 | """ | |||
118 | item = self.search_class.itemData(index) |
|
158 | item = self.search_class.itemData(index) | |
119 | regex = self.search_line.text() |
|
159 | regex = self.search_line.text() | |
120 | self.filter_magic_helper(regex = regex, cls = item) |
|
160 | self.filter_magic_helper(regex = regex, cls = item) | |
121 |
|
161 | |||
122 | def search_changed(self, search_string): |
|
162 | def search_changed(self, search_string): | |
|
163 | """Handle search_line text changes. | |||
|
164 | The text is interpreted as a regular expression | |||
|
165 | """ | |||
123 | item = self.search_class.itemData( |
|
166 | item = self.search_class.itemData( | |
124 | self.search_class.currentIndex() |
|
167 | self.search_class.currentIndex() | |
125 | ) |
|
168 | ) | |
126 | self.filter_magic_helper(regex = search_string, cls = item) |
|
169 | self.filter_magic_helper(regex = search_string, cls = item) | |
127 |
|
170 | |||
128 | def _get_current_search_item(self, item = None): |
|
171 | def _get_current_search_item(self, item = None): | |
|
172 | """Retrieve magic command currently selected in the search_list | |||
|
173 | """ | |||
129 | text = None |
|
174 | text = None | |
130 | if not isinstance(item, QtGui.QListWidgetItem): |
|
175 | if not isinstance(item, QtGui.QListWidgetItem): | |
131 | item = self.search_list.currentItem() |
|
176 | item = self.search_list.currentItem() | |
@@ -133,16 +178,24 b' class MagicHelper(QtGui.QDockWidget):' | |||||
133 | return text |
|
178 | return text | |
134 |
|
179 | |||
135 | def paste_requested(self, item = None): |
|
180 | def paste_requested(self, item = None): | |
|
181 | """Emit pasteRequested signal with currently selected item text | |||
|
182 | """ | |||
136 | text = self._get_current_search_item(item) |
|
183 | text = self._get_current_search_item(item) | |
137 | if text != None: |
|
184 | if text != None: | |
138 | self.pasteRequested.emit(text) |
|
185 | self.pasteRequested.emit(text) | |
139 |
|
186 | |||
140 | def run_requested(self, item = None): |
|
187 | def run_requested(self, item = None): | |
|
188 | """Emit runRequested signal with currently selected item text | |||
|
189 | """ | |||
141 | text = self._get_current_search_item(item) |
|
190 | text = self._get_current_search_item(item) | |
142 | if text != None: |
|
191 | if text != None: | |
143 | self.runRequested.emit(text) |
|
192 | self.runRequested.emit(text) | |
144 |
|
193 | |||
145 | def filter_magic_helper(self, regex, cls): |
|
194 | def filter_magic_helper(self, regex, cls): | |
|
195 | """Update search_list with magic commands whose text match | |||
|
196 | regex and class match cls. | |||
|
197 | If cls equals 'any' - any class matches. | |||
|
198 | """ | |||
146 | if regex == "" or regex == None: |
|
199 | if regex == "" or regex == None: | |
147 | regex = '.' |
|
200 | regex = '.' | |
148 | if cls == None: |
|
201 | if cls == None: |
General Comments 0
You need to be logged in to leave comments.
Login now