forked from 0x2620/pandora
583 lines
21 KiB
JavaScript
583 lines
21 KiB
JavaScript
// vim: et:ts=4:sw=4:sts=4:ft=javascript
|
|
|
|
'use strict';
|
|
|
|
pandora.ui.listDialog = function(section) {
|
|
|
|
section = section || 'general';
|
|
var listData = pandora.getListData(),
|
|
tabs = [].concat([
|
|
{id: 'general', title: 'General'},
|
|
{id: 'icon', title: 'Icon'}
|
|
], listData.type == 'smart'
|
|
? [{id: 'query', title: 'Query'}]
|
|
: []
|
|
),
|
|
ui = pandora.user.ui,
|
|
width = getWidth(section),
|
|
folderItems = ui.section == 'items' ? 'Lists' : Ox.toTitleCase(ui.section),
|
|
folderItem = folderItems.slice(0, -1);
|
|
Ox.getObjectById(tabs, section).selected = true;
|
|
|
|
pandora.$ui.listDialogTabPanel = Ox.TabPanel({
|
|
content: function(id) {
|
|
if (id == 'general') {
|
|
return pandora.ui.listGeneralPanel(listData);
|
|
} else if (id == 'icon') {
|
|
return pandora.$ui.listIconPanel = pandora.ui.listIconPanel(listData);
|
|
} else if (id == 'query') {
|
|
return pandora.$ui.filterForm = pandora.ui.filterForm(listData)
|
|
.bindEvent({
|
|
change: function(data) {
|
|
listData.query = data.query;
|
|
}
|
|
});
|
|
}
|
|
},
|
|
tabs: tabs
|
|
})
|
|
.bindEvent({
|
|
change: function(data) {
|
|
var width = getWidth(data.selected);
|
|
$dialog.options({
|
|
maxWidth: width,
|
|
minWidth: width
|
|
});
|
|
$dialog.setSize(width, 312);
|
|
$findElement[data.selected == 'icon' ? 'show' : 'hide']();
|
|
$updateCheckbox[data.selected == 'query' ? 'show' : 'hide']();
|
|
if (
|
|
pandora.user.ui.section == 'items'
|
|
&& data.selected != 'query'
|
|
&& !pandora.user.ui.updateAdvancedFindResults
|
|
) {
|
|
pandora.$ui.filterForm.updateResults();
|
|
}
|
|
}
|
|
});
|
|
pandora.$ui.listDialogTabPanel.find('.OxButtonGroup').css({width: '256px'});
|
|
|
|
var $findElement = Ox.FormElementGroup({
|
|
elements: [
|
|
pandora.$ui.findIconItemSelect = Ox.Select({
|
|
items: pandora.site.findKeys.map(function(findKey) {
|
|
return {id: findKey.id, title: 'Find: ' + findKey.title};
|
|
}),
|
|
overlap: 'right',
|
|
type: 'image'
|
|
})
|
|
.bindEvent({
|
|
change: function(data) {
|
|
pandora.$ui.findIconItemInput.options({
|
|
placeholder: data.title
|
|
});
|
|
// fixme: this is a bit weird
|
|
setTimeout(function() {
|
|
pandora.$ui.findIconItemInput.focusInput(true);
|
|
}, 250);
|
|
}
|
|
}),
|
|
pandora.$ui.findIconItemInput = Ox.Input({
|
|
changeOnKeypress: true,
|
|
clear: true,
|
|
placeholder: 'Find: All',
|
|
width: 128 + Ox.UI.SCROLLBAR_SIZE
|
|
})
|
|
.bindEvent({
|
|
change: function(data) {
|
|
pandora.$ui.listIconPanel.updateQuery(
|
|
pandora.$ui.findIconItemSelect.value(),
|
|
data.value
|
|
);
|
|
}
|
|
})
|
|
],
|
|
})
|
|
.css({float: 'right', margin: '4px', align: 'right'})
|
|
[section == 'icon' ? 'show' : 'hide']()
|
|
.appendTo(pandora.$ui.listDialogTabPanel.children('.OxBar')),
|
|
|
|
$dialog = Ox.Dialog({
|
|
buttons: [
|
|
Ox.Button({
|
|
id: 'done',
|
|
title: 'Done'
|
|
})
|
|
.bindEvent({
|
|
click: function() {
|
|
if (
|
|
pandora.$ui.listDialogTabPanel.selected() == 'query'
|
|
&& !pandora.user.ui.updateAdvancedFindResults
|
|
) {
|
|
pandora.$ui.filterForm.updateResults();
|
|
}
|
|
$dialog.close();
|
|
}
|
|
})
|
|
],
|
|
closeButton: true,
|
|
content: pandora.$ui.listDialogTabPanel,
|
|
maxWidth: width,
|
|
minHeight: 312,
|
|
minWidth: width,
|
|
height: 312,
|
|
// keys: {enter: 'save', escape: 'cancel'},
|
|
removeOnClose: true,
|
|
title: folderItem + ' — ' + Ox.encodeHTMLEntities(listData.name),
|
|
width: width
|
|
}),
|
|
|
|
$updateCheckbox = Ox.Checkbox({
|
|
title: 'Update Results in the Background',
|
|
value: pandora.user.ui.updateAdvancedFindResults
|
|
})
|
|
.css({float: 'left', margin: '4px'})
|
|
[section == 'query' ? 'show' : 'hide']()
|
|
.bindEvent({
|
|
change: function(data) {
|
|
pandora.UI.set({updateAdvancedFindResults: data.value});
|
|
data.value && pandora.$ui.filterForm.updateResults();
|
|
}
|
|
});
|
|
|
|
$($updateCheckbox.find('.OxButton')[0]).css({margin: 0});
|
|
$($dialog.$element.find('.OxBar')[2]).append($updateCheckbox);
|
|
|
|
function getWidth(section) {
|
|
return section == 'general' ? 496
|
|
: (section == 'icon' ? 696 : 648) + Ox.UI.SCROLLBAR_SIZE;
|
|
}
|
|
|
|
return $dialog;
|
|
|
|
};
|
|
|
|
pandora.ui.listGeneralPanel = function(listData) {
|
|
var that = Ox.Element(),
|
|
ui = pandora.user.ui,
|
|
folderItems = ui.section == 'items' ? 'Lists' : Ox.toTitleCase(ui.section),
|
|
folderItem = folderItems.slice(0, -1);
|
|
pandora.api['find' + folderItems]({
|
|
query: {conditions: [{key: 'id', value: listData.id, operator: '=='}]},
|
|
keys: ['description', 'subscribers']
|
|
}, function(result) {
|
|
var description = result.data.items[0].description,
|
|
subscribers = result.data.items[0].subscribers,
|
|
$icon = Ox.Element({
|
|
element: '<img>',
|
|
tooltip: 'Doubleclick to edit icon'
|
|
})
|
|
.attr({
|
|
src: '/' + folderItem.toLowerCase() + '/' + listData.id + '/icon256.jpg?' + Ox.uid()
|
|
})
|
|
.css({
|
|
position: 'absolute',
|
|
left: '16px',
|
|
top: '16px',
|
|
width: '128px',
|
|
height: '128px',
|
|
borderRadius: '32px'
|
|
})
|
|
.bindEvent({
|
|
doubleclick: function() {
|
|
pandora.$ui.listDialogTabPanel.select('icon');
|
|
}
|
|
})
|
|
.appendTo(that),
|
|
$nameInput = Ox.Input({
|
|
label: 'Name',
|
|
labelWidth: 80,
|
|
value: listData.name,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '16px'})
|
|
.bindEvent({
|
|
blur: editName,
|
|
submit: editName
|
|
})
|
|
.appendTo(that),
|
|
$itemsInput = ui.section == 'items'
|
|
? Ox.Input({
|
|
disabled: true,
|
|
label: 'Items',
|
|
labelWidth: 80,
|
|
value: listData.items,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '40px'})
|
|
.appendTo(that)
|
|
: Ox.Select({
|
|
items: pandora.site.textRightsLevels.map(function(rightsLevel, i) {
|
|
console.log(listData);
|
|
return {
|
|
id: i,
|
|
title: rightsLevel.name,
|
|
};
|
|
}),
|
|
label: 'Rights Level',
|
|
labelWidth: 90,
|
|
value: listData.rightslevel,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '40px'})
|
|
.bindEvent({
|
|
change: editRightsLevel
|
|
})
|
|
.appendTo(that),
|
|
$statusSelect = listData.status == 'featured'
|
|
? Ox.Input({
|
|
disabled: true,
|
|
label: 'Status',
|
|
labelWidth: 80,
|
|
value: 'Featured',
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '64px'})
|
|
.appendTo(that)
|
|
: Ox.Select({
|
|
items: [
|
|
{id: 'private', title: 'Private'},
|
|
{id: 'public', title: 'Public'}
|
|
],
|
|
label: 'Status',
|
|
labelWidth: 80,
|
|
value: listData.status,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '64px'})
|
|
.bindEvent({
|
|
change: editStatus
|
|
})
|
|
.appendTo(that),
|
|
$subscribersInput = Ox.Input({
|
|
disabled: true,
|
|
label: 'Subscribers',
|
|
labelWidth: 80,
|
|
value: subscribers,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: '88px'})
|
|
[getSubscribersAction()]()
|
|
.appendTo(that),
|
|
$descriptionInput = Ox.Input({
|
|
height: getDescriptionHeight(),
|
|
placeholder: 'Description',
|
|
type: 'textarea',
|
|
value: description,
|
|
width: 320
|
|
})
|
|
.css({position: 'absolute', left: '160px', top: getDescriptionTop() + 'px'})
|
|
.bindEvent({
|
|
blur: editDescription,
|
|
submit: editDescription
|
|
})
|
|
.appendTo(that);
|
|
function editDescription(data) {
|
|
if (data.value != description) {
|
|
pandora.api['edit' + folderItem]({
|
|
id: listData.id,
|
|
description: data.value
|
|
}, function(result) {
|
|
description = result.data.description;
|
|
Ox.Request.clearCache('find' + folderItems);
|
|
pandora.$ui.info.updateListInfo();
|
|
});
|
|
}
|
|
}
|
|
function editName(data) {
|
|
if (data.value != listData.name) {
|
|
pandora.api['edit' + folderItem]({
|
|
id: listData.id,
|
|
name: data.value
|
|
}, function(result) {
|
|
if (result.data.id != listData.id) {
|
|
pandora.renameList(listData.id, result.data.id, result.data.name);
|
|
listData.id = result.data.id;
|
|
listData.name = result.data.name;
|
|
Ox.Request.clearCache('find' + folderItems);
|
|
pandora.$ui.info.updateListInfo();
|
|
pandora.$ui.listDialog.options({
|
|
title: folderItem + ' — ' + Ox.encodeHTMLEntities(listData.name)
|
|
});
|
|
}
|
|
});
|
|
}
|
|
}
|
|
function editStatus(data) {
|
|
var status = data.value;
|
|
$statusSelect.value(status == 'private' ? 'public' : 'private');
|
|
pandora.changeFolderItemStatus(listData.id, status, function(result) {
|
|
listData.status = result.data.status;
|
|
if (result.data.status == 'private') {
|
|
subscribers = 0;
|
|
$subscribersInput.value(0);
|
|
}
|
|
$statusSelect.value(result.data.status);
|
|
$subscribersInput[getSubscribersAction()]();
|
|
$descriptionInput
|
|
.options({height: getDescriptionHeight()})
|
|
.css({top: getDescriptionTop() + 'px'});
|
|
pandora.$ui.folderList[listData.folder].value(
|
|
result.data.id, 'status', result.data.status
|
|
);
|
|
});
|
|
}
|
|
function editRightsLevel(data) {
|
|
var rightslevel = data.value;
|
|
pandora.api.editText({
|
|
id: listData.id,
|
|
rightslevel: rightslevel
|
|
}, function(result) {
|
|
Ox.Request.clearCache('getText');
|
|
//fixme: reload text and folder list
|
|
$itemsInput.value(result.data.rightslevel);
|
|
});
|
|
}
|
|
function getDescriptionHeight() {
|
|
return listData.status == 'private' ? 184 : 160;
|
|
}
|
|
function getDescriptionTop() {
|
|
return listData.status == 'private' ? 88 : 112;
|
|
}
|
|
function getSubscribersAction() {
|
|
return listData.status == 'private' ? 'hide' : 'show';
|
|
}
|
|
});
|
|
return that;
|
|
};
|
|
|
|
pandora.ui.listIconPanel = function(listData) {
|
|
|
|
var quarter = 0,
|
|
quarters = ['top-left', 'top-right', 'bottom-left', 'bottom-right'],
|
|
|
|
ui = pandora.user.ui,
|
|
folderItems = ui.section == 'items' ? 'Lists' : Ox.toTitleCase(ui.section),
|
|
folderItem = folderItems.slice(0, -1),
|
|
|
|
$iconPanel = Ox.Element(),
|
|
|
|
$icon = $('<img>')
|
|
.attr({src: '/' + folderItem.toLowerCase() + '/' + listData.id + '/icon256.jpg?' + Ox.uid()})
|
|
.css({position: 'absolute', borderRadius: '64px', margin: '16px'})
|
|
.appendTo($iconPanel),
|
|
|
|
$previewPanel = Ox.Element(),
|
|
|
|
$preview,
|
|
|
|
$list = Ox.Element(),
|
|
|
|
ui = pandora.user.ui,
|
|
folderItems = ui.section == 'items' ? 'Lists' : Ox.toTitleCase(ui.section),
|
|
folderItem = folderItems.slice(0, -1),
|
|
|
|
that = Ox.SplitPanel({
|
|
elements: [
|
|
{
|
|
element: $iconPanel,
|
|
size: 280
|
|
},
|
|
{
|
|
element: $previewPanel
|
|
},
|
|
{
|
|
element: $list,
|
|
size: 144 + Ox.UI.SCROLLBAR_SIZE
|
|
}
|
|
],
|
|
orientation: 'horizontal'
|
|
});
|
|
|
|
pandora.api['find' + folderItems]({
|
|
query: {
|
|
conditions: [{key: 'id', value: listData.id, operator: '=='}],
|
|
operator: '&'
|
|
},
|
|
keys: ['posterFrames']
|
|
}, function(result) {
|
|
|
|
var posterFrames = result.data.items[0].posterFrames,
|
|
posterFrame = posterFrames[quarter],
|
|
|
|
$interface = Ox.Element({
|
|
tooltip: function(e) {
|
|
return 'Edit ' + $(e.target).attr('id').replace('-', ' ') + ' image';
|
|
}
|
|
})
|
|
.css({
|
|
position: 'absolute',
|
|
width: '256px',
|
|
height: '256px',
|
|
marginLeft: '16px',
|
|
marginTop: '16px',
|
|
cursor: 'pointer'
|
|
})
|
|
.on({
|
|
click: function(e) {
|
|
clickIcon(e);
|
|
},
|
|
dblclick: function(e) {
|
|
clickIcon(e, true);
|
|
}
|
|
})
|
|
.appendTo($iconPanel);
|
|
|
|
renderQuarters();
|
|
|
|
$list = Ox.IconList({
|
|
borderRadius: 16,
|
|
item: function(data, sort) {
|
|
var size = 128;
|
|
return {
|
|
height: size,
|
|
id: data.id,
|
|
info: data[['title', 'director'].indexOf(sort[0].key) > -1 ? 'year' : sort[0].key],
|
|
title: data.title + (data.director.length ? ' (' + data.director.join(', ') + ')' : ''),
|
|
url: '/' + data.id + '/icon' + size + '.jpg',
|
|
width: size
|
|
};
|
|
},
|
|
items: function(data, callback) {
|
|
pandora.api.find(Ox.extend(data, {
|
|
query: {
|
|
conditions: ui.section == 'items' ? [{key: 'list', value: listData.id, operator: '=='}] : [],
|
|
operator: '&'
|
|
}
|
|
}), callback);
|
|
},
|
|
keys: ['director', 'duration', 'id', 'posterFrame', 'title', 'videoRatio', 'year'],
|
|
max: 1,
|
|
min: 1,
|
|
//orientation: 'vertical',
|
|
selected: posterFrame ? [posterFrame.item] : [],
|
|
size: 128,
|
|
sort: ui.section == 'items' ? pandora.user.ui.listSort : pandora.site.user.ui.listSort,
|
|
unique: 'id'
|
|
})
|
|
//.css({width: '144px'})
|
|
.bindEvent({
|
|
open: function(data) {
|
|
setPosterFrame(data.ids[0], $list.value(data.ids[0], 'posterFrame'))
|
|
},
|
|
select: function(data) {
|
|
renderPreview($list.value(data.ids[0]));
|
|
}
|
|
})
|
|
.bindEventOnce({
|
|
load: function() {
|
|
var itemData;
|
|
if (!posterFrame) {
|
|
itemData = $list.value(0);
|
|
$list.options({selected: [itemData.id]});
|
|
} else {
|
|
itemData = $list.value(posterFrame.item);
|
|
}
|
|
posterFrames.length && renderPreview(itemData);
|
|
}
|
|
})
|
|
.gainFocus();
|
|
|
|
that.replaceElement(2, $list);
|
|
|
|
function clickIcon(e, isDoubleClick) {
|
|
quarter = quarters.indexOf($(e.target).attr('id'));
|
|
renderQuarters();
|
|
if (isDoubleClick && posterFrames.length) {
|
|
var item = posterFrames[quarter].item;
|
|
$list.options({selected: [item]});
|
|
renderPreview($list.value(item), posterFrames[quarter].position);
|
|
}
|
|
}
|
|
|
|
function renderPreview(itemData, position) {
|
|
$preview = pandora.ui.videoPreview({
|
|
duration: itemData.duration,
|
|
frameRatio: itemData.videoRatio,
|
|
height: 256,
|
|
id: itemData.id,
|
|
position: position,
|
|
width: 256
|
|
})
|
|
.css({marginLeft: '8px', marginTop: '16px', overflow: 'hidden'})
|
|
.bindEvent({
|
|
click: function(data) {
|
|
setPosterFrame(itemData.id, data.position);
|
|
}
|
|
});
|
|
/*
|
|
// fixme: need canvas for this
|
|
$($preview.children('.OxFrame')[0])
|
|
.css('border-' + quarter + '-radius', '128px');
|
|
*/
|
|
$previewPanel.empty().append($preview);
|
|
}
|
|
|
|
function renderQuarters() {
|
|
$interface.empty();
|
|
quarters.forEach(function(q, i) {
|
|
$interface.append(
|
|
$('<div>')
|
|
.attr({id: q})
|
|
.css({
|
|
float: 'left',
|
|
width: '126px',
|
|
height: '126px',
|
|
border: '1px solid rgba(255, 255, 255, ' + (i == quarter ? 0.75 : 0) + ')',
|
|
background: 'rgba(0, 0, 0, ' + (i == quarter ? 0 : 0.75) + ')'
|
|
})
|
|
.css('border-' + q + '-radius', '64px')
|
|
);
|
|
});
|
|
}
|
|
|
|
function setPosterFrame(item, position) {
|
|
var posterFrame = {item: item, position: position};
|
|
if (posterFrames.length) {
|
|
posterFrames[quarter] = posterFrame;
|
|
} else {
|
|
posterFrames = Ox.repeat([posterFrame], 4);
|
|
}
|
|
pandora.api['edit' + folderItem]({
|
|
id: listData.id,
|
|
posterFrames: posterFrames
|
|
}, function() {
|
|
$icon.attr({
|
|
src: '/' + folderItem.toLowerCase() + '/' + listData.id + '/icon256.jpg?' + Ox.uid()
|
|
});
|
|
pandora.$ui.folderList[listData.folder].$element
|
|
.find('img[src*="/' + listData.id + '/"]')
|
|
.attr({
|
|
src: '/' + folderItem.toLowerCase() + '/' + listData.id + '/icon.jpg?' + Ox.uid()
|
|
});
|
|
pandora.$ui.info.updateListInfo();
|
|
pandora.clearListIconCache(listData.id);
|
|
});
|
|
$preview.options({position: position});
|
|
}
|
|
|
|
});
|
|
|
|
function renderFrame() {
|
|
$frame.css({borderRadius: 0});
|
|
$frame.css('border-' + quarters[quarter] + '-radius', '128px');
|
|
}
|
|
|
|
that.updateQuery = function(key, value) {
|
|
$list.options({
|
|
items: function(data, callback) {
|
|
pandora.api.find(Ox.extend(data, {
|
|
query: {
|
|
conditions: [{key: 'list', value: listData.id, operator: '=='}].concat(
|
|
value !== '' ? [{key: key, value: value, operator: '='}] : []
|
|
),
|
|
operator: '&'
|
|
}
|
|
}), callback);
|
|
}
|
|
});
|
|
};
|
|
|
|
return that;
|
|
|
|
}
|