2015-02-12 15:12:19 +03:00
|
|
|
<!DOCTYPE html>
|
|
|
|
|
<html>
|
|
|
|
|
<style>
|
|
|
|
|
|
|
|
|
|
.container {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
|
|
|
|
top: 100px;
|
|
|
|
|
left: 100px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.browse {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
min-width: 300px;
|
|
|
|
|
width: initial;
|
2015-02-13 03:09:40 +03:00
|
|
|
background: gray;
|
|
|
|
|
color: rgba(255,255,255,0.8);
|
|
|
|
|
padding: 5px;
|
|
|
|
|
font-family: sans-serif;
|
2015-02-12 15:12:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.browse .v-block {
|
|
|
|
|
width: 100%;
|
|
|
|
|
height: auto;
|
|
|
|
|
|
|
|
|
|
box-sizing: border-box;
|
2015-02-13 03:09:40 +03:00
|
|
|
|
|
|
|
|
border-top: 1px solid rgba(255,255,255, 0.3);
|
|
|
|
|
}
|
|
|
|
|
.browse .v-block:first-of-type {
|
|
|
|
|
border-top: none;
|
2015-02-12 15:12:19 +03:00
|
|
|
}
|
|
|
|
|
.browse .v-block:empty {
|
|
|
|
|
display: none;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-13 03:09:40 +03:00
|
|
|
.browse .title {
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
color: rgba(255,255,255,0.9);
|
|
|
|
|
padding: 5px;
|
|
|
|
|
padding-left: 10px;
|
|
|
|
|
padding-right: 10px;
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 15:12:19 +03:00
|
|
|
.browse .path {
|
2015-02-13 03:09:40 +03:00
|
|
|
padding: 5px;
|
|
|
|
|
padding-left: 10px;
|
|
|
|
|
padding-right: 10px;
|
2015-02-12 15:12:19 +03:00
|
|
|
white-space: nowrap;
|
|
|
|
|
}
|
2015-02-13 02:28:42 +03:00
|
|
|
.browse .path:empty {
|
|
|
|
|
display: block;
|
|
|
|
|
}
|
|
|
|
|
.browse .path:before {
|
|
|
|
|
content: "/";
|
|
|
|
|
}
|
2015-02-12 15:12:19 +03:00
|
|
|
.browse .path .dir {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
}
|
|
|
|
|
.browse .path .dir:after {
|
|
|
|
|
content: "/";
|
|
|
|
|
}
|
2015-02-13 03:09:40 +03:00
|
|
|
.browse .list div {
|
|
|
|
|
padding: 5px;
|
|
|
|
|
padding-left: 10px;
|
|
|
|
|
padding-right: 10px;
|
2015-02-12 17:16:00 +03:00
|
|
|
}
|
|
|
|
|
|
2015-02-13 03:09:40 +03:00
|
|
|
.browse .list div.selected,
|
2015-02-12 15:12:19 +03:00
|
|
|
.browse .path .dir:hover,
|
|
|
|
|
.browse .list div:hover {
|
2015-02-13 03:09:40 +03:00
|
|
|
background: rgba(0,0,0, 0.05);
|
|
|
|
|
color: white;
|
|
|
|
|
text-shadow: rgba(0,0,0,0.5) 0.05em 0.005em 0.4em,
|
|
|
|
|
rgba(0,0,0,0.5) 0.05em 0.05em;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.browse .list div.selected {
|
2015-02-12 15:12:19 +03:00
|
|
|
background: rgba(0,0,0, 0.1);
|
2015-02-13 03:09:40 +03:00
|
|
|
text-shadow: rgba(0,0,0,0.3) 0.1em 0.1em 0.4em,
|
|
|
|
|
rgba(0,0,0,0.3) 0.1em 0.1em;
|
|
|
|
|
box-shadow: rgba(0,0,0,0.2) 0.1em 0.1em 0.2em;
|
2015-02-12 15:12:19 +03:00
|
|
|
}
|
|
|
|
|
|
2015-02-13 03:09:40 +03:00
|
|
|
|
2015-02-12 15:12:19 +03:00
|
|
|
</style>
|
|
|
|
|
|
|
|
|
|
<script src="../ext-lib/jquery.js"></script>
|
|
|
|
|
<script src="../ext-lib/jquery-ui.js"></script>
|
|
|
|
|
|
|
|
|
|
<script src="../lib/jli.js"></script>
|
|
|
|
|
|
2015-02-12 19:07:13 +03:00
|
|
|
<script src="../ext-lib/require.js"></script>
|
|
|
|
|
|
2015-02-12 15:12:19 +03:00
|
|
|
<script>
|
|
|
|
|
|
|
|
|
|
var TREE = {
|
|
|
|
|
dir_a: {},
|
|
|
|
|
dir_b: {
|
|
|
|
|
file1: 'this is a file',
|
|
|
|
|
file2: 'this is a file',
|
|
|
|
|
file3: 'this is a file',
|
|
|
|
|
},
|
|
|
|
|
dir_c: {
|
|
|
|
|
file1: 'this is a file',
|
|
|
|
|
dir_b: {
|
|
|
|
|
file1: 'this is a file',
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
file: 'this is a file',
|
|
|
|
|
}
|
2015-02-13 02:28:42 +03:00
|
|
|
// add some recursion for testing...
|
|
|
|
|
TREE.dir_d = TREE.dir_c.dir_b
|
|
|
|
|
TREE.dir_a.tree = TREE
|
|
|
|
|
TREE.dir_c.dir_b.tree = TREE
|
2015-02-12 15:12:19 +03:00
|
|
|
|
2015-02-12 19:07:13 +03:00
|
|
|
|
2015-02-12 15:12:19 +03:00
|
|
|
function skipFiles(e, v){
|
|
|
|
|
return typeof(v) != typeof('str')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function make(title){
|
|
|
|
|
var browser = $('<div>')
|
|
|
|
|
.addClass('browse')
|
|
|
|
|
.append($('<div>')
|
|
|
|
|
.addClass('v-block title')
|
|
|
|
|
.text(title))
|
|
|
|
|
.append($('<div>')
|
|
|
|
|
.addClass('v-block path'))
|
|
|
|
|
.append($('<div>')
|
|
|
|
|
.addClass('v-block list'))
|
|
|
|
|
.append($('<div>')
|
|
|
|
|
.addClass('v-block info'))
|
|
|
|
|
.append($('<div>')
|
|
|
|
|
.addClass('v-block actions'))
|
|
|
|
|
return browser
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// low level update...
|
|
|
|
|
function update(browser, path, list){
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
var p = browser.find('.path').empty()
|
|
|
|
|
var l = browser.find('.list').empty()
|
|
|
|
|
|
|
|
|
|
path.forEach(function(e){
|
|
|
|
|
p.append($('<div>')
|
|
|
|
|
.addClass('dir')
|
|
|
|
|
.click(popDir)
|
|
|
|
|
.text(e))
|
|
|
|
|
})
|
|
|
|
|
list.forEach(function(e){
|
|
|
|
|
l.append($('<div>')
|
|
|
|
|
.click(pushDir)
|
|
|
|
|
.text(e))
|
|
|
|
|
})
|
|
|
|
|
return browser
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function showPath(browser, path, tree, skip){
|
|
|
|
|
// XXX remove for pruduction...
|
|
|
|
|
skip = skip || skipFiles
|
|
|
|
|
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
path = path.constructor !== Array ? path.split(/[\\\/]+/g) : path
|
|
|
|
|
path = path.filter(function(e){ return e != '' })
|
|
|
|
|
|
|
|
|
|
var dir = tree
|
|
|
|
|
path.forEach(function(d){
|
|
|
|
|
dir = dir[d]
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// XXX do error checking...
|
|
|
|
|
|
|
|
|
|
dir = skip != null
|
|
|
|
|
// skip files....
|
|
|
|
|
? Object.keys(dir).filter(function(e){ return skip(e, dir[e]) })
|
|
|
|
|
: Object.keys(dir)
|
|
|
|
|
|
|
|
|
|
update(browser, path, dir)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// XXX this will skip incuding the current dir...
|
|
|
|
|
// ...this might work and might not work...
|
|
|
|
|
function getPath(browser, to){
|
|
|
|
|
var skip = false
|
|
|
|
|
return browser.find('.path .dir')
|
|
|
|
|
.filter(function(i, e){
|
|
|
|
|
if(e === to){
|
|
|
|
|
skip = true
|
|
|
|
|
}
|
|
|
|
|
return !skip
|
|
|
|
|
})
|
|
|
|
|
.map(function(i, e){ return $(e).text() })
|
|
|
|
|
.toArray()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function pushDir(){
|
|
|
|
|
var dir = $(this).text()
|
|
|
|
|
var browser = $(this).parents('.browse')
|
|
|
|
|
var path = getPath(browser)
|
|
|
|
|
path.push(dir)
|
|
|
|
|
|
|
|
|
|
showPath(browser, path, TREE)
|
|
|
|
|
}
|
|
|
|
|
function popDir(){
|
|
|
|
|
var dir = $(this).text()
|
|
|
|
|
var browser = $(this).parents('.browse')
|
|
|
|
|
var path = getPath(browser, this)
|
|
|
|
|
|
|
|
|
|
showPath(browser, path, TREE)
|
2015-02-13 02:28:42 +03:00
|
|
|
|
|
|
|
|
select('"'+dir+'"')
|
2015-02-12 15:12:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// base control actions...
|
|
|
|
|
// XXX add keywords:
|
|
|
|
|
// 'first'
|
|
|
|
|
// 'prev'
|
|
|
|
|
// 'next'
|
|
|
|
|
// 'last'
|
|
|
|
|
// 'none' -- deselect
|
|
|
|
|
// '!' -- return selected element if it exists...
|
|
|
|
|
// <number> -- select by sequence number...
|
|
|
|
|
// <elem> -- select a specific element...
|
|
|
|
|
function select(elem, browser){
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
var elems = browser.find('.list div')
|
|
|
|
|
|
|
|
|
|
if(elems.length == 0){
|
|
|
|
|
return $()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
elem = elem || select('!')
|
|
|
|
|
// if none selected get the first...
|
|
|
|
|
elem = elem.length == 0 ? 'first' : elem
|
|
|
|
|
|
|
|
|
|
// first/last...
|
|
|
|
|
if(elem == 'first' || elem == 'last'){
|
|
|
|
|
return select(elems[elem](), browser)
|
|
|
|
|
|
|
|
|
|
// prev/next...
|
|
|
|
|
} else if(elem == 'prev' || elem == 'next'){
|
|
|
|
|
var to = select('!', browser)[elem]('.list div')
|
|
|
|
|
if(to.length == 0){
|
|
|
|
|
return select(elem == 'prev' ? 'last' : 'first', browser)
|
|
|
|
|
}
|
|
|
|
|
select('none')
|
|
|
|
|
return select(to, browser)
|
|
|
|
|
|
|
|
|
|
// deselect...
|
|
|
|
|
} else if(elem == 'none'){
|
|
|
|
|
return elems
|
|
|
|
|
.filter('.selected')
|
|
|
|
|
.removeClass('selected')
|
|
|
|
|
|
|
|
|
|
// strict...
|
|
|
|
|
} else if(elem == '!'){
|
|
|
|
|
return elems.filter('.selected')
|
|
|
|
|
|
|
|
|
|
// number...
|
|
|
|
|
} else if(typeof(elem) == typeof(123)){
|
|
|
|
|
return select($(elems[elem]), browser)
|
|
|
|
|
|
|
|
|
|
// string...
|
|
|
|
|
} else if(typeof(elem) == typeof('str') && /^'.*'$|^".*"$/.test(elem.trim())){
|
|
|
|
|
elem = elem.trim().slice(1, -1)
|
|
|
|
|
return select(browser.find('.list div').filter(function(i, e){
|
|
|
|
|
return $(e).text() == elem
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
|
|
// element...
|
|
|
|
|
} else {
|
|
|
|
|
select('none', browser)
|
|
|
|
|
return elem.addClass('selected')
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
function push(browser){
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
var elem = select('!', browser)
|
|
|
|
|
|
|
|
|
|
if(elem.length == 0){
|
|
|
|
|
return select()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var path = getPath(browser)
|
|
|
|
|
|
|
|
|
|
path.push(elem.text())
|
|
|
|
|
|
|
|
|
|
var res = showPath(browser, path, TREE)
|
|
|
|
|
|
|
|
|
|
select(null, browser)
|
|
|
|
|
|
|
|
|
|
return res
|
|
|
|
|
}
|
|
|
|
|
function pop(browser){
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
var path = getPath(browser)
|
|
|
|
|
var dir = path.pop()
|
|
|
|
|
|
|
|
|
|
var res = showPath(browser, path, TREE)
|
|
|
|
|
|
|
|
|
|
select('"'+dir+'"')
|
|
|
|
|
|
|
|
|
|
return res
|
|
|
|
|
}
|
|
|
|
|
function next(browser){
|
|
|
|
|
return select('next', browser)
|
|
|
|
|
}
|
|
|
|
|
function prev(browser){
|
|
|
|
|
return select('prev', browser)
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-13 02:28:42 +03:00
|
|
|
|
|
|
|
|
// default action...
|
|
|
|
|
function action(browser){
|
|
|
|
|
browser = browser || $('.browse')
|
|
|
|
|
|
2015-02-13 03:14:53 +03:00
|
|
|
var cur = select('!', browser).text()
|
|
|
|
|
alert('/' + getPath(browser).concat(cur == '' ? [cur] : [cur, '']).join('/'))
|
2015-02-13 02:28:42 +03:00
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-12 19:07:13 +03:00
|
|
|
var KB = {
|
|
|
|
|
'*':{
|
|
|
|
|
F5: function(){ window.location.reload() },
|
2015-02-13 02:28:42 +03:00
|
|
|
},
|
|
|
|
|
'.browse':{
|
2015-02-12 19:07:13 +03:00
|
|
|
Up: prev,
|
2015-02-13 02:28:42 +03:00
|
|
|
Backspace: 'Up',
|
2015-02-12 19:07:13 +03:00
|
|
|
Down: next,
|
|
|
|
|
Left: pop,
|
|
|
|
|
Right: push,
|
2015-02-13 02:28:42 +03:00
|
|
|
|
|
|
|
|
Enter: action,
|
2015-02-12 19:07:13 +03:00
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
requirejs(['../lib/keyboard'], function(m){
|
|
|
|
|
keyboard = m
|
|
|
|
|
|
|
|
|
|
// setup base keyboard for devel, in case something breaks...
|
|
|
|
|
$(document)
|
|
|
|
|
.keydown(
|
|
|
|
|
keyboard.makeKeyboardHandler(
|
|
|
|
|
KB,
|
2015-02-13 02:28:42 +03:00
|
|
|
function(k){ window.DEBUG && console.log(k) }))
|
2015-02-12 19:07:13 +03:00
|
|
|
|
|
|
|
|
})
|
2015-02-12 15:12:19 +03:00
|
|
|
|
|
|
|
|
$(function(){
|
|
|
|
|
var browser = make()
|
|
|
|
|
$('.container')
|
|
|
|
|
.empty()
|
|
|
|
|
.append(browser)
|
|
|
|
|
showPath(browser, '/', TREE)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<body>
|
|
|
|
|
|
2015-02-12 17:16:00 +03:00
|
|
|
<button onclick="pop()"><</button>
|
|
|
|
|
<button onclick="push()">></button>
|
|
|
|
|
<button onclick="prev()">^</button>
|
|
|
|
|
<button onclick="next()">v</button>
|
|
|
|
|
|
2015-02-12 15:12:19 +03:00
|
|
|
|
|
|
|
|
<div class="container">
|
|
|
|
|
<div class="browse">
|
|
|
|
|
<!-- title, optional -->
|
|
|
|
|
<div class="v-block title">
|
|
|
|
|
[title]
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- the actual list -->
|
|
|
|
|
<div class="v-block path">
|
|
|
|
|
<div class="dir">
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
<div class="dir">
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
<div class="dir">
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- the actual list -->
|
|
|
|
|
<div class="v-block list">
|
|
|
|
|
<div>
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
|
|
|
|
[dir]
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- info, optional -->
|
|
|
|
|
<div class="v-block info">
|
|
|
|
|
[info]
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- buttons, optional -->
|
|
|
|
|
<div class="v-block actions">
|
|
|
|
|
[actions]
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</body>
|
|
|
|
|
</html>
|