Merge branch 'master' of https://github.com/RuudBurger/CouchPotatoServer
This commit is contained in:
+1
-2
@@ -37,10 +37,9 @@ class Loader(object):
|
||||
# Create data dir if needed
|
||||
if self.options.data_dir:
|
||||
self.data_dir = self.options.data_dir
|
||||
|
||||
else:
|
||||
self.data_dir = os.path.expanduser(Env.setting('data_dir'))
|
||||
|
||||
|
||||
if self.data_dir == '':
|
||||
self.data_dir = getDataDir()
|
||||
|
||||
|
||||
@@ -18,7 +18,10 @@ log = CPLog(__name__)
|
||||
|
||||
class Core(Plugin):
|
||||
|
||||
ignore_restart = ['Core.restart', 'Core.shutdown', 'Updater.check']
|
||||
ignore_restart = [
|
||||
'Core.restart', 'Core.shutdown',
|
||||
'Updater.check', 'Updater.autoUpdate',
|
||||
]
|
||||
shutdown_started = False
|
||||
|
||||
def __init__(self):
|
||||
@@ -43,6 +46,7 @@ class Core(Plugin):
|
||||
addEvent('app.base_url', self.createBaseUrl)
|
||||
addEvent('app.api_url', self.createApiUrl)
|
||||
addEvent('app.version', self.version)
|
||||
addEvent('app.load', self.checkDataDir)
|
||||
|
||||
addEvent('setting.save.core.password', self.md5Password)
|
||||
addEvent('setting.save.core.api_key', self.checkApikey)
|
||||
@@ -54,6 +58,12 @@ class Core(Plugin):
|
||||
def checkApikey(self, value):
|
||||
return value if value and len(value) > 3 else uuid4().hex
|
||||
|
||||
def checkDataDir(self):
|
||||
if Env.get('app_dir') in Env.get('data_dir'):
|
||||
log.error('You should NOT use your CouchPotato directory to save your settings in. Files will get overwritten or be deleted.')
|
||||
|
||||
return True
|
||||
|
||||
def available(self):
|
||||
return jsonified({
|
||||
'succes': True
|
||||
|
||||
@@ -20,6 +20,8 @@ log = CPLog(__name__)
|
||||
|
||||
class Updater(Plugin):
|
||||
|
||||
available_notified = False
|
||||
|
||||
def __init__(self):
|
||||
|
||||
if Env.get('desktop'):
|
||||
@@ -64,13 +66,18 @@ class Updater(Plugin):
|
||||
|
||||
fireEventAsync('app.restart')
|
||||
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
def check(self):
|
||||
if self.isDisabled():
|
||||
return
|
||||
|
||||
if self.updater.check():
|
||||
if self.conf('notification') and not self.conf('automatic'):
|
||||
if not self.available_notified and self.conf('notification') and not self.conf('automatic'):
|
||||
fireEvent('updater.available', message = 'A new update is available', data = self.updater.info())
|
||||
self.available_notified = True
|
||||
return True
|
||||
|
||||
return False
|
||||
@@ -168,7 +175,6 @@ class GitUpdater(BaseUpdater):
|
||||
self.repo.saveStash()
|
||||
|
||||
log.info('Updating to latest version')
|
||||
info = self.info()
|
||||
self.repo.pull()
|
||||
|
||||
# Delete leftover .pyc files
|
||||
@@ -302,6 +308,13 @@ class SourceUpdater(BaseUpdater):
|
||||
except Exception, e:
|
||||
log.error('Failed overwriting file: %s', e)
|
||||
|
||||
if Env.get('app_dir') not in Env.get('data_dir'):
|
||||
for still_exists in existing_files:
|
||||
try:
|
||||
os.remove(still_exists)
|
||||
except:
|
||||
log.error('Failed removing non-used file: %s', traceback.format_exc())
|
||||
|
||||
|
||||
def removeDir(self, path):
|
||||
try:
|
||||
|
||||
@@ -22,8 +22,10 @@ var UpdaterBase = new Class({
|
||||
|
||||
if(json.update_available)
|
||||
self.doUpdate();
|
||||
else
|
||||
App.unBlockPage()
|
||||
else {
|
||||
App.unBlockPage();
|
||||
App.fireEvent('message', 'No updates available');
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
|
||||
@@ -136,23 +136,24 @@ class CoreNotifier(Notification):
|
||||
#db.close()
|
||||
return True
|
||||
|
||||
def frontend(self, type = 'notification', data = {}):
|
||||
def frontend(self, type = 'notification', data = {}, message = None):
|
||||
|
||||
self.m_lock.acquire()
|
||||
message = {
|
||||
notification = {
|
||||
'message_id': str(uuid.uuid4()),
|
||||
'time': time.time(),
|
||||
'type': type,
|
||||
'data': data,
|
||||
'message': message,
|
||||
}
|
||||
self.messages.append(message)
|
||||
self.messages.append(notification)
|
||||
|
||||
while len(self.listeners) > 0 and not self.shuttingDown():
|
||||
try:
|
||||
listener, last_id = self.listeners.pop()
|
||||
listener({
|
||||
'success': True,
|
||||
'result': [message],
|
||||
'result': [notification],
|
||||
})
|
||||
except:
|
||||
break
|
||||
|
||||
@@ -11,6 +11,7 @@ var NotificationBase = new Class({
|
||||
App.addEvent('unload', self.stopPoll.bind(self));
|
||||
App.addEvent('reload', self.startInterval.bind(self, [true]));
|
||||
App.addEvent('notification', self.notify.bind(self));
|
||||
App.addEvent('message', self.showMessage.bind(self));
|
||||
|
||||
// Add test buttons to settings page
|
||||
App.addEvent('load', self.addTestButtons.bind(self));
|
||||
@@ -86,7 +87,7 @@ var NotificationBase = new Class({
|
||||
|
||||
startInterval: function(force){
|
||||
var self = this;
|
||||
|
||||
|
||||
if(self.stopped && !force){
|
||||
self.stopped = false;
|
||||
return;
|
||||
@@ -126,11 +127,12 @@ var NotificationBase = new Class({
|
||||
processData: function(json){
|
||||
var self = this;
|
||||
|
||||
|
||||
// Process data
|
||||
if(json){
|
||||
Array.each(json.result, function(result){
|
||||
App.fireEvent(result.type, result)
|
||||
App.fireEvent(result.type, result);
|
||||
if(result.message && result.read === undefined)
|
||||
self.showMessage(result.message);
|
||||
})
|
||||
|
||||
if(json.result.length > 0)
|
||||
@@ -141,6 +143,30 @@ var NotificationBase = new Class({
|
||||
self.startPoll()
|
||||
},
|
||||
|
||||
showMessage: function(message){
|
||||
var self = this;
|
||||
|
||||
if(!self.message_container)
|
||||
self.message_container = new Element('div.messages').inject(document.body);
|
||||
|
||||
var new_message = new Element('div.message', {
|
||||
'text': message
|
||||
}).inject(self.message_container);
|
||||
|
||||
setTimeout(function(){
|
||||
new_message.addClass('show')
|
||||
}, 10);
|
||||
|
||||
setTimeout(function(){
|
||||
new_message.addClass('hide')
|
||||
setTimeout(function(){
|
||||
new_message.destroy();
|
||||
}, 1000);
|
||||
}, 4000);
|
||||
|
||||
},
|
||||
|
||||
// Notification setting tests
|
||||
addTestButtons: function(){
|
||||
var self = this;
|
||||
|
||||
|
||||
@@ -46,10 +46,13 @@ class Plex(Notification):
|
||||
def notify(self, message = '', data = {}, listener = None):
|
||||
if self.isDisabled(): return
|
||||
|
||||
for host in [x.strip() + ':3000' for x in self.conf('host').split(",")]:
|
||||
self.send({'command': 'ExecBuiltIn', 'parameter': 'Notification(CouchPotato, %s)' % message}, host)
|
||||
hosts = [x.strip() + ':3000' for x in self.conf('host').split(",")]
|
||||
successful = 0
|
||||
for host in hosts:
|
||||
if self.send({'command': 'ExecBuiltIn', 'parameter': 'Notification(CouchPotato, %s)' % message}, host):
|
||||
successful += 1
|
||||
|
||||
return True
|
||||
return successful == len(hosts)
|
||||
|
||||
def send(self, command, host):
|
||||
|
||||
@@ -60,7 +63,7 @@ class Plex(Notification):
|
||||
try:
|
||||
self.urlopen(url, headers = headers, show_error = False)
|
||||
except:
|
||||
log.error("Couldn't sent command to Plex")
|
||||
log.error("Couldn't sent command to Plex: %s", traceback.format_exc())
|
||||
return False
|
||||
|
||||
log.info('Plex notification to %s successful.', host)
|
||||
|
||||
@@ -78,15 +78,16 @@ Page.Log = new Class({
|
||||
addColors: function(text){
|
||||
var self = this;
|
||||
|
||||
var text = new Element('div', {
|
||||
'html': text
|
||||
}).get('text')
|
||||
|
||||
text = text.replace(/\u001b\[31m/gi, '</span><span class="error">')
|
||||
text = text.replace(/\u001b\[36m/gi, '</span><span class="debug">')
|
||||
text = text.replace(/\u001b\[33m/gi, '</span><span class="debug">')
|
||||
text = text.replace(/\u001b\[0m\n/gi, '</span><span class="time">')
|
||||
text = text.replace(/\u001b\[0m/gi, '</span><span>')
|
||||
text = text
|
||||
.replace(/&/g, '&')
|
||||
.replace(/</g, '<')
|
||||
.replace(/>/g, '>')
|
||||
.replace(/"/g, '"')
|
||||
.replace(/\u001b\[31m/gi, '</span><span class="error">')
|
||||
.replace(/\u001b\[36m/gi, '</span><span class="debug">')
|
||||
.replace(/\u001b\[33m/gi, '</span><span class="debug">')
|
||||
.replace(/\u001b\[0m\n/gi, '</span><span class="time">')
|
||||
.replace(/\u001b\[0m/gi, '</span><span>')
|
||||
|
||||
return '<span class="time">' + text + '</span>';
|
||||
}
|
||||
|
||||
@@ -240,7 +240,6 @@ class MoviePlugin(Plugin):
|
||||
db = get_session()
|
||||
|
||||
for id in getParam('id').split(','):
|
||||
fireEvent('notify.frontend', type = 'movie.busy.%s' % id, data = True)
|
||||
movie = db.query(Movie).filter_by(id = id).first()
|
||||
|
||||
if movie:
|
||||
@@ -250,6 +249,7 @@ class MoviePlugin(Plugin):
|
||||
for title in movie.library.titles:
|
||||
if title.default: default_title = title.title
|
||||
|
||||
fireEvent('notify.frontend', type = 'movie.busy.%s' % id, data = True, message = 'Updating "%s"' % default_title)
|
||||
fireEventAsync('library.update', identifier = movie.library.identifier, default_title = default_title, force = True, on_complete = self.createOnComplete(id))
|
||||
|
||||
|
||||
@@ -342,7 +342,7 @@ class MoviePlugin(Plugin):
|
||||
onComplete()
|
||||
|
||||
if added:
|
||||
fireEvent('notify.frontend', type = 'movie.added', data = movie_dict)
|
||||
fireEvent('notify.frontend', type = 'movie.added', data = movie_dict, message = 'Successfully added "%s" to your wanted list.' % params.get('title', ''))
|
||||
|
||||
#db.close()
|
||||
return movie_dict
|
||||
|
||||
@@ -17,10 +17,13 @@ var Movie = new Class({
|
||||
self.parent(self, options);
|
||||
|
||||
App.addEvent('movie.update.'+data.id, self.update.bind(self));
|
||||
App.addEvent('movie.busy.'+data.id, function(notification){
|
||||
if(notification.data)
|
||||
self.busy(true)
|
||||
});
|
||||
|
||||
['movie.busy', 'searcher.started'].each(function(listener){
|
||||
App.addEvent(listener+'.'+data.id, function(notification){
|
||||
if(notification.data)
|
||||
self.busy(true)
|
||||
});
|
||||
})
|
||||
},
|
||||
|
||||
busy: function(set_busy){
|
||||
|
||||
@@ -84,7 +84,7 @@ class Searcher(Plugin):
|
||||
if not default_title:
|
||||
return
|
||||
|
||||
fireEvent('notify.frontend', type = 'searcher.started.%s' % movie['id'], data = True)
|
||||
fireEvent('notify.frontend', type = 'searcher.started.%s' % movie['id'], data = True, message = 'Searching for "%s"' % default_title)
|
||||
|
||||
ret = False
|
||||
for quality_type in movie['profile']['types']:
|
||||
|
||||
@@ -89,7 +89,7 @@ class IMDBAPI(MovieProvider):
|
||||
'poster': [movie.get('Poster', '')] if movie.get('Poster') and len(movie.get('Poster', '')) > 4 else [],
|
||||
},
|
||||
'rating': {
|
||||
'imdb': (tryFloat(movie.get('imdbRating', 0)), tryInt(movie.get('imdbVotes', ''))),
|
||||
'imdb': (tryFloat(movie.get('imdbRating', 0)), tryInt(movie.get('imdbVotes', '').replace(',', ''))),
|
||||
#'rotten': (tryFloat(movie.get('tomatoRating', 0)), tryInt(movie.get('tomatoReviews', 0))),
|
||||
},
|
||||
'imdb': str(movie.get('imdbID', '')),
|
||||
|
||||
+13
-6
@@ -20,14 +20,12 @@ import warnings
|
||||
|
||||
def getOptions(base_path, args):
|
||||
|
||||
data_dir = getDataDir()
|
||||
|
||||
# Options
|
||||
parser = ArgumentParser(prog = 'CouchPotato.py')
|
||||
parser.add_argument('--config_file', default = os.path.join(data_dir, 'settings.conf'),
|
||||
dest = 'config_file', help = 'Absolute or ~/ path of the settings file (default ./_data/settings.conf)')
|
||||
parser.add_argument('--data_dir', default = data_dir,
|
||||
parser.add_argument('--data_dir',
|
||||
dest = 'data_dir', help = 'Absolute or ~/ path of the data dir')
|
||||
parser.add_argument('--config_file',
|
||||
dest = 'config_file', help = 'Absolute or ~/ path of the settings file (default DATA_DIR/settings.conf)')
|
||||
parser.add_argument('--debug', action = 'store_true',
|
||||
dest = 'debug', help = 'Debug mode')
|
||||
parser.add_argument('--console_log', action = 'store_true',
|
||||
@@ -36,12 +34,21 @@ def getOptions(base_path, args):
|
||||
dest = 'quiet', help = 'No console logging')
|
||||
parser.add_argument('--daemon', action = 'store_true',
|
||||
dest = 'daemon', help = 'Daemonize the app')
|
||||
parser.add_argument('--pid_file', default = os.path.join(data_dir, 'couchpotato.pid'),
|
||||
parser.add_argument('--pid_file',
|
||||
dest = 'pid_file', help = 'Path to pidfile needed for daemon')
|
||||
|
||||
options = parser.parse_args(args)
|
||||
|
||||
data_dir = os.path.expanduser(options.data_dir if options.data_dir else getDataDir())
|
||||
|
||||
if not options.config_file:
|
||||
options.config_file = os.path.join(data_dir, 'settings.conf')
|
||||
|
||||
if not options.pid_file:
|
||||
options.pid_file = os.path.join(data_dir, 'couchpotato.pid')
|
||||
|
||||
options.config_file = os.path.expanduser(options.config_file)
|
||||
options.pid_file = os.path.expanduser(options.pid_file)
|
||||
|
||||
return options
|
||||
|
||||
|
||||
@@ -80,7 +80,7 @@ var CouchPotato = new Class({
|
||||
}
|
||||
}),
|
||||
new Element('a', {
|
||||
'text': 'Check for updates',
|
||||
'text': 'Update to latest',
|
||||
'events': {
|
||||
'click': self.checkForUpdate.bind(self, null)
|
||||
}
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
/**
|
||||
* StyleFix 1.0.2
|
||||
* StyleFix 1.0.3
|
||||
* @author Lea Verou
|
||||
* MIT license
|
||||
*/
|
||||
@@ -52,8 +52,10 @@ var self = window.StyleFix = {
|
||||
});
|
||||
|
||||
// behavior URLs shoudn’t be converted (Issue #19)
|
||||
css = css.replace(RegExp('\\b(behavior:\\s*?url\\(\'?"?)' + base, 'gi'), '$1');
|
||||
}
|
||||
// base should be escaped before added to RegExp (Issue #81)
|
||||
var escaped_base = base.replace(/([\\\^\$*+[\]?{}.=!:(|)])/g,"\\$1");
|
||||
css = css.replace(RegExp('\\b(behavior:\\s*?url\\(\'?"?)' + escaped_base, 'gi'), '$1');
|
||||
}
|
||||
|
||||
var style = document.createElement('style');
|
||||
style.textContent = css;
|
||||
@@ -63,6 +65,8 @@ var self = window.StyleFix = {
|
||||
|
||||
parent.insertBefore(style, link);
|
||||
parent.removeChild(link);
|
||||
|
||||
style.media = link.media; // Duplicate is intentional. See issue #31
|
||||
}
|
||||
};
|
||||
|
||||
@@ -84,6 +88,9 @@ var self = window.StyleFix = {
|
||||
},
|
||||
|
||||
styleElement: function(style) {
|
||||
if (style.hasAttribute('data-noprefix')) {
|
||||
return;
|
||||
}
|
||||
var disabled = style.disabled;
|
||||
|
||||
style.textContent = self.fix(style.textContent, true, style);
|
||||
@@ -150,7 +157,7 @@ function $(expr, con) {
|
||||
})();
|
||||
|
||||
/**
|
||||
* PrefixFree 1.0.5
|
||||
* PrefixFree 1.0.6
|
||||
* @author Lea Verou
|
||||
* MIT license
|
||||
*/
|
||||
@@ -160,36 +167,39 @@ if(!window.StyleFix || !window.getComputedStyle) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Private helper
|
||||
function fix(what, before, after, replacement, css) {
|
||||
what = self[what];
|
||||
|
||||
if(what.length) {
|
||||
var regex = RegExp(before + '(' + what.join('|') + ')' + after, 'gi');
|
||||
|
||||
css = css.replace(regex, replacement);
|
||||
}
|
||||
|
||||
return css;
|
||||
}
|
||||
|
||||
var self = window.PrefixFree = {
|
||||
prefixCSS: function(css, raw) {
|
||||
var prefix = self.prefix;
|
||||
|
||||
function fix(what, before, after, replacement) {
|
||||
what = self[what];
|
||||
|
||||
if(what.length) {
|
||||
var regex = RegExp(before + '(' + what.join('|') + ')' + after, 'gi');
|
||||
|
||||
css = css.replace(regex, replacement);
|
||||
}
|
||||
}
|
||||
|
||||
fix('functions', '(\\s|:|,)', '\\s*\\(', '$1' + prefix + '$2(');
|
||||
fix('keywords', '(\\s|:)', '(\\s|;|\\}|$)', '$1' + prefix + '$2$3');
|
||||
fix('properties', '(^|\\{|\\s|;)', '\\s*:', '$1' + prefix + '$2:');
|
||||
css = fix('functions', '(\\s|:|,)', '\\s*\\(', '$1' + prefix + '$2(', css);
|
||||
css = fix('keywords', '(\\s|:)', '(\\s|;|\\}|$)', '$1' + prefix + '$2$3', css);
|
||||
css = fix('properties', '(^|\\{|\\s|;)', '\\s*:', '$1' + prefix + '$2:', css);
|
||||
|
||||
// Prefix properties *inside* values (issue #8)
|
||||
if (self.properties.length) {
|
||||
var regex = RegExp('\\b(' + self.properties.join('|') + ')(?!:)', 'gi');
|
||||
|
||||
fix('valueProperties', '\\b', ':(.+?);', function($0) {
|
||||
css = fix('valueProperties', '\\b', ':(.+?);', function($0) {
|
||||
return $0.replace(regex, prefix + "$1")
|
||||
});
|
||||
}, css);
|
||||
}
|
||||
|
||||
if(raw) {
|
||||
fix('selectors', '', '\\b', self.prefixSelector);
|
||||
fix('atrules', '@', '\\b', '@' + prefix + '$1');
|
||||
css = fix('selectors', '', '\\b', self.prefixSelector, css);
|
||||
css = fix('atrules', '@', '\\b', '@' + prefix + '$1', css);
|
||||
}
|
||||
|
||||
// Fix double prefixing
|
||||
@@ -198,11 +208,25 @@ var self = window.PrefixFree = {
|
||||
return css;
|
||||
},
|
||||
|
||||
// Warning: prefixXXX functions prefix no matter what, even if the XXX is supported prefix-less
|
||||
property: function(property) {
|
||||
return (self.properties.indexOf(property)? self.prefix : '') + property;
|
||||
},
|
||||
|
||||
value: function(value, property) {
|
||||
value = fix('functions', '(^|\\s|,)', '\\s*\\(', '$1' + self.prefix + '$2(', value);
|
||||
value = fix('keywords', '(^|\\s)', '(\\s|$)', '$1' + self.prefix + '$2$3', value);
|
||||
|
||||
// TODO properties inside values
|
||||
|
||||
return value;
|
||||
},
|
||||
|
||||
// Warning: Prefixes no matter what, even if the selector is supported prefix-less
|
||||
prefixSelector: function(selector) {
|
||||
return selector.replace(/^:{1,2}/, function($0) { return $0 + self.prefix })
|
||||
},
|
||||
|
||||
// Warning: Prefixes no matter what, even if the property is supported prefix-less
|
||||
prefixProperty: function(property, camelCase) {
|
||||
var prefixed = self.prefix + property;
|
||||
|
||||
@@ -334,7 +358,9 @@ var keywords = {
|
||||
'zoom-out': 'cursor',
|
||||
'box': 'display',
|
||||
'flexbox': 'display',
|
||||
'inline-flexbox': 'display'
|
||||
'inline-flexbox': 'display',
|
||||
'flex': 'display',
|
||||
'inline-flex': 'display'
|
||||
};
|
||||
|
||||
self.functions = [];
|
||||
|
||||
@@ -21,9 +21,9 @@ body {
|
||||
}
|
||||
|
||||
* {
|
||||
-moz-box-sizing: border-box;
|
||||
-webkit-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
-webkit-box-sizing: border-box;
|
||||
-moz-box-sizing: border-box;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
pre {
|
||||
@@ -267,10 +267,9 @@ body > .spinner, .mask{
|
||||
font-size: 8px;
|
||||
margin: -5px 0 0 15px;
|
||||
box-shadow: inset 0 1px 0 rgba(255,255,255,.6), 0 0 3px rgba(0,0,0,.7);
|
||||
background: -webkit-gradient(linear, left bottom, left top, from(rgba(255,255,255,.3)), to(rgba(255,255,255,.1)));
|
||||
background: -moz-linear-gradient(center bottom, rgba(255,255,255,.3) 0%, rgba(255,255,255,.1) 100%);
|
||||
background-color: #1b79b8;
|
||||
text-shadow: none;
|
||||
background-image: linear-gradient(0deg, rgba(255,255,255,.3) 0%, rgba(255,255,255,.1) 100%);
|
||||
}
|
||||
|
||||
.header .notification_menu .wrapper {
|
||||
@@ -354,16 +353,8 @@ body > .spinner, .mask{
|
||||
border-radius:30px;
|
||||
|
||||
box-shadow: 0 1px 1px rgba(0,0,0,0.35), inset 0 1px 0px rgba(255,255,255,0.20);
|
||||
|
||||
background: url('../images/sprite.png') no-repeat 94% -53px, -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
left top,
|
||||
color-stop(0, #406db8),
|
||||
color-stop(1, #5b9bd1)
|
||||
);
|
||||
background: url('../images/sprite.png') no-repeat 94% -53px, -moz-linear-gradient(
|
||||
center top,
|
||||
background: url('../images/sprite.png') no-repeat 94% -53px, linear-gradient(
|
||||
270deg,
|
||||
#5b9bd1 0%,
|
||||
#406db8 100%
|
||||
);
|
||||
@@ -448,15 +439,8 @@ body > .spinner, .mask{
|
||||
border: 1px solid #252930;
|
||||
box-shadow: inset 0 1px 0px rgba(255,255,255,0.20), 0 0 3px rgba(0,0,0, 0.2);
|
||||
background: rgb(55,62,74);
|
||||
background-image: -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
left top,
|
||||
color-stop(0, rgb(55,62,74)),
|
||||
color-stop(1, rgb(73,83,98))
|
||||
);
|
||||
background-image: -moz-linear-gradient(
|
||||
center bottom,
|
||||
background-image: linear-gradient(
|
||||
90deg,
|
||||
rgb(55,62,74) 0%,
|
||||
rgb(73,83,98) 100%
|
||||
);
|
||||
@@ -544,15 +528,8 @@ body > .spinner, .mask{
|
||||
text-align: center;
|
||||
color: #000;
|
||||
text-shadow: none;
|
||||
background-image: -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
right top,
|
||||
color-stop(0, rgb(200,200,200)),
|
||||
color-stop(1, rgb(255,255,255))
|
||||
);
|
||||
background-image: -moz-linear-gradient(
|
||||
left bottom,
|
||||
background-image: linear-gradient(
|
||||
45deg,
|
||||
rgb(200,200,200) 0%,
|
||||
rgb(255,255,255) 100%
|
||||
);
|
||||
@@ -601,4 +578,47 @@ body > .spinner, .mask{
|
||||
}
|
||||
.more_menu .wrapper li a:hover {
|
||||
background: rgba(0,0,0,0.05);
|
||||
}
|
||||
}
|
||||
|
||||
.messages {
|
||||
position: absolute;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
padding: 2px;
|
||||
width: 240px;
|
||||
z-index: 2;
|
||||
overflow: hidden;
|
||||
font-size: 14px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.messages .message {
|
||||
text-align: center;
|
||||
border-radius: 2px;
|
||||
margin: 2px 0 0 0;
|
||||
height: 0;
|
||||
overflow: hidden;
|
||||
transition: all .6s cubic-bezier(0.9,0,0.1,1);
|
||||
box-shadow: 0 1px 1px rgba(0,0,0,0.35), inset 0 1px 0px rgba(255,255,255,0.20);
|
||||
background-image: linear-gradient(
|
||||
270deg,
|
||||
#5b9bd1 0%,
|
||||
#406db8 100%
|
||||
);
|
||||
width: 100%;
|
||||
padding: 0 5px;
|
||||
visibility: hidden;
|
||||
max-height: 0;
|
||||
}
|
||||
.messages .message.show {
|
||||
visibility: visible;
|
||||
height: auto;
|
||||
padding-top: 3px;
|
||||
padding-bottom: 3px;
|
||||
min-height: 1px;
|
||||
max-height: 400px;
|
||||
}
|
||||
.messages .message.hide {
|
||||
margin-left: 240px;
|
||||
opacity: 0;
|
||||
}
|
||||
@@ -16,17 +16,9 @@
|
||||
padding: 40px 0;
|
||||
margin: 0;
|
||||
min-height: 470px;
|
||||
|
||||
background-image: -webkit-gradient(
|
||||
linear,
|
||||
right top,
|
||||
40% 4%,
|
||||
color-stop(0, rgba(0,0,0, 0.3)),
|
||||
color-stop(1, rgba(0,0,0, 0))
|
||||
);
|
||||
background-image: -moz-linear-gradient(
|
||||
10% 0% 16deg,
|
||||
rgba(0,0,0,0) 0%,
|
||||
background-image: linear-gradient(
|
||||
20deg,
|
||||
rgba(0,0,0,0) 50%,
|
||||
rgba(0,0,0,0.3) 100%
|
||||
);
|
||||
}
|
||||
@@ -364,30 +356,16 @@
|
||||
border-radius: 2px;
|
||||
}
|
||||
.page .tag_input > ul:hover > li.choice {
|
||||
background: -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
left top,
|
||||
color-stop(0, rgba(255,255,255,0.1)),
|
||||
color-stop(1, rgba(255,255,255,0.3))
|
||||
);
|
||||
background: -moz-linear-gradient(
|
||||
center top,
|
||||
background: linear-gradient(
|
||||
270deg,
|
||||
rgba(255,255,255,0.3) 0%,
|
||||
rgba(255,255,255,0.1) 100%
|
||||
);
|
||||
}
|
||||
.page .tag_input > ul > li.choice:hover,
|
||||
.page .tag_input > ul > li.choice.selected {
|
||||
background: -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
left top,
|
||||
color-stop(0, #406db8),
|
||||
color-stop(1, #5b9bd1)
|
||||
);
|
||||
background: -moz-linear-gradient(
|
||||
center top,
|
||||
background: linear-gradient(
|
||||
270deg,
|
||||
#5b9bd1 0%,
|
||||
#406db8 100%
|
||||
);
|
||||
@@ -425,15 +403,8 @@
|
||||
margin: -9px 0 0 -16px;
|
||||
border-radius: 30px 30px 0 0;
|
||||
cursor: pointer;
|
||||
background: url('../../images/icon.delete.png') no-repeat center 2px, -webkit-gradient(
|
||||
linear,
|
||||
left bottom,
|
||||
left top,
|
||||
color-stop(0, #5b9bd1),
|
||||
color-stop(1, #5b9bd1)
|
||||
);
|
||||
background: url('../../images/icon.delete.png') no-repeat center 2px, -moz-linear-gradient(
|
||||
center top,
|
||||
background: url('../../images/icon.delete.png') no-repeat center 2px, -webkit-linear-gradient(
|
||||
270deg,
|
||||
#5b9bd1 0%,
|
||||
#5b9bd1 100%
|
||||
);
|
||||
|
||||
@@ -10,9 +10,7 @@
|
||||
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/mootools.js') }}"></script>
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/mootools_more.js') }}"></script>
|
||||
{% if not env.get('dev') %}
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/prefix_free.js') }}"></script>
|
||||
{% endif %}
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/uniform.js') }}"></script>
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/form_replacement/form_check.js') }}"></script>
|
||||
<script type="text/javascript" src="{{ url_for('web.static', filename='scripts/library/form_replacement/form_radio.js') }}"></script>
|
||||
|
||||
Reference in New Issue
Block a user