mirror of
				https://github.com/Alexander-D-Karpov/akarpov
				synced 2025-11-04 03:27:24 +03:00 
			
		
		
		
	added multiple file upload
This commit is contained in:
		
							parent
							
								
									e072f71d9b
								
							
						
					
					
						commit
						26a523a9b5
					
				| 
						 | 
				
			
			@ -64,7 +64,7 @@ def check_permissions(self, request):
 | 
			
		|||
class MyChunkedUploadCompleteView(ChunkedUploadCompleteView):
 | 
			
		||||
    def __init__(self, **kwargs):
 | 
			
		||||
        super().__init__(**kwargs)
 | 
			
		||||
        self.message = "file is successfully uploaded"
 | 
			
		||||
        self.message = {}
 | 
			
		||||
 | 
			
		||||
    model = ChunkedUpload
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -76,15 +76,22 @@ def check_permissions(self, request):
 | 
			
		|||
 | 
			
		||||
    def on_completion(self, uploaded_file, request):
 | 
			
		||||
        if uploaded_file.size <= request.user.left_file_upload:
 | 
			
		||||
            File.objects.create(
 | 
			
		||||
            f = File.objects.create(
 | 
			
		||||
                user=request.user, file=uploaded_file, name=uploaded_file.name
 | 
			
		||||
            )
 | 
			
		||||
            request.user.left_file_upload -= uploaded_file.size
 | 
			
		||||
            request.user.save()
 | 
			
		||||
            self.message = {
 | 
			
		||||
                "message": f"File {f.file.name.split('/')[-1]} successfully uploaded",
 | 
			
		||||
                "status": True,
 | 
			
		||||
            }
 | 
			
		||||
        else:
 | 
			
		||||
            self.message = "File is too large"
 | 
			
		||||
            self.message = {
 | 
			
		||||
                "message": "File is too large, please increase disk space",
 | 
			
		||||
                "status": False,
 | 
			
		||||
            }
 | 
			
		||||
        if os.path.isfile(uploaded_file.file.path):
 | 
			
		||||
            os.remove(uploaded_file.file.path)
 | 
			
		||||
 | 
			
		||||
    def get_response_data(self, chunked_upload, request):
 | 
			
		||||
        return {"message": self.message}
 | 
			
		||||
        return self.message
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										625
									
								
								akarpov/static/js/notify.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										625
									
								
								akarpov/static/js/notify.js
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,625 @@
 | 
			
		|||
/* Notify.js - http://notifyjs.com/ Copyright (c) 2015 MIT */
 | 
			
		||||
(function (factory) {
 | 
			
		||||
	// UMD start
 | 
			
		||||
	// https://github.com/umdjs/umd/blob/master/jqueryPluginCommonjs.js
 | 
			
		||||
	if (typeof define === 'function' && define.amd) {
 | 
			
		||||
		// AMD. Register as an anonymous module.
 | 
			
		||||
		define(['jquery'], factory);
 | 
			
		||||
	} else if (typeof module === 'object' && module.exports) {
 | 
			
		||||
		// Node/CommonJS
 | 
			
		||||
		module.exports = function( root, jQuery ) {
 | 
			
		||||
			if ( jQuery === undefined ) {
 | 
			
		||||
				// require('jQuery') returns a factory that requires window to
 | 
			
		||||
				// build a jQuery instance, we normalize how we use modules
 | 
			
		||||
				// that require this pattern but the window provided is a noop
 | 
			
		||||
				// if it's defined (how jquery works)
 | 
			
		||||
				if ( typeof window !== 'undefined' ) {
 | 
			
		||||
					jQuery = require('jquery');
 | 
			
		||||
				}
 | 
			
		||||
				else {
 | 
			
		||||
					jQuery = require('jquery')(root);
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			factory(jQuery);
 | 
			
		||||
			return jQuery;
 | 
			
		||||
		};
 | 
			
		||||
	} else {
 | 
			
		||||
		// Browser globals
 | 
			
		||||
		factory(jQuery);
 | 
			
		||||
	}
 | 
			
		||||
}(function ($) {
 | 
			
		||||
	//IE8 indexOf polyfill
 | 
			
		||||
	var indexOf = [].indexOf || function(item) {
 | 
			
		||||
		for (var i = 0, l = this.length; i < l; i++) {
 | 
			
		||||
			if (i in this && this[i] === item) {
 | 
			
		||||
				return i;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		return -1;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var pluginName = "notify";
 | 
			
		||||
	var pluginClassName = pluginName + "js";
 | 
			
		||||
	var blankFieldName = pluginName + "!blank";
 | 
			
		||||
 | 
			
		||||
	var positions = {
 | 
			
		||||
		t: "top",
 | 
			
		||||
		m: "middle",
 | 
			
		||||
		b: "bottom",
 | 
			
		||||
		l: "left",
 | 
			
		||||
		c: "center",
 | 
			
		||||
		r: "right"
 | 
			
		||||
	};
 | 
			
		||||
	var hAligns = ["l", "c", "r"];
 | 
			
		||||
	var vAligns = ["t", "m", "b"];
 | 
			
		||||
	var mainPositions = ["t", "b", "l", "r"];
 | 
			
		||||
	var opposites = {
 | 
			
		||||
		t: "b",
 | 
			
		||||
		m: null,
 | 
			
		||||
		b: "t",
 | 
			
		||||
		l: "r",
 | 
			
		||||
		c: null,
 | 
			
		||||
		r: "l"
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var parsePosition = function(str) {
 | 
			
		||||
		var pos;
 | 
			
		||||
		pos = [];
 | 
			
		||||
		$.each(str.split(/\W+/), function(i, word) {
 | 
			
		||||
			var w;
 | 
			
		||||
			w = word.toLowerCase().charAt(0);
 | 
			
		||||
			if (positions[w]) {
 | 
			
		||||
				return pos.push(w);
 | 
			
		||||
			}
 | 
			
		||||
		});
 | 
			
		||||
		return pos;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var styles = {};
 | 
			
		||||
 | 
			
		||||
	var coreStyle = {
 | 
			
		||||
		name: "core",
 | 
			
		||||
		html: "<div class=\"" + pluginClassName + "-wrapper\">\n	<div class=\"" + pluginClassName + "-arrow\"></div>\n	<div class=\"" + pluginClassName + "-container\"></div>\n</div>",
 | 
			
		||||
		css: "." + pluginClassName + "-corner {\n	position: fixed;\n	margin: 5px;\n	z-index: 1050;\n}\n\n." + pluginClassName + "-corner ." + pluginClassName + "-wrapper,\n." + pluginClassName + "-corner ." + pluginClassName + "-container {\n	position: relative;\n	display: block;\n	height: inherit;\n	width: inherit;\n	margin: 3px;\n}\n\n." + pluginClassName + "-wrapper {\n	z-index: 1;\n	position: absolute;\n	display: inline-block;\n	height: 0;\n	width: 0;\n}\n\n." + pluginClassName + "-container {\n	display: none;\n	z-index: 1;\n	position: absolute;\n}\n\n." + pluginClassName + "-hidable {\n	cursor: pointer;\n}\n\n[data-notify-text],[data-notify-html] {\n	position: relative;\n}\n\n." + pluginClassName + "-arrow {\n	position: absolute;\n	z-index: 2;\n	width: 0;\n	height: 0;\n}"
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var stylePrefixes = {
 | 
			
		||||
		"border-radius": ["-webkit-", "-moz-"]
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var getStyle = function(name) {
 | 
			
		||||
		return styles[name];
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var removeStyle = function(name) {
 | 
			
		||||
		if (!name) {
 | 
			
		||||
			throw "Missing Style name";
 | 
			
		||||
		}
 | 
			
		||||
		if (styles[name]) {
 | 
			
		||||
			delete styles[name];
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var addStyle = function(name, def) {
 | 
			
		||||
		if (!name) {
 | 
			
		||||
			throw "Missing Style name";
 | 
			
		||||
		}
 | 
			
		||||
		if (!def) {
 | 
			
		||||
			throw "Missing Style definition";
 | 
			
		||||
		}
 | 
			
		||||
		if (!def.html) {
 | 
			
		||||
			throw "Missing Style HTML";
 | 
			
		||||
		}
 | 
			
		||||
		//remove existing style
 | 
			
		||||
		var existing = styles[name];
 | 
			
		||||
		if (existing && existing.cssElem) {
 | 
			
		||||
			if (window.console) {
 | 
			
		||||
				console.warn(pluginName + ": overwriting style '" + name + "'");
 | 
			
		||||
			}
 | 
			
		||||
			styles[name].cssElem.remove();
 | 
			
		||||
		}
 | 
			
		||||
		def.name = name;
 | 
			
		||||
		styles[name] = def;
 | 
			
		||||
		var cssText = "";
 | 
			
		||||
		if (def.classes) {
 | 
			
		||||
			$.each(def.classes, function(className, props) {
 | 
			
		||||
				cssText += "." + pluginClassName + "-" + def.name + "-" + className + " {\n";
 | 
			
		||||
				$.each(props, function(name, val) {
 | 
			
		||||
					if (stylePrefixes[name]) {
 | 
			
		||||
						$.each(stylePrefixes[name], function(i, prefix) {
 | 
			
		||||
							return cssText += "	" + prefix + name + ": " + val + ";\n";
 | 
			
		||||
						});
 | 
			
		||||
					}
 | 
			
		||||
					return cssText += "	" + name + ": " + val + ";\n";
 | 
			
		||||
				});
 | 
			
		||||
				return cssText += "}\n";
 | 
			
		||||
			});
 | 
			
		||||
		}
 | 
			
		||||
		if (def.css) {
 | 
			
		||||
			cssText += "/* styles for " + def.name + " */\n" + def.css;
 | 
			
		||||
		}
 | 
			
		||||
		if (cssText) {
 | 
			
		||||
			def.cssElem = insertCSS(cssText);
 | 
			
		||||
			def.cssElem.attr("id", "notify-" + def.name);
 | 
			
		||||
		}
 | 
			
		||||
		var fields = {};
 | 
			
		||||
		var elem = $(def.html);
 | 
			
		||||
		findFields("html", elem, fields);
 | 
			
		||||
		findFields("text", elem, fields);
 | 
			
		||||
		def.fields = fields;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var insertCSS = function(cssText) {
 | 
			
		||||
		var e, elem, error;
 | 
			
		||||
		elem = createElem("style");
 | 
			
		||||
		elem.attr("type", 'text/css');
 | 
			
		||||
		$("head").append(elem);
 | 
			
		||||
		try {
 | 
			
		||||
			elem.html(cssText);
 | 
			
		||||
		} catch (_) {
 | 
			
		||||
			elem[0].styleSheet.cssText = cssText;
 | 
			
		||||
		}
 | 
			
		||||
		return elem;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var findFields = function(type, elem, fields) {
 | 
			
		||||
		var attr;
 | 
			
		||||
		if (type !== "html") {
 | 
			
		||||
			type = "text";
 | 
			
		||||
		}
 | 
			
		||||
		attr = "data-notify-" + type;
 | 
			
		||||
		return find(elem, "[" + attr + "]").each(function() {
 | 
			
		||||
			var name;
 | 
			
		||||
			name = $(this).attr(attr);
 | 
			
		||||
			if (!name) {
 | 
			
		||||
				name = blankFieldName;
 | 
			
		||||
			}
 | 
			
		||||
			fields[name] = type;
 | 
			
		||||
		});
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var find = function(elem, selector) {
 | 
			
		||||
		if (elem.is(selector)) {
 | 
			
		||||
			return elem;
 | 
			
		||||
		} else {
 | 
			
		||||
			return elem.find(selector);
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var pluginOptions = {
 | 
			
		||||
		clickToHide: true,
 | 
			
		||||
		autoHide: true,
 | 
			
		||||
		autoHideDelay: 5000,
 | 
			
		||||
		arrowShow: true,
 | 
			
		||||
		arrowSize: 5,
 | 
			
		||||
		breakNewLines: true,
 | 
			
		||||
		elementPosition: "bottom",
 | 
			
		||||
		globalPosition: "top right",
 | 
			
		||||
		style: "bootstrap",
 | 
			
		||||
		className: "error",
 | 
			
		||||
		showAnimation: "slideDown",
 | 
			
		||||
		showDuration: 400,
 | 
			
		||||
		hideAnimation: "slideUp",
 | 
			
		||||
		hideDuration: 200,
 | 
			
		||||
		gap: 5
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var inherit = function(a, b) {
 | 
			
		||||
		var F;
 | 
			
		||||
		F = function() {};
 | 
			
		||||
		F.prototype = a;
 | 
			
		||||
		return $.extend(true, new F(), b);
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var defaults = function(opts) {
 | 
			
		||||
		return $.extend(pluginOptions, opts);
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var createElem = function(tag) {
 | 
			
		||||
		return $("<" + tag + "></" + tag + ">");
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var globalAnchors = {};
 | 
			
		||||
 | 
			
		||||
	var getAnchorElement = function(element) {
 | 
			
		||||
		var radios;
 | 
			
		||||
		if (element.is('[type=radio]')) {
 | 
			
		||||
			radios = element.parents('form:first').find('[type=radio]').filter(function(i, e) {
 | 
			
		||||
				return $(e).attr("name") === element.attr("name");
 | 
			
		||||
			});
 | 
			
		||||
			element = radios.first();
 | 
			
		||||
		}
 | 
			
		||||
		return element;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var incr = function(obj, pos, val) {
 | 
			
		||||
		var opp, temp;
 | 
			
		||||
		if (typeof val === "string") {
 | 
			
		||||
			val = parseInt(val, 10);
 | 
			
		||||
		} else if (typeof val !== "number") {
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
		if (isNaN(val)) {
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
		opp = positions[opposites[pos.charAt(0)]];
 | 
			
		||||
		temp = pos;
 | 
			
		||||
		if (obj[opp] !== undefined) {
 | 
			
		||||
			pos = positions[opp.charAt(0)];
 | 
			
		||||
			val = -val;
 | 
			
		||||
		}
 | 
			
		||||
		if (obj[pos] === undefined) {
 | 
			
		||||
			obj[pos] = val;
 | 
			
		||||
		} else {
 | 
			
		||||
			obj[pos] += val;
 | 
			
		||||
		}
 | 
			
		||||
		return null;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var realign = function(alignment, inner, outer) {
 | 
			
		||||
		if (alignment === "l" || alignment === "t") {
 | 
			
		||||
			return 0;
 | 
			
		||||
		} else if (alignment === "c" || alignment === "m") {
 | 
			
		||||
			return outer / 2 - inner / 2;
 | 
			
		||||
		} else if (alignment === "r" || alignment === "b") {
 | 
			
		||||
			return outer - inner;
 | 
			
		||||
		}
 | 
			
		||||
		throw "Invalid alignment";
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	var encode = function(text) {
 | 
			
		||||
		encode.e = encode.e || createElem("div");
 | 
			
		||||
		return encode.e.text(text).html();
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	function Notification(elem, data, options) {
 | 
			
		||||
		if (typeof options === "string") {
 | 
			
		||||
			options = {
 | 
			
		||||
				className: options
 | 
			
		||||
			};
 | 
			
		||||
		}
 | 
			
		||||
		this.options = inherit(pluginOptions, $.isPlainObject(options) ? options : {});
 | 
			
		||||
		this.loadHTML();
 | 
			
		||||
		this.wrapper = $(coreStyle.html);
 | 
			
		||||
		if (this.options.clickToHide) {
 | 
			
		||||
			this.wrapper.addClass(pluginClassName + "-hidable");
 | 
			
		||||
		}
 | 
			
		||||
		this.wrapper.data(pluginClassName, this);
 | 
			
		||||
		this.arrow = this.wrapper.find("." + pluginClassName + "-arrow");
 | 
			
		||||
		this.container = this.wrapper.find("." + pluginClassName + "-container");
 | 
			
		||||
		this.container.append(this.userContainer);
 | 
			
		||||
		if (elem && elem.length) {
 | 
			
		||||
			this.elementType = elem.attr("type");
 | 
			
		||||
			this.originalElement = elem;
 | 
			
		||||
			this.elem = getAnchorElement(elem);
 | 
			
		||||
			this.elem.data(pluginClassName, this);
 | 
			
		||||
			this.elem.before(this.wrapper);
 | 
			
		||||
		}
 | 
			
		||||
		this.container.hide();
 | 
			
		||||
		this.run(data);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.loadHTML = function() {
 | 
			
		||||
		var style;
 | 
			
		||||
		style = this.getStyle();
 | 
			
		||||
		this.userContainer = $(style.html);
 | 
			
		||||
		this.userFields = style.fields;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.show = function(show, userCallback) {
 | 
			
		||||
		var args, callback, elems, fn, hidden;
 | 
			
		||||
		callback = (function(_this) {
 | 
			
		||||
			return function() {
 | 
			
		||||
				if (!show && !_this.elem) {
 | 
			
		||||
					_this.destroy();
 | 
			
		||||
				}
 | 
			
		||||
				if (userCallback) {
 | 
			
		||||
					return userCallback();
 | 
			
		||||
				}
 | 
			
		||||
			};
 | 
			
		||||
		})(this);
 | 
			
		||||
		hidden = this.container.parent().parents(':hidden').length > 0;
 | 
			
		||||
		elems = this.container.add(this.arrow);
 | 
			
		||||
		args = [];
 | 
			
		||||
		if (hidden && show) {
 | 
			
		||||
			fn = "show";
 | 
			
		||||
		} else if (hidden && !show) {
 | 
			
		||||
			fn = "hide";
 | 
			
		||||
		} else if (!hidden && show) {
 | 
			
		||||
			fn = this.options.showAnimation;
 | 
			
		||||
			args.push(this.options.showDuration);
 | 
			
		||||
		} else if (!hidden && !show) {
 | 
			
		||||
			fn = this.options.hideAnimation;
 | 
			
		||||
			args.push(this.options.hideDuration);
 | 
			
		||||
		} else {
 | 
			
		||||
			return callback();
 | 
			
		||||
		}
 | 
			
		||||
		args.push(callback);
 | 
			
		||||
		return elems[fn].apply(elems, args);
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.setGlobalPosition = function() {
 | 
			
		||||
		var p = this.getPosition();
 | 
			
		||||
		var pMain = p[0];
 | 
			
		||||
		var pAlign = p[1];
 | 
			
		||||
		var main = positions[pMain];
 | 
			
		||||
		var align = positions[pAlign];
 | 
			
		||||
		var key = pMain + "|" + pAlign;
 | 
			
		||||
		var anchor = globalAnchors[key];
 | 
			
		||||
		if (!anchor || !document.body.contains(anchor[0])) {
 | 
			
		||||
			anchor = globalAnchors[key] = createElem("div");
 | 
			
		||||
			var css = {};
 | 
			
		||||
			css[main] = 0;
 | 
			
		||||
			if (align === "middle") {
 | 
			
		||||
				css.top = '45%';
 | 
			
		||||
			} else if (align === "center") {
 | 
			
		||||
				css.left = '45%';
 | 
			
		||||
			} else {
 | 
			
		||||
				css[align] = 0;
 | 
			
		||||
			}
 | 
			
		||||
			anchor.css(css).addClass(pluginClassName + "-corner");
 | 
			
		||||
			$("body").append(anchor);
 | 
			
		||||
		}
 | 
			
		||||
		return anchor.prepend(this.wrapper);
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.setElementPosition = function() {
 | 
			
		||||
		var arrowColor, arrowCss, arrowSize, color, contH, contW, css, elemH, elemIH, elemIW, elemPos, elemW, gap, j, k, len, len1, mainFull, margin, opp, oppFull, pAlign, pArrow, pMain, pos, posFull, position, ref, wrapPos;
 | 
			
		||||
		position = this.getPosition();
 | 
			
		||||
		pMain = position[0];
 | 
			
		||||
		pAlign = position[1];
 | 
			
		||||
		pArrow = position[2];
 | 
			
		||||
		elemPos = this.elem.position();
 | 
			
		||||
		elemH = this.elem.outerHeight();
 | 
			
		||||
		elemW = this.elem.outerWidth();
 | 
			
		||||
		elemIH = this.elem.innerHeight();
 | 
			
		||||
		elemIW = this.elem.innerWidth();
 | 
			
		||||
		wrapPos = this.wrapper.position();
 | 
			
		||||
		contH = this.container.height();
 | 
			
		||||
		contW = this.container.width();
 | 
			
		||||
		mainFull = positions[pMain];
 | 
			
		||||
		opp = opposites[pMain];
 | 
			
		||||
		oppFull = positions[opp];
 | 
			
		||||
		css = {};
 | 
			
		||||
		css[oppFull] = pMain === "b" ? elemH : pMain === "r" ? elemW : 0;
 | 
			
		||||
		incr(css, "top", elemPos.top - wrapPos.top);
 | 
			
		||||
		incr(css, "left", elemPos.left - wrapPos.left);
 | 
			
		||||
		ref = ["top", "left"];
 | 
			
		||||
		for (j = 0, len = ref.length; j < len; j++) {
 | 
			
		||||
			pos = ref[j];
 | 
			
		||||
			margin = parseInt(this.elem.css("margin-" + pos), 10);
 | 
			
		||||
			if (margin) {
 | 
			
		||||
				incr(css, pos, margin);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		gap = Math.max(0, this.options.gap - (this.options.arrowShow ? arrowSize : 0));
 | 
			
		||||
		incr(css, oppFull, gap);
 | 
			
		||||
		if (!this.options.arrowShow) {
 | 
			
		||||
			this.arrow.hide();
 | 
			
		||||
		} else {
 | 
			
		||||
			arrowSize = this.options.arrowSize;
 | 
			
		||||
			arrowCss = $.extend({}, css);
 | 
			
		||||
			arrowColor = this.userContainer.css("border-color") || this.userContainer.css("border-top-color") || this.userContainer.css("background-color") || "white";
 | 
			
		||||
			for (k = 0, len1 = mainPositions.length; k < len1; k++) {
 | 
			
		||||
				pos = mainPositions[k];
 | 
			
		||||
				posFull = positions[pos];
 | 
			
		||||
				if (pos === opp) {
 | 
			
		||||
					continue;
 | 
			
		||||
				}
 | 
			
		||||
				color = posFull === mainFull ? arrowColor : "transparent";
 | 
			
		||||
				arrowCss["border-" + posFull] = arrowSize + "px solid " + color;
 | 
			
		||||
			}
 | 
			
		||||
			incr(css, positions[opp], arrowSize);
 | 
			
		||||
			if (indexOf.call(mainPositions, pAlign) >= 0) {
 | 
			
		||||
				incr(arrowCss, positions[pAlign], arrowSize * 2);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		if (indexOf.call(vAligns, pMain) >= 0) {
 | 
			
		||||
			incr(css, "left", realign(pAlign, contW, elemW));
 | 
			
		||||
			if (arrowCss) {
 | 
			
		||||
				incr(arrowCss, "left", realign(pAlign, arrowSize, elemIW));
 | 
			
		||||
			}
 | 
			
		||||
		} else if (indexOf.call(hAligns, pMain) >= 0) {
 | 
			
		||||
			incr(css, "top", realign(pAlign, contH, elemH));
 | 
			
		||||
			if (arrowCss) {
 | 
			
		||||
				incr(arrowCss, "top", realign(pAlign, arrowSize, elemIH));
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		if (this.container.is(":visible")) {
 | 
			
		||||
			css.display = "block";
 | 
			
		||||
		}
 | 
			
		||||
		this.container.removeAttr("style").css(css);
 | 
			
		||||
		if (arrowCss) {
 | 
			
		||||
			return this.arrow.removeAttr("style").css(arrowCss);
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.getPosition = function() {
 | 
			
		||||
		var pos, ref, ref1, ref2, ref3, ref4, ref5, text;
 | 
			
		||||
		text = this.options.position || (this.elem ? this.options.elementPosition : this.options.globalPosition);
 | 
			
		||||
		pos = parsePosition(text);
 | 
			
		||||
		if (pos.length === 0) {
 | 
			
		||||
			pos[0] = "b";
 | 
			
		||||
		}
 | 
			
		||||
		if (ref = pos[0], indexOf.call(mainPositions, ref) < 0) {
 | 
			
		||||
			throw "Must be one of [" + mainPositions + "]";
 | 
			
		||||
		}
 | 
			
		||||
		if (pos.length === 1 || ((ref1 = pos[0], indexOf.call(vAligns, ref1) >= 0) && (ref2 = pos[1], indexOf.call(hAligns, ref2) < 0)) || ((ref3 = pos[0], indexOf.call(hAligns, ref3) >= 0) && (ref4 = pos[1], indexOf.call(vAligns, ref4) < 0))) {
 | 
			
		||||
			pos[1] = (ref5 = pos[0], indexOf.call(hAligns, ref5) >= 0) ? "m" : "l";
 | 
			
		||||
		}
 | 
			
		||||
		if (pos.length === 2) {
 | 
			
		||||
			pos[2] = pos[1];
 | 
			
		||||
		}
 | 
			
		||||
		return pos;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.getStyle = function(name) {
 | 
			
		||||
		var style;
 | 
			
		||||
		if (!name) {
 | 
			
		||||
			name = this.options.style;
 | 
			
		||||
		}
 | 
			
		||||
		if (!name) {
 | 
			
		||||
			name = "default";
 | 
			
		||||
		}
 | 
			
		||||
		style = styles[name];
 | 
			
		||||
		if (!style) {
 | 
			
		||||
			throw "Missing style: " + name;
 | 
			
		||||
		}
 | 
			
		||||
		return style;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.updateClasses = function() {
 | 
			
		||||
		var classes, style;
 | 
			
		||||
		classes = ["base"];
 | 
			
		||||
		if ($.isArray(this.options.className)) {
 | 
			
		||||
			classes = classes.concat(this.options.className);
 | 
			
		||||
		} else if (this.options.className) {
 | 
			
		||||
			classes.push(this.options.className);
 | 
			
		||||
		}
 | 
			
		||||
		style = this.getStyle();
 | 
			
		||||
		classes = $.map(classes, function(n) {
 | 
			
		||||
			return pluginClassName + "-" + style.name + "-" + n;
 | 
			
		||||
		}).join(" ");
 | 
			
		||||
		return this.userContainer.attr("class", classes);
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.run = function(data, options) {
 | 
			
		||||
		var d, datas, name, type, value;
 | 
			
		||||
		if ($.isPlainObject(options)) {
 | 
			
		||||
			$.extend(this.options, options);
 | 
			
		||||
		} else if ($.type(options) === "string") {
 | 
			
		||||
			this.options.className = options;
 | 
			
		||||
		}
 | 
			
		||||
		if (this.container && !data) {
 | 
			
		||||
			this.show(false);
 | 
			
		||||
			return;
 | 
			
		||||
		} else if (!this.container && !data) {
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
		datas = {};
 | 
			
		||||
		if ($.isPlainObject(data)) {
 | 
			
		||||
			datas = data;
 | 
			
		||||
		} else {
 | 
			
		||||
			datas[blankFieldName] = data;
 | 
			
		||||
		}
 | 
			
		||||
		for (name in datas) {
 | 
			
		||||
			d = datas[name];
 | 
			
		||||
			type = this.userFields[name];
 | 
			
		||||
			if (!type) {
 | 
			
		||||
				continue;
 | 
			
		||||
			}
 | 
			
		||||
			if (type === "text") {
 | 
			
		||||
				d = encode(d);
 | 
			
		||||
				if (this.options.breakNewLines) {
 | 
			
		||||
					d = d.replace(/\n/g, '<br/>');
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			value = name === blankFieldName ? '' : '=' + name;
 | 
			
		||||
			find(this.userContainer, "[data-notify-" + type + value + "]").html(d);
 | 
			
		||||
		}
 | 
			
		||||
		this.updateClasses();
 | 
			
		||||
		if (this.elem) {
 | 
			
		||||
			this.setElementPosition();
 | 
			
		||||
		} else {
 | 
			
		||||
			this.setGlobalPosition();
 | 
			
		||||
		}
 | 
			
		||||
		this.show(true);
 | 
			
		||||
		if (this.options.autoHide) {
 | 
			
		||||
			clearTimeout(this.autohideTimer);
 | 
			
		||||
			this.autohideTimer = setTimeout(this.show.bind(this, false), this.options.autoHideDelay);
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	Notification.prototype.destroy = function() {
 | 
			
		||||
		this.wrapper.data(pluginClassName, null);
 | 
			
		||||
		this.wrapper.remove();
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	$[pluginName] = function(elem, data, options) {
 | 
			
		||||
		if ((elem && elem.nodeName) || elem.jquery) {
 | 
			
		||||
			$(elem)[pluginName](data, options);
 | 
			
		||||
		} else {
 | 
			
		||||
			options = data;
 | 
			
		||||
			data = elem;
 | 
			
		||||
			new Notification(null, data, options);
 | 
			
		||||
		}
 | 
			
		||||
		return elem;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	$.fn[pluginName] = function(data, options) {
 | 
			
		||||
		$(this).each(function() {
 | 
			
		||||
			var prev = getAnchorElement($(this)).data(pluginClassName);
 | 
			
		||||
			if (prev) {
 | 
			
		||||
				prev.destroy();
 | 
			
		||||
			}
 | 
			
		||||
			var curr = new Notification($(this), data, options);
 | 
			
		||||
		});
 | 
			
		||||
		return this;
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	$.extend($[pluginName], {
 | 
			
		||||
		defaults: defaults,
 | 
			
		||||
		addStyle: addStyle,
 | 
			
		||||
		removeStyle: removeStyle,
 | 
			
		||||
		pluginOptions: pluginOptions,
 | 
			
		||||
		getStyle: getStyle,
 | 
			
		||||
		insertCSS: insertCSS
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	//always include the default bootstrap style
 | 
			
		||||
	addStyle("bootstrap", {
 | 
			
		||||
		html: "<div>\n<span data-notify-text></span>\n</div>",
 | 
			
		||||
		classes: {
 | 
			
		||||
			base: {
 | 
			
		||||
				"font-weight": "bold",
 | 
			
		||||
				"padding": "8px 15px 8px 14px",
 | 
			
		||||
				"text-shadow": "0 1px 0 rgba(255, 255, 255, 0.5)",
 | 
			
		||||
				"background-color": "#fcf8e3",
 | 
			
		||||
				"border": "1px solid #fbeed5",
 | 
			
		||||
				"border-radius": "4px",
 | 
			
		||||
				"white-space": "nowrap",
 | 
			
		||||
				"padding-left": "25px",
 | 
			
		||||
				"background-repeat": "no-repeat",
 | 
			
		||||
				"background-position": "3px 7px"
 | 
			
		||||
			},
 | 
			
		||||
			error: {
 | 
			
		||||
				"color": "#B94A48",
 | 
			
		||||
				"background-color": "#F2DEDE",
 | 
			
		||||
				"border-color": "#EED3D7",
 | 
			
		||||
				"background-image": "url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAtRJREFUeNqkVc1u00AQHq+dOD+0poIQfkIjalW0SEGqRMuRnHos3DjwAH0ArlyQeANOOSMeAA5VjyBxKBQhgSpVUKKQNGloFdw4cWw2jtfMOna6JOUArDTazXi/b3dm55socPqQhFka++aHBsI8GsopRJERNFlY88FCEk9Yiwf8RhgRyaHFQpPHCDmZG5oX2ui2yilkcTT1AcDsbYC1NMAyOi7zTX2Agx7A9luAl88BauiiQ/cJaZQfIpAlngDcvZZMrl8vFPK5+XktrWlx3/ehZ5r9+t6e+WVnp1pxnNIjgBe4/6dAysQc8dsmHwPcW9C0h3fW1hans1ltwJhy0GxK7XZbUlMp5Ww2eyan6+ft/f2FAqXGK4CvQk5HueFz7D6GOZtIrK+srupdx1GRBBqNBtzc2AiMr7nPplRdKhb1q6q6zjFhrklEFOUutoQ50xcX86ZlqaZpQrfbBdu2R6/G19zX6XSgh6RX5ubyHCM8nqSID6ICrGiZjGYYxojEsiw4PDwMSL5VKsC8Yf4VRYFzMzMaxwjlJSlCyAQ9l0CW44PBADzXhe7xMdi9HtTrdYjFYkDQL0cn4Xdq2/EAE+InCnvADTf2eah4Sx9vExQjkqXT6aAERICMewd/UAp/IeYANM2joxt+q5VI+ieq2i0Wg3l6DNzHwTERPgo1ko7XBXj3vdlsT2F+UuhIhYkp7u7CarkcrFOCtR3H5JiwbAIeImjT/YQKKBtGjRFCU5IUgFRe7fF4cCNVIPMYo3VKqxwjyNAXNepuopyqnld602qVsfRpEkkz+GFL1wPj6ySXBpJtWVa5xlhpcyhBNwpZHmtX8AGgfIExo0ZpzkWVTBGiXCSEaHh62/PoR0p/vHaczxXGnj4bSo+G78lELU80h1uogBwWLf5YlsPmgDEd4M236xjm+8nm4IuE/9u+/PH2JXZfbwz4zw1WbO+SQPpXfwG/BBgAhCNZiSb/pOQAAAAASUVORK5CYII=)"
 | 
			
		||||
			},
 | 
			
		||||
			success: {
 | 
			
		||||
				"color": "#468847",
 | 
			
		||||
				"background-color": "#DFF0D8",
 | 
			
		||||
				"border-color": "#D6E9C6",
 | 
			
		||||
				"background-image": "url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAutJREFUeNq0lctPE0Ecx38zu/RFS1EryqtgJFA08YCiMZIAQQ4eRG8eDGdPJiYeTIwHTfwPiAcvXIwXLwoXPaDxkWgQ6islKlJLSQWLUraPLTv7Gme32zoF9KSTfLO7v53vZ3d/M7/fIth+IO6INt2jjoA7bjHCJoAlzCRw59YwHYjBnfMPqAKWQYKjGkfCJqAF0xwZjipQtA3MxeSG87VhOOYegVrUCy7UZM9S6TLIdAamySTclZdYhFhRHloGYg7mgZv1Zzztvgud7V1tbQ2twYA34LJmF4p5dXF1KTufnE+SxeJtuCZNsLDCQU0+RyKTF27Unw101l8e6hns3u0PBalORVVVkcaEKBJDgV3+cGM4tKKmI+ohlIGnygKX00rSBfszz/n2uXv81wd6+rt1orsZCHRdr1Imk2F2Kob3hutSxW8thsd8AXNaln9D7CTfA6O+0UgkMuwVvEFFUbbAcrkcTA8+AtOk8E6KiQiDmMFSDqZItAzEVQviRkdDdaFgPp8HSZKAEAL5Qh7Sq2lIJBJwv2scUqkUnKoZgNhcDKhKg5aH+1IkcouCAdFGAQsuWZYhOjwFHQ96oagWgRoUov1T9kRBEODAwxM2QtEUl+Wp+Ln9VRo6BcMw4ErHRYjH4/B26AlQoQQTRdHWwcd9AH57+UAXddvDD37DmrBBV34WfqiXPl61g+vr6xA9zsGeM9gOdsNXkgpEtTwVvwOklXLKm6+/p5ezwk4B+j6droBs2CsGa/gNs6RIxazl4Tc25mpTgw/apPR1LYlNRFAzgsOxkyXYLIM1V8NMwyAkJSctD1eGVKiq5wWjSPdjmeTkiKvVW4f2YPHWl3GAVq6ymcyCTgovM3FzyRiDe2TaKcEKsLpJvNHjZgPNqEtyi6mZIm4SRFyLMUsONSSdkPeFtY1n0mczoY3BHTLhwPRy9/lzcziCw9ACI+yql0VLzcGAZbYSM5CCSZg1/9oc/nn7+i8N9p/8An4JMADxhH+xHfuiKwAAAABJRU5ErkJggg==)"
 | 
			
		||||
			},
 | 
			
		||||
			info: {
 | 
			
		||||
				"color": "#3A87AD",
 | 
			
		||||
				"background-color": "#D9EDF7",
 | 
			
		||||
				"border-color": "#BCE8F1",
 | 
			
		||||
				"background-image": "url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAAABmJLR0QA/wD/AP+gvaeTAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3QYFAhkSsdes/QAAA8dJREFUOMvVlGtMW2UYx//POaWHXg6lLaW0ypAtw1UCgbniNOLcVOLmAjHZolOYlxmTGXVZdAnRfXQm+7SoU4mXaOaiZsEpC9FkiQs6Z6bdCnNYruM6KNBw6YWewzl9z+sHImEWv+vz7XmT95f/+3/+7wP814v+efDOV3/SoX3lHAA+6ODeUFfMfjOWMADgdk+eEKz0pF7aQdMAcOKLLjrcVMVX3xdWN29/GhYP7SvnP0cWfS8caSkfHZsPE9Fgnt02JNutQ0QYHB2dDz9/pKX8QjjuO9xUxd/66HdxTeCHZ3rojQObGQBcuNjfplkD3b19Y/6MrimSaKgSMmpGU5WevmE/swa6Oy73tQHA0Rdr2Mmv/6A1n9w9suQ7097Z9lM4FlTgTDrzZTu4StXVfpiI48rVcUDM5cmEksrFnHxfpTtU/3BFQzCQF/2bYVoNbH7zmItbSoMj40JSzmMyX5qDvriA7QdrIIpA+3cdsMpu0nXI8cV0MtKXCPZev+gCEM1S2NHPvWfP/hL+7FSr3+0p5RBEyhEN5JCKYr8XnASMT0xBNyzQGQeI8fjsGD39RMPk7se2bd5ZtTyoFYXftF6y37gx7NeUtJJOTFlAHDZLDuILU3j3+H5oOrD3yWbIztugaAzgnBKJuBLpGfQrS8wO4FZgV+c1IxaLgWVU0tMLEETCos4xMzEIv9cJXQcyagIwigDGwJgOAtHAwAhisQUjy0ORGERiELgG4iakkzo4MYAxcM5hAMi1WWG1yYCJIcMUaBkVRLdGeSU2995TLWzcUAzONJ7J6FBVBYIggMzmFbvdBV44Corg8vjhzC+EJEl8U1kJtgYrhCzgc/vvTwXKSib1paRFVRVORDAJAsw5FuTaJEhWM2SHB3mOAlhkNxwuLzeJsGwqWzf5TFNdKgtY5qHp6ZFf67Y/sAVadCaVY5YACDDb3Oi4NIjLnWMw2QthCBIsVhsUTU9tvXsjeq9+X1d75/KEs4LNOfcdf/+HthMnvwxOD0wmHaXr7ZItn2wuH2SnBzbZAbPJwpPx+VQuzcm7dgRCB57a1uBzUDRL4bfnI0RE0eaXd9W89mpjqHZnUI5Hh2l2dkZZUhOqpi2qSmpOmZ64Tuu9qlz/SEXo6MEHa3wOip46F1n7633eekV8ds8Wxjn37Wl63VVa+ej5oeEZ/82ZBETJjpJ1Rbij2D3Z/1trXUvLsblCK0XfOx0SX2kMsn9dX+d+7Kf6h8o4AIykuffjT8L20LU+w4AZd5VvEPY+XpWqLV327HR7DzXuDnD8r+ovkBehJ8i+y8YAAAAASUVORK5CYII=)"
 | 
			
		||||
			},
 | 
			
		||||
			warn: {
 | 
			
		||||
				"color": "#C09853",
 | 
			
		||||
				"background-color": "#FCF8E3",
 | 
			
		||||
				"border-color": "#FBEED5",
 | 
			
		||||
				"background-image": "url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAMAAAC6V+0/AAABJlBMVEXr6eb/2oD/wi7/xjr/0mP/ykf/tQD/vBj/3o7/uQ//vyL/twebhgD/4pzX1K3z8e349vK6tHCilCWbiQymn0jGworr6dXQza3HxcKkn1vWvV/5uRfk4dXZ1bD18+/52YebiAmyr5S9mhCzrWq5t6ufjRH54aLs0oS+qD751XqPhAybhwXsujG3sm+Zk0PTwG6Shg+PhhObhwOPgQL4zV2nlyrf27uLfgCPhRHu7OmLgAafkyiWkD3l49ibiAfTs0C+lgCniwD4sgDJxqOilzDWowWFfAH08uebig6qpFHBvH/aw26FfQTQzsvy8OyEfz20r3jAvaKbhgG9q0nc2LbZxXanoUu/u5WSggCtp1anpJKdmFz/zlX/1nGJiYmuq5Dx7+sAAADoPUZSAAAAAXRSTlMAQObYZgAAAAFiS0dEAIgFHUgAAAAJcEhZcwAACxMAAAsTAQCanBgAAAAHdElNRQfdBgUBGhh4aah5AAAAlklEQVQY02NgoBIIE8EUcwn1FkIXM1Tj5dDUQhPU502Mi7XXQxGz5uVIjGOJUUUW81HnYEyMi2HVcUOICQZzMMYmxrEyMylJwgUt5BljWRLjmJm4pI1hYp5SQLGYxDgmLnZOVxuooClIDKgXKMbN5ggV1ACLJcaBxNgcoiGCBiZwdWxOETBDrTyEFey0jYJ4eHjMGWgEAIpRFRCUt08qAAAAAElFTkSuQmCC)"
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
	$(function() {
 | 
			
		||||
		insertCSS(coreStyle.css).attr("id", "core-notify");
 | 
			
		||||
		$(document).on("click", "." + pluginClassName + "-hidable", function(e) {
 | 
			
		||||
			$(this).trigger("notify-hide");
 | 
			
		||||
		});
 | 
			
		||||
		$(document).on("notify-hide", "." + pluginClassName + "-wrapper", function(e) {
 | 
			
		||||
			var elem = $(this).data(pluginClassName);
 | 
			
		||||
			if(elem) {
 | 
			
		||||
				elem.show(false);
 | 
			
		||||
			}
 | 
			
		||||
		});
 | 
			
		||||
	});
 | 
			
		||||
 | 
			
		||||
}));
 | 
			
		||||
| 
						 | 
				
			
			@ -1,5 +1,18 @@
 | 
			
		|||
{% extends 'base.html' %}
 | 
			
		||||
{% load humanize %}
 | 
			
		||||
{% load humanize static %}
 | 
			
		||||
 | 
			
		||||
{% block javascript %}
 | 
			
		||||
  <script src="{% static 'js/jquery.js' %}"></script>
 | 
			
		||||
  <script src="{% static 'js/jquery.ui.widget.js' %}"></script>
 | 
			
		||||
  <!-- The Iframe Transport is required for browsers without support for XHR file uploads -->
 | 
			
		||||
  <script src="{% static 'js/jquery.iframe-transport.js' %}"></script>
 | 
			
		||||
  <!-- The basic File Upload plugin -->
 | 
			
		||||
  <script src="{% static 'js/jquery.fileupload.js' %}"></script>
 | 
			
		||||
  <!-- Calculate md5 -->
 | 
			
		||||
  <script src="{% static 'js/spark-md5.js' %}"></script>
 | 
			
		||||
  <script src="{% static 'js/bootstrap.bundle.min.js' %}"></script>
 | 
			
		||||
  <script src="{% static 'js/notify.js' %}"></script>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
{% block css %}
 | 
			
		||||
<style>
 | 
			
		||||
| 
						 | 
				
			
			@ -24,6 +37,33 @@
 | 
			
		|||
  {{ folder.name }}
 | 
			
		||||
{% endfor %}
 | 
			
		||||
<div class="row">
 | 
			
		||||
<div class="col-lg-2 col-xxl-2 col-md-4 col-sm-6 col-xs-12 mb-3 m-3 d-flex align-items-stretch card">
 | 
			
		||||
    <div class="card-body d-flex flex-column justify-content-center align-items-center">
 | 
			
		||||
      {% csrf_token %}
 | 
			
		||||
      <fieldset class="upload_dropZone text-center mb-3 p-4">
 | 
			
		||||
        <legend class="visually-hidden">Image uploader</legend>
 | 
			
		||||
        <svg class="upload_svg" width="60" height="60" aria-hidden="true">
 | 
			
		||||
          <use href="#icon-imageUpload"></use>
 | 
			
		||||
        </svg>
 | 
			
		||||
        <p class="small my-2">Drag & Drop background image(s) inside dashed region<br><i>or</i></p>
 | 
			
		||||
        <input multiple class="position-absolute invisible" id="chunked_upload" type="file" name="the_file">
 | 
			
		||||
        <label class="btn btn-upload mb-3" for="chunked_upload">Choose file(s)</label>
 | 
			
		||||
        <div class="upload_gallery d-flex flex-wrap justify-content-center gap-3 mb-0"></div>
 | 
			
		||||
      </fieldset>
 | 
			
		||||
      <svg style="display:none">
 | 
			
		||||
        <defs>
 | 
			
		||||
          <symbol id="icon-imageUpload" clip-rule="evenodd" viewBox="0 0 96 96">
 | 
			
		||||
            <path d="M47 6a21 21 0 0 0-12.3 3.8c-2.7 2.1-4.4 5-4.7 7.1-5.8 1.2-10.3 5.6-10.3 10.6 0 6 5.8 11 13 11h12.6V22.7l-7.1 6.8c-.4.3-.9.5-1.4.5-1 0-2-.8-2-1.7 0-.4.3-.9.6-1.2l10.3-8.8c.3-.4.8-.6 1.3-.6.6 0 1 .2 1.4.6l10.2 8.8c.4.3.6.8.6 1.2 0 1-.9 1.7-2 1.7-.5 0-1-.2-1.3-.5l-7.2-6.8v15.6h14.4c6.1 0 11.2-4.1 11.2-9.4 0-5-4-8.8-9.5-9.4C63.8 11.8 56 5.8 47 6Zm-1.7 42.7V38.4h3.4v10.3c0 .8-.7 1.5-1.7 1.5s-1.7-.7-1.7-1.5Z M27 49c-4 0-7 2-7 6v29c0 3 3 6 6 6h42c3 0 6-3 6-6V55c0-4-3-6-7-6H28Zm41 3c1 0 3 1 3 3v19l-13-6a2 2 0 0 0-2 0L44 79l-10-5a2 2 0 0 0-2 0l-9 7V55c0-2 2-3 4-3h41Z M40 62c0 2-2 4-5 4s-5-2-5-4 2-4 5-4 5 2 5 4Z"></path>
 | 
			
		||||
          </symbol>
 | 
			
		||||
        </defs>
 | 
			
		||||
      </svg>
 | 
			
		||||
      <div id="progress" class="progress w-100" style="display: none" role="progressbar" aria-label="Warning example" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100">
 | 
			
		||||
        <h1 id="progress-message"></h1>
 | 
			
		||||
        <div id="progress-bar" class="progress-bar text-bg-warning" style="width: 0%">0%</div>
 | 
			
		||||
      </div>
 | 
			
		||||
      <div id="messages"></div>
 | 
			
		||||
    </div>
 | 
			
		||||
</div>
 | 
			
		||||
{% for file in file_list %}
 | 
			
		||||
  <div class="col-lg-2 col-xxl-2 col-md-4 col-sm-6 col-xs-12 mb-3 m-3 d-flex align-items-stretch card">
 | 
			
		||||
      <div class="card-body d-flex flex-column">
 | 
			
		||||
| 
						 | 
				
			
			@ -38,3 +78,121 @@
 | 
			
		|||
{% endfor %}
 | 
			
		||||
</div>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
{% block inline_javascript %}
 | 
			
		||||
  <script type="text/javascript">
 | 
			
		||||
    $.notify.defaults(
 | 
			
		||||
      {
 | 
			
		||||
        // whether to hide the notification on click
 | 
			
		||||
        clickToHide: true,
 | 
			
		||||
        // whether to auto-hide the notification
 | 
			
		||||
        autoHide: true,
 | 
			
		||||
        // if autoHide, hide after milliseconds
 | 
			
		||||
        autoHideDelay: 5000,
 | 
			
		||||
        // show the arrow pointing at the element
 | 
			
		||||
        arrowShow: true,
 | 
			
		||||
        // arrow size in pixels
 | 
			
		||||
        arrowSize: 5,
 | 
			
		||||
        // position defines the notification position though uses the defaults below
 | 
			
		||||
        position: '...',
 | 
			
		||||
        // default positions
 | 
			
		||||
        elementPosition: 'bottom right',
 | 
			
		||||
        globalPosition: 'top right',
 | 
			
		||||
        // default style
 | 
			
		||||
        style: 'bootstrap',
 | 
			
		||||
        // default class (string or [string])
 | 
			
		||||
        className: 'error',
 | 
			
		||||
        // show animation
 | 
			
		||||
        showAnimation: 'slideDown',
 | 
			
		||||
        // show animation duration
 | 
			
		||||
        showDuration: 400,
 | 
			
		||||
        // hide animation
 | 
			
		||||
        hideAnimation: 'slideUp',
 | 
			
		||||
        // hide animation duration
 | 
			
		||||
        hideDuration: 200,
 | 
			
		||||
        // padding between element and notification
 | 
			
		||||
        gap: 2
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    var md5 = "",
 | 
			
		||||
        csrf = $("input[name='csrfmiddlewaretoken']")[0].value,
 | 
			
		||||
        form_data = [{"name": "csrfmiddlewaretoken", "value": csrf}];
 | 
			
		||||
    function calculate_md5(file, chunk_size) {
 | 
			
		||||
      var slice = File.prototype.slice || File.prototype.mozSlice || File.prototype.webkitSlice,
 | 
			
		||||
          chunks = Math.ceil(file.size / chunk_size),
 | 
			
		||||
          current_chunk = 0,
 | 
			
		||||
          spark = new SparkMD5.ArrayBuffer();
 | 
			
		||||
      function onload(e) {
 | 
			
		||||
        spark.append(e.target.result);  // append chunk
 | 
			
		||||
        current_chunk++;
 | 
			
		||||
        if (current_chunk < chunks) {
 | 
			
		||||
          read_next_chunk();
 | 
			
		||||
        } else {
 | 
			
		||||
          md5 = spark.end();
 | 
			
		||||
        }
 | 
			
		||||
      };
 | 
			
		||||
      function read_next_chunk() {
 | 
			
		||||
        var reader = new FileReader();
 | 
			
		||||
        reader.onload = onload;
 | 
			
		||||
        var start = current_chunk * chunk_size,
 | 
			
		||||
            end = Math.min(start + chunk_size, file.size);
 | 
			
		||||
        reader.readAsArrayBuffer(slice.call(file, start, end));
 | 
			
		||||
      };
 | 
			
		||||
      read_next_chunk();
 | 
			
		||||
    }
 | 
			
		||||
    let ind = 0;
 | 
			
		||||
    let files = []
 | 
			
		||||
    $("#chunked_upload").fileupload({
 | 
			
		||||
      url: "{% url 'files:api_chunked_upload' %}",
 | 
			
		||||
      dataType: "json",
 | 
			
		||||
      maxChunkSize: 1000000, // Chunks of 1000 kB
 | 
			
		||||
      formData: form_data,
 | 
			
		||||
      sequentialUploads: true,
 | 
			
		||||
      add: function(e, data) { // Called before starting upload
 | 
			
		||||
        files.push(data.files[0])
 | 
			
		||||
        calculate_md5(files[ind], 1000000);  // Again, chunks of 1000 kB
 | 
			
		||||
        data.submit();
 | 
			
		||||
        $("#progress").css("display", "flex");
 | 
			
		||||
      },
 | 
			
		||||
      chunkdone: function (e, data) { // Called after uploading each chunk
 | 
			
		||||
        if (form_data.length < 2) {
 | 
			
		||||
          form_data.push(
 | 
			
		||||
            {"name": "upload_id", "value": data.result.upload_id}
 | 
			
		||||
          );
 | 
			
		||||
        }
 | 
			
		||||
        var progress = parseInt(data.loaded / data.total * 100.0, 10);
 | 
			
		||||
        let sel = $("#progress-bar");
 | 
			
		||||
        sel.css("width", progress + "%");
 | 
			
		||||
        sel.text(progress + "%");
 | 
			
		||||
      },
 | 
			
		||||
      done: function (e, data) { // Called when the file has completely uploaded
 | 
			
		||||
        $.ajax({
 | 
			
		||||
          type: "POST",
 | 
			
		||||
          url: "{% url 'files:api_chunked_upload_complete' %}",
 | 
			
		||||
          data: {
 | 
			
		||||
            csrfmiddlewaretoken: csrf,
 | 
			
		||||
            upload_id: data.result.upload_id,
 | 
			
		||||
            md5: md5
 | 
			
		||||
          },
 | 
			
		||||
          dataType: "json",
 | 
			
		||||
            success: function(data) {
 | 
			
		||||
            let s = ""
 | 
			
		||||
            if(data['status'] === true){
 | 
			
		||||
              s = "success"
 | 
			
		||||
            } else {
 | 
			
		||||
              s = "error"
 | 
			
		||||
            }
 | 
			
		||||
            $.notify(data['message'], s);
 | 
			
		||||
            $("#progress").css("display", "none");
 | 
			
		||||
            $("#progress-message").empty()
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
        ind ++;
 | 
			
		||||
        if (ind !== files.length){
 | 
			
		||||
        calculate_md5(files[ind], 1000000);
 | 
			
		||||
        }
 | 
			
		||||
      },
 | 
			
		||||
    })
 | 
			
		||||
  </script>
 | 
			
		||||
{% endblock %}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue
	
	Block a user