browse-everything 0.4.5 → 0.5.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +8 -8
- data/HISTORY.md +6 -0
- data/README.md +3 -1
- data/app/assets/javascripts/browse_everything.js +2 -1
- data/app/assets/javascripts/browse_everything/behavior.js.coffee +65 -5
- data/app/assets/stylesheets/browse_everything.css.scss +28 -4
- data/app/assets/stylesheets/jquery.treetable.theme.browse.css +91 -0
- data/app/controllers/browse_everything_controller.rb +2 -2
- data/app/helpers/browse_everything_helper.rb +6 -0
- data/app/views/browse_everything/_file.html.erb +18 -0
- data/app/views/browse_everything/_files.html.erb +14 -8
- data/app/views/browse_everything/index.html.erb +1 -0
- data/app/views/layouts/browse_everything.html.erb +1 -1
- data/browse-everything.gemspec +2 -1
- data/config/routes.rb +1 -1
- data/lib/browse_everything/browser.rb +2 -2
- data/lib/browse_everything/driver/base.rb +4 -4
- data/lib/browse_everything/driver/drop_box.rb +2 -2
- data/lib/browse_everything/engine.rb +3 -1
- data/lib/browse_everything/file_entry.rb +5 -1
- data/lib/browse_everything/version.rb +1 -1
- data/spec/spec_helper.rb +5 -2
- data/spec/unit/browser_spec.rb +2 -2
- data/spec/unit/drop_box_spec.rb +1 -1
- data/spec/unit/file_entry_spec.rb +78 -30
- data/vendor/assets/javascripts/jquery.treetable.js +629 -0
- data/vendor/assets/stylesheets/jquery.treetable.css +28 -0
- data/vendor/assets/stylesheets/screen.css +28 -0
- metadata +23 -4
checksums.yaml
CHANGED
@@ -1,15 +1,15 @@
|
|
1
1
|
---
|
2
2
|
!binary "U0hBMQ==":
|
3
3
|
metadata.gz: !binary |-
|
4
|
-
|
4
|
+
NWJlNzAxY2ZjZTYxOGJkMGI3ZDYzNzkyMzI3YTEyYzA2ZWYyZGZhYg==
|
5
5
|
data.tar.gz: !binary |-
|
6
|
-
|
6
|
+
NDExYjU3YTQyOGMyZTI4ZWY3YzEwZTNmZmQyMmQ2ODU4NWFiYzUyZQ==
|
7
7
|
SHA512:
|
8
8
|
metadata.gz: !binary |-
|
9
|
-
|
10
|
-
|
11
|
-
|
9
|
+
MDhkNTgxMTlhYjQzMTAyZWI4Y2E3MTdiZTkzODdkNGNkMGUwOTc3YzM4Y2Ji
|
10
|
+
NjMwOGM0M2U4MjBiNzZjZmI1YzlkZGQ5ZDQyNWZiMzgwYjRmYThiZTg4MmVh
|
11
|
+
YjNiZTg0YzhmMzllNTYyYWZiYzQzMzM5OWUzOTZlMzUwM2IyMDA=
|
12
12
|
data.tar.gz: !binary |-
|
13
|
-
|
14
|
-
|
15
|
-
|
13
|
+
YmFlOWJmNmIzZjg3NDUxNjA5NDRlM2FmOGJkNGYxMWI2NzdlZDlhNmM4ZDU0
|
14
|
+
ZjJhOTE1M2E3YzgyZWEzNDJmOTMwMjVjNzA2NDcyNDZhM2I2NzNiYjZhNDc4
|
15
|
+
NDY3NGFmODY3ZDg1YWIxNjZhYmMxNmYwZWY1ODYxODY1NTRlNmM=
|
data/HISTORY.md
CHANGED
@@ -1,3 +1,9 @@
|
|
1
|
+
### 0.5.0 (2014-07-29)
|
2
|
+
- New, prettier tree-oriented UI
|
3
|
+
- Added app-specific `context` parameter
|
4
|
+
- Added `accept` parameter to allow filtering of results based on MIME type
|
5
|
+
- Added `show()` callback
|
6
|
+
|
1
7
|
### 0.4.5 (2014-06-20)
|
2
8
|
- Fix for filenames with special entities in them
|
3
9
|
|
data/README.md
CHANGED
@@ -56,10 +56,12 @@ browse-everything can be triggered in two ways -- either via data attributes in
|
|
56
56
|
#### Options
|
57
57
|
|
58
58
|
|
59
|
-
| Name |
|
59
|
+
| Name | Type | Default | Description |
|
60
60
|
|-----------------|-----------------|-----------------|----------------------------------------------------------------|
|
61
61
|
| route | path (required) | '' | The base route of the browse-everything engine. |
|
62
62
|
| target | xpath or jQuery | null | A form object to add the results to as hidden fields. |
|
63
|
+
| context | text | null | App-specific context information (passed with each request) |
|
64
|
+
| accept | MIME mask | */* | A list of acceptable MIME types to browse (e.g., 'video/*') |
|
63
65
|
|
64
66
|
If a `target` is provided, browse-everything will automatically convert the JSON response to a series of hidden form fields
|
65
67
|
that can be posted back to Rails to re-create the array on the server side.
|
@@ -8,10 +8,12 @@ $ ->
|
|
8
8
|
ctx =
|
9
9
|
opts: $.extend(true, {}, options)
|
10
10
|
callbacks:
|
11
|
+
show: $.Callbacks()
|
11
12
|
done: $.Callbacks()
|
12
13
|
cancel: $.Callbacks()
|
13
14
|
fail: $.Callbacks()
|
14
15
|
ctx.callback_proxy =
|
16
|
+
show: (func) -> ctx.callbacks.show.add(func) ; return this
|
15
17
|
done: (func) -> ctx.callbacks.done.add(func) ; return this
|
16
18
|
cancel: (func) -> ctx.callbacks.cancel.add(func) ; return this
|
17
19
|
fail: (func) -> ctx.callbacks.fail.add(func) ; return this
|
@@ -28,13 +30,55 @@ $ ->
|
|
28
30
|
.val(decodeURIComponent(this[1]))[0].outerHTML
|
29
31
|
$(elements.toArray().join("\n"))
|
30
32
|
|
33
|
+
indicateSelected = () ->
|
34
|
+
$('input.ev-url').each () ->
|
35
|
+
$("*[data-ev-location='#{$(this).val()}']").addClass('ev-selected')
|
36
|
+
|
37
|
+
tableSetup = (table) ->
|
38
|
+
table.treetable
|
39
|
+
expandable: true
|
40
|
+
onNodeCollapse: ->
|
41
|
+
node = this;
|
42
|
+
table.treetable("unloadBranch", node)
|
43
|
+
onNodeExpand: ->
|
44
|
+
node = this
|
45
|
+
$('body').css('cursor','wait')
|
46
|
+
$.ajax
|
47
|
+
async: false # Must be false, otherwise loadBranch happens after showChildren?
|
48
|
+
url: $('a.ev-link',node.row).attr('href')
|
49
|
+
data:
|
50
|
+
accept: dialog.data('context').opts.accept
|
51
|
+
context: dialog.data('context').opts.context
|
52
|
+
.done (html) ->
|
53
|
+
rows = $('tbody tr',$(html))
|
54
|
+
table.treetable("loadBranch", node, rows)
|
55
|
+
sizeColumns(table)
|
56
|
+
indicateSelected()
|
57
|
+
.always ->
|
58
|
+
$('body').css('cursor','default')
|
59
|
+
sizeColumns(table)
|
60
|
+
|
61
|
+
sizeColumns = (table) ->
|
62
|
+
full_width = $('.ev-files').width()
|
63
|
+
table.width(full_width)
|
64
|
+
set_size = (selector, pct) ->
|
65
|
+
$(selector, table).width(full_width * pct).css('width',full_width * pct).css('max-width',full_width * pct)
|
66
|
+
set_size '.ev-file', 0.4
|
67
|
+
set_size '.ev-size', 0.1
|
68
|
+
set_size '.ev-kind', 0.3
|
69
|
+
set_size '.ev-date', 0.2
|
70
|
+
|
71
|
+
$(window).on('resize', -> sizeColumns($('table#file-list')))
|
72
|
+
|
31
73
|
$.fn.browseEverything = (options) ->
|
32
74
|
ctx = $(this).data('context')
|
33
75
|
if options?
|
34
76
|
ctx = initialize(this[0], options)
|
35
77
|
$(this).click () ->
|
36
78
|
dialog.data('context',ctx)
|
37
|
-
dialog.load ctx.opts.route, () ->
|
79
|
+
dialog.load ctx.opts.route, () ->
|
80
|
+
ctx.callbacks.show.fire()
|
81
|
+
dialog.modal('show')
|
38
82
|
ctx.callback_proxy
|
39
83
|
|
40
84
|
$(document).on 'click', 'button.ev-cancel', (event) ->
|
@@ -49,6 +93,7 @@ $ ->
|
|
49
93
|
main_form = $(this).closest('form')
|
50
94
|
resolver_url = main_form.data('resolver')
|
51
95
|
ctx = dialog.data('context')
|
96
|
+
$(main_form).find('input[name=context]').val(ctx.opts.context)
|
52
97
|
$.ajax resolver_url,
|
53
98
|
type: 'POST'
|
54
99
|
dataType: 'json'
|
@@ -64,14 +109,29 @@ $ ->
|
|
64
109
|
$('body').css('cursor','default')
|
65
110
|
$('.ev-browser').modal('hide')
|
66
111
|
|
67
|
-
$(document).on 'click', '.ev-
|
112
|
+
$(document).on 'click', '.ev-files table tr', (event) ->
|
113
|
+
$('a.ev-link',this).click() unless event.target.nodeName == 'A'
|
114
|
+
|
115
|
+
$(document).on 'click', '.ev-files .ev-container a.ev-link', (event) ->
|
116
|
+
event.stopPropagation()
|
117
|
+
event.preventDefault()
|
118
|
+
row = $(this).closest('tr')
|
119
|
+
action = if row.hasClass('expanded') then 'collapseNode' else 'expandNode'
|
120
|
+
node_id = $(this).attr('href')
|
121
|
+
$('table#file-list').treetable(action,node_id)
|
122
|
+
|
123
|
+
$(document).on 'click', '.ev-providers .ev-container a', (event) ->
|
68
124
|
event.preventDefault()
|
69
125
|
$('body').css('cursor','wait')
|
70
|
-
$.ajax
|
126
|
+
$.ajax
|
127
|
+
url: $(this).attr('href'),
|
128
|
+
data:
|
129
|
+
accept: dialog.data('context').opts.accept
|
130
|
+
context: dialog.data('context').opts.context
|
71
131
|
.done (data) ->
|
72
132
|
$('.ev-files').html(data)
|
73
|
-
|
74
|
-
|
133
|
+
indicateSelected();
|
134
|
+
tableSetup($('table#file-list'))
|
75
135
|
.fail (xhr,status,error) ->
|
76
136
|
$('.ev-files').html(xhr.responseText)
|
77
137
|
.always ->
|
@@ -3,6 +3,8 @@
|
|
3
3
|
@import "bootstrap";
|
4
4
|
@import "bootstrap-responsive";
|
5
5
|
@import "font-awesome";
|
6
|
+
@import "jquery.treetable";
|
7
|
+
@import "jquery.treetable.theme.browse";
|
6
8
|
|
7
9
|
@mixin border-highlight($color) {
|
8
10
|
$bl: 3;
|
@@ -50,7 +52,7 @@
|
|
50
52
|
|
51
53
|
.ev-files {
|
52
54
|
position: relative;
|
53
|
-
overflow-x:
|
55
|
+
overflow-x: auto;
|
54
56
|
overflow-y: auto;
|
55
57
|
|
56
58
|
li {
|
@@ -84,12 +86,34 @@
|
|
84
86
|
background-color: lightgray;
|
85
87
|
}
|
86
88
|
|
89
|
+
.ev-files {
|
90
|
+
.ev-selected .indenter {
|
91
|
+
@extend .fa;
|
92
|
+
@extend .fa-check;
|
93
|
+
@extend .icon-check;
|
94
|
+
}
|
95
|
+
table {
|
96
|
+
width: 100%;
|
97
|
+
max-width: 100%;
|
98
|
+
tr {
|
99
|
+
width: 100%;
|
100
|
+
max-width: 100%;
|
101
|
+
td {
|
102
|
+
white-space: nowrap;
|
103
|
+
overflow: hidden;
|
104
|
+
text-overflow: ellipsis;
|
105
|
+
}
|
106
|
+
}
|
107
|
+
}
|
108
|
+
}
|
109
|
+
|
110
|
+
.ev-providers .ev-selected {
|
111
|
+
@include border-highlight(black);
|
112
|
+
}
|
113
|
+
|
87
114
|
.ev-providers, .ev-files {
|
88
115
|
height: 50vh;
|
89
116
|
li { @include ev-link; }
|
90
|
-
.ev-selected {
|
91
|
-
@include border-highlight(black);
|
92
|
-
}
|
93
117
|
}
|
94
118
|
|
95
119
|
}
|
@@ -0,0 +1,91 @@
|
|
1
|
+
table.treetable {
|
2
|
+
border: 1px none #888;
|
3
|
+
border-collapse: collapse;
|
4
|
+
font-size: 1em;
|
5
|
+
line-height: 1;
|
6
|
+
margin: .6em 0 1.8em 0;
|
7
|
+
width: 100%;
|
8
|
+
}
|
9
|
+
|
10
|
+
table.treetable caption {
|
11
|
+
font-size: .9em;
|
12
|
+
font-weight: bold;
|
13
|
+
margin-bottom: .2em;
|
14
|
+
}
|
15
|
+
|
16
|
+
table.treetable thead {
|
17
|
+
background: #aaa url() repeat-x top left;
|
18
|
+
font-size: 1em;
|
19
|
+
}
|
20
|
+
|
21
|
+
table.treetable thead tr th {
|
22
|
+
border: 1px none #888;
|
23
|
+
font-weight: bold;
|
24
|
+
padding: .3em 0.1em .1em 0.1em;
|
25
|
+
text-align: left;
|
26
|
+
}
|
27
|
+
|
28
|
+
table.treetable tbody tr td {
|
29
|
+
cursor: default;
|
30
|
+
padding: .3em 0.1em;
|
31
|
+
white-space: nowrap;
|
32
|
+
overflow: ellipsis;
|
33
|
+
}
|
34
|
+
|
35
|
+
table.treetable span {
|
36
|
+
background-position: center left;
|
37
|
+
background-repeat: no-repeat;
|
38
|
+
padding: .2em 0 .2em 1.5em;
|
39
|
+
}
|
40
|
+
|
41
|
+
table.treetable span.file {
|
42
|
+
background-image: url();
|
43
|
+
}
|
44
|
+
|
45
|
+
table.treetable span.folder {
|
46
|
+
background-image: url();
|
47
|
+
}
|
48
|
+
|
49
|
+
table.treetable tr.collapsed span.indenter a {
|
50
|
+
background-image: url();
|
51
|
+
}
|
52
|
+
|
53
|
+
table.treetable tr.expanded span.indenter a {
|
54
|
+
background-image: url();
|
55
|
+
}
|
56
|
+
|
57
|
+
/*
|
58
|
+
table.treetable tr.branch {
|
59
|
+
background-color: #f9f9f9;
|
60
|
+
}
|
61
|
+
*/
|
62
|
+
|
63
|
+
table.treetable tr.selected {
|
64
|
+
background-color: #3875d7;
|
65
|
+
color: #fff;
|
66
|
+
}
|
67
|
+
|
68
|
+
table.treetable tr span.indenter a {
|
69
|
+
outline: none; /* Expander shows outline after upgrading to 3.0 (#141) */
|
70
|
+
}
|
71
|
+
|
72
|
+
table.treetable tr.collapsed.selected span.indenter a {
|
73
|
+
background-image: url();
|
74
|
+
}
|
75
|
+
|
76
|
+
table.treetable tr.expanded.selected span.indenter a {
|
77
|
+
background-image: url();
|
78
|
+
}
|
79
|
+
|
80
|
+
table.treetable tr.accept {
|
81
|
+
background-color: #a3bce4;
|
82
|
+
color: #fff
|
83
|
+
}
|
84
|
+
|
85
|
+
table.treetable tr.collapsed.accept td span.indenter a {
|
86
|
+
background-image: url();
|
87
|
+
}
|
88
|
+
|
89
|
+
table.treetable tr.expanded.accept td span.indenter a {
|
90
|
+
background-image: url();
|
91
|
+
}
|
@@ -65,7 +65,7 @@ class BrowseEverythingController < ActionController::Base
|
|
65
65
|
end
|
66
66
|
|
67
67
|
def provider_name
|
68
|
-
@
|
68
|
+
@provider_name ||= params[:provider] || params[:state].to_s.split(/\|/).last
|
69
69
|
end
|
70
70
|
|
71
71
|
helper_method :auth_link
|
@@ -73,4 +73,4 @@ class BrowseEverythingController < ActionController::Base
|
|
73
73
|
helper_method :browse_path
|
74
74
|
helper_method :provider
|
75
75
|
helper_method :provider_name
|
76
|
-
end
|
76
|
+
end
|
@@ -8,4 +8,10 @@ module BrowseEverythingHelper
|
|
8
8
|
fields.join("\n").html_safe
|
9
9
|
end
|
10
10
|
|
11
|
+
def is_acceptable?(file)
|
12
|
+
acceptable = params[:accept] || '*/*'
|
13
|
+
acceptable_types = acceptable.split(/,\s*/)
|
14
|
+
acceptable_types << 'application/x-directory'
|
15
|
+
acceptable_types.any? { |type| Rack::Mime.match?(file.type, type) }
|
16
|
+
end
|
11
17
|
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
<% unless file.relative_parent_path? %>
|
2
|
+
<tr data-ev-location="<%= file.location %>" data-tt-id="<%=path%>" data-tt-parent-id="<%=File.dirname(path)%>" data-tt-branch="<%=file.container? ? 'true' : 'false'%>">
|
3
|
+
<td class="<%=file.container? ? 'ev-container' : 'ev-file'%> ev-file-name">
|
4
|
+
<span class="<%=file.container? ? 'folder' : 'file'%>">
|
5
|
+
<%= link_to(file.name, browse_everything_engine.contents_path(provider_name,file.id), {:class=>'ev-link'}) %>
|
6
|
+
</span>
|
7
|
+
</td>
|
8
|
+
<td class="ev-file-size">
|
9
|
+
<%= number_to_human_size(file.size).sub(/Bytes/,'bytes') %>
|
10
|
+
</td>
|
11
|
+
<td class="ev-file-kind">
|
12
|
+
<%= file.type %>
|
13
|
+
</td>
|
14
|
+
<td class="ev-file-date">
|
15
|
+
<%= file.mtime.strftime('%F %R') %>
|
16
|
+
</td>
|
17
|
+
</tr>
|
18
|
+
<% end %>
|
@@ -1,12 +1,18 @@
|
|
1
1
|
<% if provider.present? %>
|
2
|
-
<
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
2
|
+
<table id="file-list">
|
3
|
+
<thead>
|
4
|
+
<tr>
|
5
|
+
<th>Name</th>
|
6
|
+
<th>Size</th>
|
7
|
+
<th>Kind</th>
|
8
|
+
<th>Modified</th>
|
9
|
+
</tr>
|
10
|
+
</thead>
|
11
|
+
<% provider.contents(browse_path).each_with_index do |file,index| %>
|
12
|
+
<% if is_acceptable?(file) %>
|
13
|
+
<% path = browse_everything_engine.contents_path(provider_name,file.id) %>
|
14
|
+
<%= render :partial => 'file', :locals => { :file => file, :index => index, :path => path } %>
|
15
|
+
<% end %>
|
9
16
|
<% end %>
|
10
17
|
</table>
|
11
18
|
<% end %>
|
12
|
-
|
@@ -13,6 +13,7 @@
|
|
13
13
|
<div class="modal-footer">
|
14
14
|
<span class="pull-left ev-status">0 files selected</span>
|
15
15
|
<%= form_tag '#', :class => "ev-submit-form form-horizontal", :data => { :resolver => browse_everything_engine.resolver_path } do %>
|
16
|
+
<input type="hidden" name="context" value=""/>
|
16
17
|
<button class="ev-cancel btn btn-danger"><%= t('browse_everything.modal_form.cancel')%></button>
|
17
18
|
<button class="ev-submit btn btn-primary" data-loading-text="Loading..."><%= t('browse_everything.modal_form.submit')%></button>
|
18
19
|
<% end %>
|
data/browse-everything.gemspec
CHANGED
@@ -27,8 +27,9 @@ Gem::Specification.new do |spec|
|
|
27
27
|
spec.add_dependency "bootstrap-sass"
|
28
28
|
spec.add_dependency "font-awesome-rails"
|
29
29
|
spec.add_dependency "google-api-client"
|
30
|
-
spec.add_development_dependency "rspec", "~>
|
30
|
+
spec.add_development_dependency "rspec", "~> 3.0"
|
31
31
|
spec.add_development_dependency "rspec-rails"
|
32
|
+
spec.add_development_dependency "rspec-its"
|
32
33
|
spec.add_development_dependency "simplecov"
|
33
34
|
spec.add_development_dependency "bundler", "~> 1.3"
|
34
35
|
spec.add_development_dependency "pry"
|
data/config/routes.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
BrowseEverything::Engine.routes.draw do
|
2
2
|
get "connect", to: 'browse_everything#auth', as: 'connector_response'
|
3
3
|
match "resolve", to: 'browse_everything#resolve', as: 'resolver', via: [:get, :post]
|
4
|
-
|
4
|
+
match ":provider(/*path)", to: 'browse_everything#show', as: 'contents', via: [:get, :post]
|
5
5
|
root to: 'browse_everything#index'
|
6
6
|
end
|
@@ -14,7 +14,7 @@ module BrowseEverything
|
|
14
14
|
@providers = {}
|
15
15
|
opts.each_pair do |driver,config|
|
16
16
|
begin
|
17
|
-
driver_klass = BrowseEverything::Driver.const_get(driver.to_s.camelize.to_sym)
|
17
|
+
driver_klass = BrowseEverything::Driver.const_get((config[:driver] || driver.to_s).camelize.to_sym)
|
18
18
|
@providers[driver] = driver_klass.new(config.merge(url_options: url_options))
|
19
19
|
rescue
|
20
20
|
Rails.logger.warn "Unknown provider: #{driver.to_s}"
|
@@ -22,4 +22,4 @@ module BrowseEverything
|
|
22
22
|
end
|
23
23
|
end
|
24
24
|
end
|
25
|
-
end
|
25
|
+
end
|
@@ -3,9 +3,9 @@ module BrowseEverything
|
|
3
3
|
class Base
|
4
4
|
include BrowseEverything::Engine.routes.url_helpers
|
5
5
|
|
6
|
-
attr_reader :config
|
6
|
+
attr_reader :config, :name
|
7
7
|
attr_accessor :token
|
8
|
-
|
8
|
+
|
9
9
|
def initialize(config,session_info={})
|
10
10
|
@config = config
|
11
11
|
validate_config
|
@@ -20,7 +20,7 @@ module BrowseEverything
|
|
20
20
|
end
|
21
21
|
|
22
22
|
def name
|
23
|
-
self.class.name.split(/::/).last.titleize
|
23
|
+
@name ||= (@config[:name] || self.class.name.split(/::/).last.titleize)
|
24
24
|
end
|
25
25
|
|
26
26
|
def validate_config
|
@@ -52,4 +52,4 @@ module BrowseEverything
|
|
52
52
|
|
53
53
|
end
|
54
54
|
end
|
55
|
-
end
|
55
|
+
end
|