/** * Module containing core application logic. * @param {jQuery} $ Insulated jQuery object * @param {JSON} settings Insulated `window.snipeit.settings` object. * @return {IIFE} Immediately invoked. Returns self. */ lineOptions = { legend: { position: "bottom" }, scales: { yAxes: [{ ticks: { fontColor: "rgba(0,0,0,0.5)", fontStyle: "bold", beginAtZero: true, maxTicksLimit: 5, padding: 20 }, gridLines: { drawTicks: false, display: false } }], xAxes: [{ gridLines: { zeroLineColor: "transparent" }, ticks: { padding: 20, fontColor: "rgba(0,0,0,0.5)", fontStyle: "bold" } }] } }; pieOptions = { //Boolean - Whether we should show a stroke on each segment segmentShowStroke: true, //String - The colour of each segment stroke segmentStrokeColor: "#fff", //Number - The width of each segment stroke segmentStrokeWidth: 1, //Number - The percentage of the chart that we cut out of the middle percentageInnerCutout: 50, // This is 0 for Pie charts //Number - Amount of animation steps animationSteps: 100, //String - Animation easing effect animationEasing: "easeOutBounce", //Boolean - Whether we animate the rotation of the Doughnut animateRotate: true, //Boolean - Whether we animate scaling the Doughnut from the centre animateScale: false, //Boolean - whether to make the chart responsive to window resizing responsive: true, // Boolean - whether to maintain the starting aspect ratio or not when responsive, if set to false, will take up entire container maintainAspectRatio: false, //String - A legend template legendTemplate: "", //String - A tooltip template tooltipTemplate: "<%=value %> <%=label%> " }; //----------------- //- END PIE CHART - //----------------- var baseUrl = $('meta[name="baseUrl"]').attr('content'); (function($, settings) { var Components = {}; Components.modals = {}; // confirm delete modal Components.modals.confirmDelete = function() { var $el = $('table'); var events = { 'click': function(evnt) { var $context = $(this); var $dataConfirmModal = $('#dataConfirmModal'); var href = $context.attr('href'); var message = $context.attr('data-content'); var title = $context.attr('data-title'); $('#myModalLabel').text(title); $dataConfirmModal.find('.modal-body').text(message); $('#deleteForm').attr('action', href); $dataConfirmModal.modal({ show: true }); return false; } }; var render = function() { $el.on('click', '.delete-asset', events['click']); }; return { render: render }; }; /** * Application start point * Component definition stays out of load event, execution only happens. */ $(function() { new Components.modals.confirmDelete().render(); }); }(jQuery, window.snipeit.settings)); $(document).ready(function () { /* * Slideout help menu */ $('.slideout-menu-toggle').on('click', function(event){ event.preventDefault(); // create menu variables var slideoutMenu = $('.slideout-menu'); var slideoutMenuWidth = $('.slideout-menu').width(); // toggle open class slideoutMenu.toggleClass("open"); // slide menu if (slideoutMenu.hasClass("open")) { slideoutMenu.show(); slideoutMenu.animate({ right: "0px" }); } else { slideoutMenu.animate({ right: -slideoutMenuWidth }, "-350px"); slideoutMenu.fadeOut(); } }); /* * iCheck checkbox plugin */ $('input[type="checkbox"].minimal, input[type="radio"].minimal').iCheck({ checkboxClass: 'icheckbox_minimal-blue', radioClass: 'iradio_minimal-blue' }); /* * Select2 */ var iOS = /iPhone|iPad|iPod/.test(navigator.userAgent) && !window.MSStream; if(!iOS) { // Vue collision: Avoid overriding a vue select2 instance // by checking to see if the item has already been select2'd. $('select.select2:not(".select2-hidden-accessible")').each(function (i,obj) { { $(obj).select2(); } }); } // $('.datepicker').datepicker(); // var datepicker = $.fn.datepicker.noConflict(); // return $.fn.datepicker to previously assigned value // $.fn.bootstrapDP = datepicker; // $('.datepicker').datepicker(); // Crazy select2 rich dropdowns with images! $('.js-data-ajax').each( function (i,item) { var link = $(item); var endpoint = link.data("endpoint"); var select = link.data("select"); link.select2({ /** * Adds an empty placeholder, allowing every select2 instance to be cleared. * This placeholder can be overridden with the "data-placeholder" attribute. */ placeholder: '', allowClear: true, ajax: { // the baseUrl includes a trailing slash url: Ziggy.baseUrl + 'api/v1/' + endpoint + '/selectlist', dataType: 'json', delay: 250, headers: { "X-Requested-With": 'XMLHttpRequest', "X-CSRF-TOKEN": $('meta[name="csrf-token"]').attr('content') }, data: function (params) { var data = { search: params.term, page: params.page || 1, assetStatusType: link.data("asset-status-type"), }; return data; }, /* processResults: function (data, params) { params.page = params.page || 1; var answer = { results: data.items, pagination: { more: data.pagination.more } }; return answer; }, */ cache: true }, //escapeMarkup: function (markup) { return markup; }, // let our custom formatter work templateResult: formatDatalistSafe, //templateSelection: formatDataSelection }); }); function getSelect2Value(element) { // if the passed object is not a jquery object, assuming 'element' is a selector if (!(element instanceof jQuery)) element = $(element); var select = element.data("select2"); // There's two different locations where the select2-generated input element can be. searchElement = select.dropdown.$search || select.$container.find(".select2-search__field"); var value = searchElement.val(); return value; } $(".select2-hidden-accessible").on('select2:selecting', function (e) { var data = e.params.args.data; var isMouseUp = false; var element = $(this); var value = getSelect2Value(element); if(e.params.args.originalEvent) isMouseUp = e.params.args.originalEvent.type == "mouseup"; // if selected item does not match typed text, do not allow it to pass - force close for ajax. if(!isMouseUp) { if(value.toLowerCase() && data.text.toLowerCase().indexOf(value) < 0) { e.preventDefault(); element.select2('close'); // if it does match, we set a flag in the event (which gets passed to subsequent events), telling it not to worry about the ajax } else if(value.toLowerCase() && data.text.toLowerCase().indexOf(value) > -1) { e.params.args.noForceAjax = true; } } }); $(".select2-hidden-accessible").on('select2:closing', function (e) { var element = $(this); var value = getSelect2Value(element); var noForceAjax = false; var isMouseUp = false; if(e.params.args.originalSelect2Event) noForceAjax = e.params.args.originalSelect2Event.noForceAjax; if(e.params.args.originalEvent) isMouseUp = e.params.args.originalEvent.type == "mouseup"; if(value && !noForceAjax && !isMouseUp) { var endpoint = element.data("endpoint"); var assetStatusType = element.data("asset-status-type"); $.ajax({ url: Ziggy.baseUrl + 'api/v1/' + endpoint + '/selectlist?search='+value+'&page=1' + (assetStatusType ? '&assetStatusType='+assetStatusType : ''), dataType: 'json', headers: { "X-Requested-With": 'XMLHttpRequest', "X-CSRF-TOKEN": $('meta[name="csrf-token"]').attr('content') }, }).done(function(response) { var currentlySelected = element.select2('data').map(function (x){ return +x.id; }).filter(function (x) { return x !== 0; }); // makes sure we're not selecting the same thing twice for multiples var filteredResponse = response.items.filter(function(item) { return currentlySelected.indexOf(+item.id) < 0; }); var first = (currentlySelected.length > 0) ? filteredResponse[0] : response.items[0]; if(first && first.id) { first.selected = true; if($("option[value='" + first.id + "']", element).length < 1) { var option = new Option(first.text, first.id, true, true); element.append(option); } else { var isMultiple = element.attr("multiple") == "multiple"; element.val(isMultiple? element.val().concat(first.id) : element.val(first.id)); } element.trigger('change'); element.trigger({ type: 'select2:select', params: { data: first } }); } }); } }); function formatDatalist (datalist) { var loading_markup = ' Loading...'; if (datalist.loading) { return loading_markup; } var markup = '
' ; markup += '
'; if (datalist.image) { markup += "
" +  datalist.text + "
"; } else { markup += '
'; } markup += "
" + datalist.text + "
"; markup += "
"; return markup; } function formatDatalistSafe(datalist) { // console.warn("What in the hell is going on with Select2?!?!!?!?"); // console.warn($.select2); if (datalist.loading) { return $(' Loading...'); } var root_div = $("
") ; var left_pull = $("
"); if (datalist.image) { var inner_div = $("
"); /****************************************************************** * * We are specifically chosing empty alt-text below, because this * image conveys no additional information, relative to the text * that will *always* be there in any select2 list that is in use * in Snipe-IT. If that changes, we would probably want to change * some signatures of some functions, but right now, we don't want * screen readers to say "HP SuperJet 5000, .... picture of HP * SuperJet 5000..." and so on, for every single row in a list of * assets or models or whatever. * *******************************************************************/ var img = $(""); // console.warn("Img is: "); // console.dir(img); // console.warn("Strigularly, that's: "); // console.log(img); img.attr("src", datalist.image ); inner_div.append(img) } else { var inner_div=$("
"); } left_pull.append(inner_div); root_div.append(left_pull); var name_div = $("
"); name_div.text(datalist.text); root_div.append(name_div) var safe_html = root_div.get(0).outerHTML; var old_html = formatDatalist(datalist); if(safe_html != old_html) { console.log("HTML MISMATCH: "); console.log("FormatDatalistSafe: "); // console.dir(root_div.get(0)); console.log(safe_html); console.log("FormatDataList: "); console.log(old_html); } return root_div; } function formatDataSelection (datalist) { // This a heinous workaround for a known bug in Select2. // Without this, the rich selectlists are vulnerable to XSS. // Many thanks to @uberbrady for this fix. It ain't pretty, // but it resolves the issue until Select2 addresses it on their end. // // Bug was reported in 2016 :{ // https://github.com/select2/select2/issues/4587 return datalist.text.replace(/>/g, '>') .replace(/Click me $('a[data-toggle="tab"]').click(function (e) { var href = $(this).attr("href"); history.pushState(null, null, href); e.preventDefault(); $('a[href="' + $(this).attr('href') + '"]').tab('show'); }); // ------------------------------------------------ // End Deep Linking for Bootstrap tabs // ------------------------------------------------ // Image preview function readURL(input, $preview) { if (input.files && input.files[0]) { var reader = new FileReader(); reader.onload = function(e) { $preview.attr('src', e.target.result); }; reader.readAsDataURL(input.files[0]); } } function formatBytes(bytes) { if(bytes < 1024) return bytes + " Bytes"; else if(bytes < 1048576) return(bytes / 1024).toFixed(2) + " KB"; else if(bytes < 1073741824) return(bytes / 1048576).toFixed(2) + " MB"; else return(bytes / 1073741824).toFixed(2) + " GB"; } // File size validation $('.js-uploadFile').bind('change', function() { var $this = $(this); var id = '#' + $this.attr('id'); var status = id + '-status'; var $status = $(status); $status.removeClass('text-success').removeClass('text-danger'); $(status + ' .goodfile').remove(); $(status + ' .badfile').remove(); $(status + ' .previewSize').hide(); $(id + '-info').html(''); var max_size = $this.data('maxsize'); var total_size = 0; for (var i = 0; i < this.files.length; i++) { total_size += this.files[i].size; $(id + '-info').append('' + this.files[i].name + ' (' + formatBytes(this.files[i].size) + ') '); } console.log('Max size is: ' + max_size); console.log('Real size is: ' + total_size); if (total_size > max_size) { $status.addClass('text-danger').removeClass('help-block').prepend(' ').append(' Upload is ' + formatBytes(total_size) + '.'); } else { $status.addClass('text-success').removeClass('help-block').prepend(' '); var $preview = $(id + '-imagePreview'); readURL(this, $preview); $preview.fadeIn(); } }); }); /** * Toggle disabled */ (function($){ $.fn.toggleDisabled = function(callback){ return this.each(function(){ var disabled, $this = $(this); if($this.attr('disabled')){ $this.removeAttr('disabled'); disabled = false; } else { $this.attr('disabled', 'disabled'); disabled = true; } if(callback && typeof callback === 'function'){ callback(this, disabled); } }); }; })(jQuery);