forked from aziz/SublimeFileBrowser
-
Notifications
You must be signed in to change notification settings - Fork 0
/
jumping.py
237 lines (198 loc) · 8.98 KB
/
jumping.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
# coding: utf8
import sublime
from sublime import status_message, ok_cancel_dialog, load_settings, save_settings, Region
from sublime_plugin import TextCommand, WindowCommand
import os
import re
from os.path import dirname, realpath, join, isdir, basename, exists
ST3 = int(sublime.version()) >= 3000
if ST3:
from .common import DiredBaseCommand
from . import prompt
from .show import show
else:
from common import DiredBaseCommand
import prompt
from show import show
def load_jump_points():
return load_settings('dired.sublime-settings').get('dired_jump_points', {})
def save_jump_points(points, reverse=False):
if reverse:
points = dict((n, t) for t, n in points.items())
load_settings('dired.sublime-settings').set('dired_jump_points', points)
save_settings('dired.sublime-settings')
def jump_points():
sorted_jp = sorted(load_jump_points().items(), key=lambda x:x[0].lower())
return sorted_jp
def jump_targets():
return load_jump_points()
def jump_names():
return dict((t, n if ST3 else n.decode('utf8')) for n, t in load_jump_points().items())
class DiredJumpCommand(TextCommand, DiredBaseCommand):
def run(self, edit, new_window=False):
if not jump_points():
status_message("No jump points available. To create jump point for this directory use 'P'.")
return
# show_quick_panel didn't work with dict_items
self.new_window = new_window
self.jump_points = [[n, t] for n, t in jump_points()]
self.display_jump_points = [[n if ST3 else n.decode('utf8'), self.display_path(t)] for n, t in jump_points()]
self.view.window().show_quick_panel(self.display_jump_points, self.on_pick_point, sublime.MONOSPACE_FONT)
def display_path(self, folder):
display = folder
home = os.path.expanduser("~")
if folder.startswith(home):
display = folder.replace(home, "~", 1)
return display
def on_pick_point(self, index):
if index == -1:
return
name, target = self.jump_points[index]
if exists(target) and isdir(target) and target[-1] == os.sep:
if self.new_window:
print(target)
self.view.run_command("dired_open_in_new_window", { "project_folder": [target] })
else:
show(self.view.window(), target, view_id=self.view.id())
status_message(u"Jumping to point '{0}' complete".format(name if ST3 else name.decode('utf8')))
else:
# workaround ST3 bug https://github.com/SublimeText/Issues/issues/39
self.view.window().run_command('hide_overlay')
msg = u"Can't jump to '{0} → {1}'.\n\nRemove that jump point?".format(name, target)
if ok_cancel_dialog(msg):
points = load_jump_points()
del points[name]
save_jump_points(points)
status_message(u"Jump point '{0}' was removed".format(name))
self.view.run_command('dired_refresh')
class DiredEditJumpPointCommand(TextCommand, DiredBaseCommand):
def run(self, edit, item=False):
self.names = jump_names()
self.project_path = item and item[1] or self.path
self.item = item
name = item and item[0] or self.names.get(self.project_path)
if not name:
prompt = 'Create jump point:'
name = basename(self.project_path)
else:
prompt = 'Edit jump point (clear to Remove):'
self.view.window().show_input_panel(prompt, name, self.edit_jump_point, None, None)
def edit_jump_point(self, name):
if name: # edit or create jump point
if ST3:
iterable = list(self.names.items())
else:
iterable = self.names.items()
for t, n in iterable:
if n == name:
msg = u"The jump point with name '{0}' is already exists ({1})\n\n".format(n, t)
msg += "Do you want to overwrite it?"
if ok_cancel_dialog(msg):
del self.names[t]
else:
return
self.names[self.project_path] = name
status_message(u"Jump point for this directory was set to '{0}'".format(name))
elif self.project_path in self.names:
del self.names[self.project_path]
status_message("Jump point for this directory was removed")
else:
status_message("Jump point wasn't created")
return
save_jump_points(self.names, reverse=True)
if self.item:
self.view.run_command('dired_jump_list', {"reuse": True})
else:
self.view.run_command('dired_refresh')
class DiredJumpListRenderCommand(TextCommand):
def run(self, edit):
self.view.erase(edit, Region(0, self.view.size()))
self.view.insert(edit, 0, self.render())
self.view.sel().clear()
pt = self.view.text_point(3, 0)
self.view.sel().clear()
self.view.sel().add(Region(pt, pt))
self.view.set_read_only(True)
def render(self):
self.view_width = 79
self.col_padding = 2
self.jump_points = [[n, t] for n, t in jump_points()]
self.names = [n for n, t in jump_points()]
content = u"Jump to…\n" + u"—" * self.view_width + u"\n\n"
if len(self.names) > 0:
self.max_len_names = max([len(n if ST3 else n.decode('utf8')) for n, _ in jump_points()])
self.view.settings().set('dired_project_count', len(self.names))
else:
content += "Jump list is empty!\n\nAdd a folder to your jump list by pressing P (shift + p)\nwhile you are browsing that folder in FileBrowser"
for p in self.jump_points:
content += u'★ {0}→{1}\n'.format(self.display_name(p[0]), self.display_path(p[1]))
return content
def display_name(self, name):
name = name if ST3 else name.decode('utf8')
return name + " " * (self.max_len_names - len(name) + self.col_padding)
def display_path(self, folder):
display = folder
home = os.path.expanduser("~")
label_characters = self.view_width - 4 - (self.col_padding*2) - self.max_len_names
if folder.startswith(home):
display = folder.replace(home, "~", 1)
if folder.endswith("/") or folder.endswith("\\"):
display = display[:-1]
if len(display) > label_characters:
chars = int(label_characters/2)
display = display[:chars] + u"…" + display[-chars:]
return " " * self.col_padding + display
class DiredJumpListCommand(TextCommand):
def run(self, edit, reuse=False):
if reuse:
view = self.view
view.set_read_only(False)
else:
view = self.view.window().new_file()
view.set_name("FileBrowser: Jump List")
view.set_scratch(True)
view.set_syntax_file('Packages/FileBrowser/dired_jumplist.hidden-tmLanguage')
view.settings().set('color_scheme','Packages/FileBrowser/dired.hidden-tmTheme')
view.settings().set('line_numbers', False)
view.settings().set('draw_centered', True)
view.run_command('dired_jump_list_render')
sublime.active_window().focus_view(view)
class DiredProjectNextLineCommand(TextCommand):
def run(self, edit, forward=None):
assert forward in (True, False), 'forward must be set to True or False'
count = self.view.settings().get('dired_project_count', 0)
files = Region(self.view.text_point(3, 0), self.view.text_point(count+3-1, 0))
if files.empty():
return
pt = self.view.sel()[0].a
if files.contains(pt):
# Try moving by one line.
line = self.view.line(pt)
pt = forward and (line.b + 1) or (line.a - 1)
if not files.contains(pt):
# Not (or no longer) in the list of files, so move to the closest edge.
pt = (pt > files.b) and files.b or files.a
# print(pt)
line = self.view.line(pt)
self.view.sel().clear()
self.view.sel().add(Region(line.a, line.a))
class DiredProjectSelectCommand(TextCommand):
def run(self, edit):
pt = self.view.sel()[0].a
row, col = self.view.rowcol(pt)
points = [[n, t] for n, t in jump_points()]
current_project = [points[row - 3][1]]
self.view.run_command("dired_open_in_new_window", { "project_folder": current_project})
def close_view(view):
if ST3:
view.close()
else:
view.window().run_command("close_file")
sublime.set_timeout(close_view(self.view), 100)
class DiredProjectEditJumpPointCommand(TextCommand):
def run(self, edit):
pt = self.view.sel()[0].a
row, col = self.view.rowcol(pt)
points = [[n, t] for n, t in jump_points()]
current_project = points[row - 3]
self.view.run_command("dired_edit_jump_point", { "item": current_project})