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 CHANGED
@@ -1,15 +1,15 @@
1
1
  ---
2
2
  !binary "U0hBMQ==":
3
3
  metadata.gz: !binary |-
4
- NDAxNjZiMTdhZTE3MmQ2Yjc5NmU0ZGEwNmVjNDU2MDZhZWI1ZGVkOA==
4
+ NWJlNzAxY2ZjZTYxOGJkMGI3ZDYzNzkyMzI3YTEyYzA2ZWYyZGZhYg==
5
5
  data.tar.gz: !binary |-
6
- MzFiZjY5ZTBiOGVhMWU2MTlkY2U3Y2I1YzBmNjAzYWFhMTdhNTg3NA==
6
+ NDExYjU3YTQyOGMyZTI4ZWY3YzEwZTNmZmQyMmQ2ODU4NWFiYzUyZQ==
7
7
  SHA512:
8
8
  metadata.gz: !binary |-
9
- ZGI2Nzc4MzlmMGJlYmU4NTU4NWYwNGZlMGMwYTMzNTFhNGZjMzY1N2U5NWM0
10
- MmUyMDQ1Nzk1YzAwY2I5MjZlM2JkNTZhMWUyZTRmNmVkMzQ5MmI2YTdmNWU2
11
- ZWVmYTQyNjg5NjA5YTk5N2Y4MDQzNWM3NWE3ZWVhN2JlZGZhODc=
9
+ MDhkNTgxMTlhYjQzMTAyZWI4Y2E3MTdiZTkzODdkNGNkMGUwOTc3YzM4Y2Ji
10
+ NjMwOGM0M2U4MjBiNzZjZmI1YzlkZGQ5ZDQyNWZiMzgwYjRmYThiZTg4MmVh
11
+ YjNiZTg0YzhmMzllNTYyYWZiYzQzMzM5OWUzOTZlMzUwM2IyMDA=
12
12
  data.tar.gz: !binary |-
13
- ZDRhYmNhYjMzODkyYjg0ZDlmZjY4MWMwMmRhMjk1ZDQyOTFkZWNmMDA3ZGVh
14
- YjU5OGZmOTU1N2VjZWRhODA4ZDgzMzlmNjYwMTAyYzcwODhkZjA0Nzg5NGY0
15
- NmY1NmE0ZmQ5MTkzODUwMGYyMGVjNmI4OTQ4ZTdjMTY2MTA2Mjc=
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 | type | default | description |
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.
@@ -1,3 +1,4 @@
1
1
  //= require jquery
2
+ //= require jquery.treetable
2
3
  //= require bootstrap
3
- //= require browse_everything/behavior
4
+ //= require browse_everything/behavior
@@ -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, () -> dialog.modal('show')
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-container a', (event) ->
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($(this).attr('href'))
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
- $('input.ev-url').each () ->
74
- $("*[data-ev-location='#{$(this).val()}']").addClass('ev-selected')
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: hidden;
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
- @provider_nane ||= params[:provider] || params[:state].to_s.split(/\|/).last
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
- <ul class="unstyled">
3
- <% provider.contents(browse_path).each do |file| %>
4
- <li data-ev-location="<%= file.location %>" title="<%= number_to_human_size(file.size).sub(/Bytes/,'bytes') %> • <%= file.mtime.strftime('%F %R') %>">
5
- <span class="<%=file.container? ? 'ev-container' : 'ev-file'%> ev-file-name">
6
- <%= link_to(file.name, browse_everything_engine.contents_path(provider_name,file.id)) %>
7
- </span>
8
- </li>
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 %>
@@ -8,4 +8,4 @@
8
8
  <%= yield %>
9
9
  </div>
10
10
  </body>
11
- </html>
11
+ </html>
@@ -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", "~> 2.0"
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
- get ":provider(/*path)", to: 'browse_everything#show', as: 'contents'
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