2011-01-01 11:49:14 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# vi:si:et:sw=4:sts=4:ts=4
|
|
|
|
from __future__ import division, with_statement
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
import re
|
|
|
|
import os
|
|
|
|
import shutil
|
2013-07-14 16:58:01 +00:00
|
|
|
from glob import glob
|
|
|
|
import subprocess
|
2013-07-15 13:47:30 +00:00
|
|
|
from urllib import quote
|
2014-11-20 18:30:12 +00:00
|
|
|
import tempfile
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
import ox
|
2014-02-05 11:27:17 +00:00
|
|
|
from ox.django.fields import DictField, TupleField
|
2013-05-27 20:06:56 +00:00
|
|
|
from django.conf import settings
|
2013-07-18 10:38:10 +00:00
|
|
|
from django.db import models, transaction
|
2013-05-27 20:06:56 +00:00
|
|
|
from django.db.models import Max
|
2011-01-01 11:49:14 +00:00
|
|
|
from django.contrib.auth.models import User
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
from annotation.models import Annotation
|
|
|
|
from item.models import Item
|
2014-02-05 13:21:17 +00:00
|
|
|
from item.utils import get_by_id
|
|
|
|
import clip.models
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
from archive import extract
|
2011-01-01 11:49:14 +00:00
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
import managers
|
2011-01-01 11:49:14 +00:00
|
|
|
|
2011-10-07 08:50:51 +00:00
|
|
|
class Edit(models.Model):
|
2011-01-01 11:49:14 +00:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
unique_together = ("user", "name")
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
objects = managers.EditManager()
|
|
|
|
|
2011-01-01 11:49:14 +00:00
|
|
|
created = models.DateTimeField(auto_now_add=True)
|
|
|
|
modified = models.DateTimeField(auto_now=True)
|
2013-07-17 13:02:21 +00:00
|
|
|
user = models.ForeignKey(User, related_name='edits')
|
2011-01-01 11:49:14 +00:00
|
|
|
name = models.CharField(max_length=255)
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
status = models.CharField(max_length=20, default='private')
|
|
|
|
_status = ['private', 'public', 'featured']
|
|
|
|
description = models.TextField(default='')
|
|
|
|
rightslevel = models.IntegerField(db_index=True, default=0)
|
|
|
|
|
2014-02-05 11:27:17 +00:00
|
|
|
query = DictField(default={"static": True})
|
|
|
|
type = models.CharField(max_length=255, default='static')
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
icon = models.ImageField(default=None, blank=True, null=True,
|
|
|
|
upload_to=lambda i, x: i.path("icon.jpg"))
|
|
|
|
|
|
|
|
poster_frames = TupleField(default=[], editable=False)
|
|
|
|
subscribed_users = models.ManyToManyField(User, related_name='subscribed_edits')
|
|
|
|
|
2011-01-01 11:49:14 +00:00
|
|
|
def __unicode__(self):
|
2013-07-22 17:37:57 +00:00
|
|
|
return u'%s (%s)' % (self.name, self.user)
|
2014-01-22 18:24:21 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get(cls, id):
|
|
|
|
id = id.split(':')
|
|
|
|
username = id[0]
|
|
|
|
name = ":".join(id[1:])
|
|
|
|
return cls.objects.get(user__username=username, name=name)
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
def get_id(self):
|
|
|
|
return u'%s:%s' % (self.user.username, self.name)
|
|
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
|
return ('/edits/%s' % quote(self.get_id())).replace('%3A', ':')
|
|
|
|
|
2013-07-18 10:38:10 +00:00
|
|
|
def add_clip(self, data, index):
|
|
|
|
ids = [i['id'] for i in self.clips.order_by('index').values('id')]
|
2013-05-27 20:06:56 +00:00
|
|
|
clip = Clip(edit=self)
|
2013-07-13 23:00:39 +00:00
|
|
|
if 'annotation' in data and data['annotation']:
|
2013-05-27 20:06:56 +00:00
|
|
|
clip.annotation = Annotation.objects.get(public_id=data['annotation'])
|
2013-07-14 12:47:40 +00:00
|
|
|
clip.item = clip.annotation.item
|
2013-05-27 20:06:56 +00:00
|
|
|
else:
|
2014-09-19 12:26:46 +00:00
|
|
|
clip.item = Item.objects.get(public_id=data['item'])
|
2013-05-27 20:06:56 +00:00
|
|
|
clip.start = data['in']
|
|
|
|
clip.end = data['out']
|
2013-07-18 10:38:10 +00:00
|
|
|
clip.index = index
|
2013-07-13 11:53:13 +00:00
|
|
|
# dont add clip if in/out are invalid
|
|
|
|
if not clip.annotation:
|
|
|
|
duration = clip.item.sort.duration
|
2015-03-26 14:58:07 +00:00
|
|
|
if clip.start > clip.end \
|
|
|
|
or round(clip.start, 3) >= round(duration, 3) \
|
|
|
|
or round(clip.end, 3) > round(duration, 3):
|
2013-07-13 11:53:13 +00:00
|
|
|
return False
|
2013-05-27 20:06:56 +00:00
|
|
|
clip.save()
|
2013-07-18 10:38:10 +00:00
|
|
|
ids.insert(index, clip.id)
|
|
|
|
index = 0
|
|
|
|
with transaction.commit_on_success():
|
|
|
|
for i in ids:
|
|
|
|
Clip.objects.filter(id=i).update(index=index)
|
|
|
|
index += 1
|
2013-05-27 20:06:56 +00:00
|
|
|
return clip
|
|
|
|
|
|
|
|
def accessible(self, user):
|
|
|
|
return self.user == user or self.status in ('public', 'featured')
|
2011-01-01 11:49:14 +00:00
|
|
|
|
|
|
|
def editable(self, user):
|
2013-05-27 20:06:56 +00:00
|
|
|
if not user or user.is_anonymous():
|
|
|
|
return False
|
|
|
|
if self.user == user or \
|
|
|
|
user.is_staff or \
|
|
|
|
user.get_profile().capability('canEditFeaturedEdits') == True:
|
2011-01-01 11:49:14 +00:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
def edit(self, data, user):
|
|
|
|
for key in data:
|
|
|
|
if key == 'status':
|
|
|
|
value = data[key]
|
|
|
|
if value not in self._status:
|
|
|
|
value = self._status[0]
|
|
|
|
if value == 'private':
|
|
|
|
for user in self.subscribed_users.all():
|
|
|
|
self.subscribed_users.remove(user)
|
|
|
|
qs = Position.objects.filter(user=user,
|
|
|
|
section='section', edit=self)
|
|
|
|
if qs.count() > 1:
|
|
|
|
pos = qs[0]
|
|
|
|
pos.section = 'personal'
|
|
|
|
pos.save()
|
|
|
|
elif value == 'featured':
|
|
|
|
if user.get_profile().capability('canEditFeaturedEdits'):
|
|
|
|
pos, created = Position.objects.get_or_create(edit=self, user=user,
|
|
|
|
section='featured')
|
|
|
|
if created:
|
|
|
|
qs = Position.objects.filter(user=user, section='featured')
|
|
|
|
pos.position = qs.aggregate(Max('position'))['position__max'] + 1
|
|
|
|
pos.save()
|
|
|
|
Position.objects.filter(edit=self).exclude(id=pos.id).delete()
|
|
|
|
else:
|
|
|
|
value = self.status
|
|
|
|
elif self.status == 'featured' and value == 'public':
|
|
|
|
Position.objects.filter(edit=self).delete()
|
|
|
|
pos, created = Position.objects.get_or_create(edit=self,
|
|
|
|
user=self.user,section='personal')
|
|
|
|
qs = Position.objects.filter(user=self.user,
|
|
|
|
section='personal')
|
|
|
|
pos.position = qs.aggregate(Max('position'))['position__max'] + 1
|
|
|
|
pos.save()
|
|
|
|
for u in self.subscribed_users.all():
|
|
|
|
pos, created = Position.objects.get_or_create(edit=self, user=u,
|
|
|
|
section='public')
|
|
|
|
qs = Position.objects.filter(user=u, section='public')
|
|
|
|
pos.position = qs.aggregate(Max('position'))['position__max'] + 1
|
|
|
|
pos.save()
|
|
|
|
|
|
|
|
self.status = value
|
|
|
|
elif key == 'name':
|
|
|
|
data['name'] = re.sub(' \[\d+\]$', '', data['name']).strip()
|
|
|
|
if not data['name']:
|
|
|
|
data['name'] = "Untitled"
|
|
|
|
name = data['name']
|
|
|
|
num = 1
|
|
|
|
while Edit.objects.filter(name=name, user=self.user).exclude(id=self.id).count()>0:
|
|
|
|
num += 1
|
|
|
|
name = data['name'] + ' [%d]' % num
|
|
|
|
self.name = name
|
|
|
|
elif key == 'description':
|
|
|
|
self.description = ox.sanitize_html(data['description'])
|
|
|
|
elif key == 'rightslevel':
|
|
|
|
self.rightslevel = int(data['rightslevel'])
|
2014-02-05 13:21:17 +00:00
|
|
|
if key == 'query' and not data['query']:
|
|
|
|
setattr(self, key, {"static":True})
|
|
|
|
elif key == 'query':
|
|
|
|
setattr(self, key, data[key])
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
if 'position' in data:
|
|
|
|
pos, created = Position.objects.get_or_create(edit=self, user=user)
|
|
|
|
pos.position = data['position']
|
|
|
|
pos.section = 'featured'
|
|
|
|
if self.status == 'private':
|
|
|
|
pos.section = 'personal'
|
|
|
|
pos.save()
|
|
|
|
if 'type' in data:
|
2014-02-05 11:27:17 +00:00
|
|
|
if data['type'] == 'static':
|
|
|
|
self.query = {"static":True}
|
|
|
|
self.type = 'static'
|
|
|
|
else:
|
2014-02-05 13:21:17 +00:00
|
|
|
self.type = 'smart'
|
2014-02-05 11:27:17 +00:00
|
|
|
if self.query.get('static', False):
|
2014-02-05 14:35:59 +00:00
|
|
|
self.query = {'conditions': [], 'operator': '&'}
|
2013-05-27 20:06:56 +00:00
|
|
|
if 'posterFrames' in data:
|
|
|
|
self.poster_frames = tuple(data['posterFrames'])
|
|
|
|
self.save()
|
|
|
|
if 'posterFrames' in data:
|
|
|
|
self.update_icon()
|
|
|
|
|
|
|
|
def path(self, name=''):
|
|
|
|
h = "%07d" % self.id
|
|
|
|
return os.path.join('edits', h[:2], h[2:4], h[4:6], h[6:], name)
|
2011-01-01 11:49:14 +00:00
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
def get_items(self, user=None):
|
2014-02-05 13:21:17 +00:00
|
|
|
if self.type == 'static':
|
|
|
|
return Item.objects.filter(editclip__id__in=self.clips.all()).distinct()
|
|
|
|
else:
|
|
|
|
return Item.objects.find({'query': self.query}, user)
|
|
|
|
|
|
|
|
def get_clips(self, user=None):
|
|
|
|
if self.type == 'static':
|
2014-02-05 13:50:23 +00:00
|
|
|
clips = self.clips.all()
|
2014-02-05 13:21:17 +00:00
|
|
|
else:
|
2014-02-05 14:35:59 +00:00
|
|
|
clips = clip.models.Clip.objects.find({'query': self.clip_query()}, user)
|
2014-02-06 14:22:53 +00:00
|
|
|
clips = clips.filter(item__in=self.get_items(user))
|
2014-02-05 13:50:23 +00:00
|
|
|
return clips
|
|
|
|
|
|
|
|
def get_clips_json(self, user=None):
|
2014-06-06 11:38:03 +00:00
|
|
|
qs = self.get_clips(user)
|
2014-02-05 13:50:23 +00:00
|
|
|
if self.type == 'static':
|
|
|
|
clips = [c.json(user) for c in qs.order_by('index')]
|
|
|
|
else:
|
2014-02-07 04:41:01 +00:00
|
|
|
if qs.count() <= 1000:
|
2014-02-05 13:50:23 +00:00
|
|
|
clips = [c.edit_json(user) for c in qs]
|
|
|
|
index = 0
|
|
|
|
for c in clips:
|
|
|
|
c['index'] = index
|
|
|
|
index += 1
|
|
|
|
else:
|
|
|
|
clips = []
|
2014-02-05 13:21:17 +00:00
|
|
|
return clips
|
|
|
|
|
|
|
|
def clip_query(self):
|
2014-02-07 10:13:49 +00:00
|
|
|
def get_conditions(conditions):
|
|
|
|
clip_conditions = []
|
|
|
|
for condition in conditions:
|
|
|
|
if 'conditions' in condition:
|
|
|
|
clip_conditions.append({
|
|
|
|
'operator': condition.get('operator', '&'),
|
|
|
|
'conditions': get_conditions(condition['conditions'])
|
|
|
|
})
|
|
|
|
elif condition['key'] == 'annotations' or \
|
|
|
|
get_by_id(settings.CONFIG['layers'], condition['key']):
|
|
|
|
clip_conditions.append(condition)
|
|
|
|
return clip_conditions
|
|
|
|
|
|
|
|
return {
|
|
|
|
'conditions': get_conditions(self.query.get('conditions', [])),
|
2014-02-05 14:35:59 +00:00
|
|
|
'operator': self.query.get('operator', '&')
|
2014-02-05 13:21:17 +00:00
|
|
|
}
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
def update_icon(self):
|
|
|
|
frames = []
|
|
|
|
if not self.poster_frames:
|
|
|
|
items = self.get_items(self.user).filter(rendered=True)
|
|
|
|
if items.count():
|
|
|
|
poster_frames = []
|
|
|
|
for i in range(0, items.count(), max(1, int(items.count()/4))):
|
|
|
|
poster_frames.append({
|
2014-09-19 12:26:46 +00:00
|
|
|
'item': items[int(i)].public_id,
|
2013-05-27 20:06:56 +00:00
|
|
|
'position': items[int(i)].poster_frame
|
|
|
|
})
|
|
|
|
self.poster_frames = tuple(poster_frames)
|
|
|
|
self.save()
|
|
|
|
for i in self.poster_frames:
|
2014-09-19 12:26:46 +00:00
|
|
|
qs = Item.objects.filter(public_id=i['item'])
|
2013-05-27 20:06:56 +00:00
|
|
|
if qs.count() > 0:
|
|
|
|
frame = qs[0].frame(i['position'])
|
|
|
|
if frame:
|
|
|
|
frames.append(frame)
|
|
|
|
self.icon.name = self.path('icon.jpg')
|
|
|
|
icon = self.icon.path
|
|
|
|
if frames:
|
|
|
|
while len(frames) < 4:
|
|
|
|
frames += frames
|
|
|
|
folder = os.path.dirname(icon)
|
|
|
|
ox.makedirs(folder)
|
|
|
|
for f in glob("%s/icon*.jpg" % folder):
|
|
|
|
os.unlink(f)
|
|
|
|
cmd = [
|
|
|
|
settings.LIST_ICON,
|
|
|
|
'-f', ','.join(frames),
|
|
|
|
'-o', icon
|
|
|
|
]
|
2014-09-02 14:24:12 +00:00
|
|
|
p = subprocess.Popen(cmd, close_fds=True)
|
2013-05-27 20:06:56 +00:00
|
|
|
p.wait()
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
def get_icon(self, size=16):
|
|
|
|
path = self.path('icon%d.jpg' % size)
|
|
|
|
path = os.path.join(settings.MEDIA_ROOT, path)
|
|
|
|
if not os.path.exists(path):
|
|
|
|
folder = os.path.dirname(path)
|
|
|
|
ox.makedirs(folder)
|
|
|
|
if self.icon and os.path.exists(self.icon.path):
|
|
|
|
source = self.icon.path
|
|
|
|
max_size = min(self.icon.width, self.icon.height)
|
|
|
|
else:
|
|
|
|
source = os.path.join(settings.STATIC_ROOT, 'jpg/list256.jpg')
|
|
|
|
max_size = 256
|
|
|
|
if size < max_size:
|
|
|
|
extract.resize_image(source, path, size=size)
|
|
|
|
else:
|
|
|
|
path = source
|
|
|
|
return path
|
|
|
|
|
|
|
|
def json(self, keys=None, user=None):
|
|
|
|
if not keys:
|
|
|
|
keys=[
|
2014-02-05 11:27:17 +00:00
|
|
|
'clips',
|
2013-05-27 20:06:56 +00:00
|
|
|
'description',
|
2014-02-05 11:27:17 +00:00
|
|
|
'duration',
|
2013-05-27 20:06:56 +00:00
|
|
|
'editable',
|
|
|
|
'id',
|
2013-07-13 11:15:46 +00:00
|
|
|
'items',
|
2013-05-27 20:06:56 +00:00
|
|
|
'name',
|
|
|
|
'posterFrames',
|
2014-02-05 11:27:17 +00:00
|
|
|
'query',
|
|
|
|
'rightslevel',
|
2013-05-27 20:06:56 +00:00
|
|
|
'status',
|
|
|
|
'subscribed',
|
2014-02-05 11:27:17 +00:00
|
|
|
'type',
|
|
|
|
'user',
|
2013-05-27 20:06:56 +00:00
|
|
|
]
|
2014-02-05 13:21:17 +00:00
|
|
|
response = {
|
|
|
|
}
|
2013-05-27 20:06:56 +00:00
|
|
|
_map = {
|
|
|
|
'posterFrames': 'poster_frames'
|
|
|
|
}
|
2014-11-10 10:39:38 +00:00
|
|
|
if 'clips' in keys:
|
2014-02-05 13:50:23 +00:00
|
|
|
clips = self.get_clips_json(user)
|
2014-11-21 14:15:42 +00:00
|
|
|
clips_qs = self.get_clips(user)
|
2014-02-05 13:21:17 +00:00
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
for key in keys:
|
|
|
|
if key == 'id':
|
|
|
|
response[key] = self.get_id()
|
2013-07-13 11:15:46 +00:00
|
|
|
elif key == 'items':
|
2014-11-21 14:15:42 +00:00
|
|
|
response[key] = clips_qs.count()
|
2014-02-05 11:27:17 +00:00
|
|
|
elif key == 'query':
|
|
|
|
if not self.query.get('static', False):
|
|
|
|
response[key] = self.query
|
2013-05-27 20:06:56 +00:00
|
|
|
elif key == 'clips':
|
2014-02-05 13:21:17 +00:00
|
|
|
response[key] = clips
|
2013-07-13 11:15:46 +00:00
|
|
|
elif key == 'duration':
|
2014-11-10 10:39:38 +00:00
|
|
|
if self.type == 'static':
|
|
|
|
response[key] = sum([(c['annotation__end'] or c['end']) - (c['annotation__start'] or c['start'])
|
2014-11-21 14:15:42 +00:00
|
|
|
for c in clips_qs.values('start', 'end', 'annotation__start', 'annotation__end')])
|
2014-11-10 10:39:38 +00:00
|
|
|
else:
|
2014-11-21 14:15:42 +00:00
|
|
|
response[key] = sum([c['end'] - c['start'] for c in clips_qs.values('start', 'end')])
|
2013-05-27 20:06:56 +00:00
|
|
|
elif key == 'editable':
|
|
|
|
response[key] = self.editable(user)
|
|
|
|
elif key == 'user':
|
|
|
|
response[key] = self.user.username
|
|
|
|
elif key == 'subscribers':
|
|
|
|
response[key] = self.subscribed_users.all().count()
|
|
|
|
elif key == 'subscribed':
|
|
|
|
if user and not user.is_anonymous():
|
|
|
|
response[key] = self.subscribed_users.filter(id=user.id).exists()
|
|
|
|
elif hasattr(self, _map.get(key, key)):
|
|
|
|
response[key] = getattr(self, _map.get(key,key))
|
|
|
|
return response
|
|
|
|
|
|
|
|
def render(self):
|
|
|
|
#creating a new file from clips
|
|
|
|
tmp = tempfile.mkdtemp()
|
|
|
|
clips = []
|
|
|
|
for clip in self.clips.all().order_by('index'):
|
|
|
|
data = clip.json()
|
|
|
|
clips.append(os.path.join(tmp, '%06d.webm' % data['index']))
|
2014-11-20 18:30:12 +00:00
|
|
|
path = clip.item.streams()[0].media.path
|
2013-05-27 20:06:56 +00:00
|
|
|
cmd = ['avconv', '-i', path,
|
|
|
|
'-ss', data['in'], '-t', data['out'],
|
|
|
|
'-vcodec', 'copy', '-acodec', 'copy',
|
|
|
|
clips[-1]]
|
2014-09-02 14:24:12 +00:00
|
|
|
#p = subprocess.Popen(cmd, close_fds=True)
|
2013-05-27 20:06:56 +00:00
|
|
|
#p.wait()
|
|
|
|
cmd = ['mkvmerge', clips[0]] \
|
|
|
|
+ ['+'+c for c in clips[1:]] \
|
|
|
|
+ [os.path.join(tmp, 'render.webm')]
|
2014-09-02 14:24:12 +00:00
|
|
|
#p = subprocess.Popen(cmd, close_fds=True)
|
2013-05-27 20:06:56 +00:00
|
|
|
#p.wait()
|
|
|
|
shutil.rmtree(tmp)
|
2011-01-01 11:49:14 +00:00
|
|
|
|
|
|
|
class Clip(models.Model):
|
|
|
|
created = models.DateTimeField(auto_now_add=True)
|
|
|
|
modified = models.DateTimeField(auto_now=True)
|
2011-10-07 08:50:51 +00:00
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
edit = models.ForeignKey(Edit, related_name='clips')
|
|
|
|
index = models.IntegerField(default=0)
|
|
|
|
item = models.ForeignKey(Item, null=True, default=None, related_name='editclip')
|
|
|
|
annotation = models.ForeignKey(Annotation, null=True, default=None, related_name='editclip')
|
2011-01-01 11:49:14 +00:00
|
|
|
start = models.FloatField(default=0)
|
2011-10-07 08:50:51 +00:00
|
|
|
end = models.FloatField(default=0)
|
2013-07-14 12:47:40 +00:00
|
|
|
duration = models.FloatField(default=0)
|
|
|
|
|
|
|
|
hue = models.FloatField(default=0)
|
|
|
|
saturation= models.FloatField(default=0)
|
|
|
|
lightness= models.FloatField(default=0)
|
|
|
|
volume = models.FloatField(default=0)
|
2014-02-12 12:42:25 +00:00
|
|
|
sortvalue = models.CharField(max_length=1000, null=True, db_index=True)
|
2011-01-01 11:49:14 +00:00
|
|
|
|
2014-02-07 06:47:06 +00:00
|
|
|
objects = managers.ClipManager()
|
|
|
|
|
2011-01-01 11:49:14 +00:00
|
|
|
def __unicode__(self):
|
2013-05-27 20:06:56 +00:00
|
|
|
if self.annotation:
|
|
|
|
return u'%s' % self.annotation.public_id
|
2014-09-19 12:26:46 +00:00
|
|
|
return u'%s/%0.3f-%0.3f' % (self.item.public_id, self.start, self.end)
|
2013-07-14 12:47:40 +00:00
|
|
|
|
|
|
|
def get_id(self):
|
|
|
|
return ox.toAZ(self.id)
|
|
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
if self.duration != self.end - self.start:
|
|
|
|
self.update_calculated_values()
|
2014-02-12 12:42:25 +00:00
|
|
|
sortvalue = ''
|
|
|
|
if self.id:
|
|
|
|
for l in settings.CONFIG.get('clipLayers', []):
|
|
|
|
sortvalue += ''.join(filter(lambda s: s,
|
|
|
|
[a.sortvalue
|
|
|
|
for a in self.get_annotations().filter(layer=l)]))
|
|
|
|
if sortvalue:
|
|
|
|
self.sortvalue = sortvalue[:900]
|
|
|
|
else:
|
|
|
|
self.sortvalue = None
|
2013-07-14 12:47:40 +00:00
|
|
|
super(Clip, self).save(*args, **kwargs)
|
|
|
|
|
|
|
|
def update_calculated_values(self):
|
|
|
|
start = self.start
|
|
|
|
end = self.end
|
|
|
|
self.duration = end - start
|
|
|
|
if int(end*25) - int(start*25) > 0:
|
|
|
|
self.hue, self.saturation, self.lightness = extract.average_color(
|
|
|
|
self.item.timeline_prefix, self.start, self.end)
|
|
|
|
self.volume = extract.average_volume(self.item.timeline_prefix, self.start, self.end)
|
|
|
|
else:
|
|
|
|
self.hue = self.saturation = self.lightness = 0
|
|
|
|
self.volume = 0
|
2013-05-27 20:06:56 +00:00
|
|
|
|
|
|
|
def json(self, user=None):
|
|
|
|
data = {
|
2013-07-14 12:47:40 +00:00
|
|
|
'id': self.get_id(),
|
2013-05-27 20:06:56 +00:00
|
|
|
'index': self.index
|
|
|
|
}
|
|
|
|
if self.annotation:
|
2013-07-15 13:47:30 +00:00
|
|
|
data['annotation'] = self.annotation.public_id
|
2014-09-19 12:26:46 +00:00
|
|
|
data['item'] = self.item.public_id
|
2013-05-27 20:06:56 +00:00
|
|
|
data['in'] = self.annotation.start
|
|
|
|
data['out'] = self.annotation.end
|
2013-07-09 22:49:01 +00:00
|
|
|
data['parts'] = self.annotation.item.json['parts']
|
|
|
|
data['durations'] = self.annotation.item.json['durations']
|
2013-05-27 20:06:56 +00:00
|
|
|
else:
|
2014-09-19 12:26:46 +00:00
|
|
|
data['item'] = self.item.public_id
|
2013-05-27 20:06:56 +00:00
|
|
|
data['in'] = self.start
|
|
|
|
data['out'] = self.end
|
2013-07-09 22:49:01 +00:00
|
|
|
data['parts'] = self.item.json['parts']
|
|
|
|
data['durations'] = self.item.json['durations']
|
2013-07-15 11:44:05 +00:00
|
|
|
for key in ('title', 'director', 'year', 'videoRatio'):
|
|
|
|
value = self.item.json.get(key)
|
|
|
|
if value:
|
|
|
|
data[key] = value
|
2013-05-27 20:06:56 +00:00
|
|
|
data['duration'] = data['out'] - data['in']
|
2014-12-23 12:49:59 +00:00
|
|
|
data['cuts'] = tuple([c for c in self.item.get('cuts', []) if c > self.start and c < self.end])
|
2014-01-22 18:24:21 +00:00
|
|
|
data['layers'] = self.get_layers(user)
|
2013-05-27 20:06:56 +00:00
|
|
|
return data
|
|
|
|
|
2014-02-12 12:42:25 +00:00
|
|
|
def get_annotations(self):
|
|
|
|
if self.annotation:
|
|
|
|
start = self.annotation.start
|
|
|
|
end = self.annotation.end
|
|
|
|
item = self.annotation.item
|
|
|
|
else:
|
|
|
|
start = self.start
|
|
|
|
end = self.end
|
|
|
|
item = self.item
|
|
|
|
qs = Annotation.objects.filter(item=item).order_by('start', 'sortvalue')
|
|
|
|
qs = qs.filter(start__lt=end, end__gt=start)
|
|
|
|
return qs
|
|
|
|
|
2014-01-22 18:24:21 +00:00
|
|
|
def get_layers(self, user=None):
|
|
|
|
if self.annotation:
|
|
|
|
start = self.annotation.start
|
|
|
|
end = self.annotation.end
|
|
|
|
item = self.annotation.item
|
|
|
|
else:
|
|
|
|
start = self.start
|
|
|
|
end = self.end
|
|
|
|
item = self.item
|
|
|
|
layers = {}
|
|
|
|
for l in settings.CONFIG['layers']:
|
|
|
|
name = l['id']
|
|
|
|
ll = layers.setdefault(name, [])
|
|
|
|
qs = Annotation.objects.filter(layer=name, item=item).order_by(
|
|
|
|
'start', 'end', 'sortvalue')
|
|
|
|
if name == 'subtitles':
|
|
|
|
qs = qs.exclude(value='')
|
|
|
|
qs = qs.filter(start__lt=end, end__gt=start)
|
|
|
|
if l.get('private'):
|
|
|
|
if user and user.is_anonymous():
|
|
|
|
user = None
|
|
|
|
qs = qs.filter(user=user)
|
|
|
|
for a in qs.order_by('start'):
|
|
|
|
ll.append(a.json(user=user))
|
|
|
|
return layers
|
|
|
|
|
2013-05-27 20:06:56 +00:00
|
|
|
class Position(models.Model):
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
unique_together = ("user", "edit", "section")
|
|
|
|
|
|
|
|
edit = models.ForeignKey(Edit, related_name='position')
|
|
|
|
user = models.ForeignKey(User, related_name='edit_position')
|
|
|
|
section = models.CharField(max_length='255')
|
|
|
|
position = models.IntegerField(default=0)
|
|
|
|
|
|
|
|
def __unicode__(self):
|
|
|
|
return u'%s/%s/%s' % (self.section, self.position, self.edit)
|
|
|
|
|