12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812 |
- /* ------------------------------------------------------------------------------
- *
- * # Dashboard configuration
- *
- * Demo dashboard configuration. Contains charts and plugin initializations
- *
- * ---------------------------------------------------------------------------- */
- // Setup module
- // ------------------------------
- var Dashboard = function () {
- //
- // Setup module components
- //
- // Setup Switchery
- var _componentSwitchery = function() {
- if (typeof Switchery == 'undefined') {
- console.warn('Warning - switchery.min.js is not loaded.');
- return;
- }
- // Initialize multiple switches
- var switches = Array.prototype.slice.call(document.querySelectorAll('.form-input-switchery'));
- switches.forEach(function(html) {
- var switchery = new Switchery(html);
- });
- };
- // Setup Daterangepicker
- var _componentDaterange = function() {
- if (!$().daterangepicker) {
- console.warn('Warning - daterangepicker.js is not loaded.');
- return;
- }
- // Initialize
- $('.daterange-ranges').daterangepicker(
- {
- startDate: moment().subtract(29, 'days'),
- endDate: moment(),
- minDate: '01/01/2015',
- maxDate: '12/31/2019',
- dateLimit: { days: 60 },
- ranges: {
- 'Today': [moment(), moment()],
- 'Yesterday': [moment().subtract(1, 'days'), moment().subtract(1, 'days')],
- 'Last 7 Days': [moment().subtract(6, 'days'), moment()],
- 'Last 30 Days': [moment().subtract(29, 'days'), moment()],
- 'This Month': [moment().startOf('month'), moment().endOf('month')],
- 'Last Month': [moment().subtract(1, 'month').startOf('month'), moment().subtract(1, 'month').endOf('month')]
- },
- opens: $('html').attr('dir') == 'rtl' ? 'right' : 'left',
- applyClass: 'btn-sm bg-slate-600 btn-block',
- cancelClass: 'btn-sm btn-light btn-block',
- locale: {
- format: 'MM/DD/YYYY',
- direction: $('html').attr('dir') == 'rtl' ? 'rtl' : 'ltr'
- }
- },
- function(start, end) {
- $('.daterange-ranges span').html(start.format('MMMM D') + ' - ' + end.format('MMMM D'));
- }
- );
- $('.daterange-ranges span').html(moment().subtract(29, 'days').format('MMMM D') + ' - ' + moment().format('MMMM D'));
- };
- // Use first letter as an icon
- var _componentIconLetter = function() {
- // Grab first letter and insert to the icon
- $('.table tr').each(function() {
- // Title
- var $title = $(this).find('.letter-icon-title'),
- letter = $title.eq(0).text().charAt(0).toUpperCase();
- // Icon
- var $icon = $(this).find('.letter-icon');
- $icon.eq(0).text(letter);
- });
- };
- //
- // Charts configs
- //
- // Traffic sources stream chart
- var _TrafficSourcesStreamChart = function(element, height) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element),
- margin = {top: 5, right: 50, bottom: 40, left: 50},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom,
- tooltipOffset = 30;
- // Tooltip
- var tooltip = d3Container
- .append('div')
- .attr('class', 'd3-tip e')
- .style('display', 'none')
- // Format date
- var format = d3.time.format('%m/%d/%y %H:%M');
- var formatDate = d3.time.format('%H:%M');
- // Colors
- var colorrange = ['#03A9F4', '#29B6F6', '#4FC3F7', '#81D4FA', '#B3E5FC', '#E1F5FE'];
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale().range([0, width]);
- // Vertical
- var y = d3.scale.linear().range([height, 0]);
- // Colors
- var z = d3.scale.ordinal().range(colorrange);
- // Create axes
- // ------------------------------
- // Horizontal
- var xAxis = d3.svg.axis()
- .scale(x)
- .orient('bottom')
- .ticks(d3.time.hours, 4)
- .innerTickSize(4)
- .tickPadding(8)
- .tickFormat(d3.time.format('%H:%M')); // Display hours and minutes in 24h format
- // Left vertical
- var yAxis = d3.svg.axis()
- .scale(y)
- .ticks(6)
- .innerTickSize(4)
- .outerTickSize(0)
- .tickPadding(8)
- .tickFormat(function (d) { return (d/1000) + 'k'; });
- // Right vertical
- var yAxis2 = yAxis;
- // Dash lines
- var gridAxis = d3.svg.axis()
- .scale(y)
- .orient('left')
- .ticks(6)
- .tickPadding(8)
- .tickFormat('')
- .tickSize(-width, 0, 0);
- // Create chart
- // ------------------------------
- // Container
- var container = d3Container.append('svg')
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');
- // Construct chart layout
- // ------------------------------
- // Stack
- var stack = d3.layout.stack()
- .offset('silhouette')
- .values(function(d) { return d.values; })
- .x(function(d) { return d.date; })
- .y(function(d) { return d.value; });
- // Nest
- var nest = d3.nest()
- .key(function(d) { return d.key; });
- // Area
- var area = d3.svg.area()
- .interpolate('cardinal')
- .x(function(d) { return x(d.date); })
- .y0(function(d) { return y(d.y0); })
- .y1(function(d) { return y(d.y0 + d.y); });
- // Load data
- // ------------------------------
- d3.csv('../../../../global_assets/demo_data/dashboard/traffic_sources.csv', function (error, data) {
- // Pull out values
- data.forEach(function (d) {
- d.date = format.parse(d.date);
- d.value = +d.value;
- });
- // Stack and nest layers
- var layers = stack(nest.entries(data));
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain(d3.extent(data, function(d, i) { return d.date; }));
- // Vertical
- y.domain([0, d3.max(data, function(d) { return d.y0 + d.y; })]);
- // Add grid
- // ------------------------------
- // Horizontal grid. Must be before the group
- svg.append('g')
- .attr('class', 'd3-grid-dashed')
- .call(gridAxis);
- //
- // Append chart elements
- //
- // Stream layers
- // ------------------------------
- // Create group
- var group = svg.append('g')
- .attr('class', 'streamgraph-layers-group');
- // And append paths to this group
- var layer = group.selectAll('.streamgraph-layer')
- .data(layers)
- .enter()
- .append('path')
- .attr('class', 'streamgraph-layer')
- .attr('d', function(d) { return area(d.values); })
- .style('stroke', '#fff')
- .style('stroke-width', 0.5)
- .style('fill', function(d, i) { return z(i); });
- // Add transition
- var layerTransition = layer
- .style('opacity', 0)
- .transition()
- .duration(750)
- .delay(function(d, i) { return i * 50; })
- .style('opacity', 1)
- // Append axes
- // ------------------------------
- //
- // Left vertical
- //
- svg.append('g')
- .attr('class', 'd3-axis d3-axis-left d3-axis-solid')
- .call(yAxis.orient('left'));
- // Hide first tick
- d3.select(svg.selectAll('.d3-axis-left .tick text')[0][0])
- .style('visibility', 'hidden');
- //
- // Right vertical
- //
- svg.append('g')
- .attr('class', 'd3-axis d3-axis-right d3-axis-solid')
- .attr('transform', 'translate(' + width + ', 0)')
- .call(yAxis2.orient('right'));
- // Hide first tick
- d3.select(svg.selectAll('.d3-axis-right .tick text')[0][0])
- .style('visibility', 'hidden');
- //
- // Horizontal
- //
- var xaxisg = svg.append('g')
- .attr('class', 'd3-axis d3-axis-horizontal d3-axis-solid')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xAxis);
- // Add extra subticks for hidden hours
- xaxisg.selectAll('.d3-axis-subticks')
- .data(x.ticks(d3.time.hours), function(d) { return d; })
- .enter()
- .append('line')
- .attr('class', 'd3-axis-subticks')
- .attr('y1', 0)
- .attr('y2', 4)
- .attr('x1', x)
- .attr('x2', x);
- // Add hover line and pointer
- // ------------------------------
- // Append group to the group of paths to prevent appearance outside chart area
- var hoverLineGroup = group.append('g')
- .attr('class', 'hover-line');
- // Add line
- var hoverLine = hoverLineGroup
- .append('line')
- .attr('y1', 0)
- .attr('y2', height)
- .style('fill', 'none')
- .style('stroke', '#fff')
- .style('stroke-width', 1)
- .style('pointer-events', 'none')
- .style('shape-rendering', 'crispEdges')
- .style('opacity', 0);
- // Add pointer
- var hoverPointer = hoverLineGroup
- .append('rect')
- .attr('x', 2)
- .attr('y', 2)
- .attr('width', 6)
- .attr('height', 6)
- .style('fill', '#03A9F4')
- .style('stroke', '#fff')
- .style('stroke-width', 1)
- .style('shape-rendering', 'crispEdges')
- .style('pointer-events', 'none')
- .style('opacity', 0);
- // Append events to the layers group
- // ------------------------------
- layerTransition.each('end', function() {
- layer
- .on('mouseover', function (d, i) {
- svg.selectAll('.streamgraph-layer')
- .transition()
- .duration(250)
- .style('opacity', function (d, j) {
- return j != i ? 0.75 : 1; // Mute all except hovered
- });
- })
- .on('mousemove', function (d, i) {
- mouse = d3.mouse(this);
- mousex = mouse[0];
- mousey = mouse[1];
- datearray = [];
- var invertedx = x.invert(mousex);
- invertedx = invertedx.getHours();
- var selected = (d.values);
- for (var k = 0; k < selected.length; k++) {
- datearray[k] = selected[k].date
- datearray[k] = datearray[k].getHours();
- }
- mousedate = datearray.indexOf(invertedx);
- pro = d.values[mousedate].value;
- // Display mouse pointer
- hoverPointer
- .attr('x', mousex - 3)
- .attr('y', mousey - 6)
- .style('opacity', 1);
- hoverLine
- .attr('x1', mousex)
- .attr('x2', mousex)
- .style('opacity', 1);
- //
- // Tooltip
- //
- // Tooltip data
- tooltip.html(
- '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base my-1"><i class="icon-circle-left2 mr-2"></i>' + d.key + '</div>' + '</li>' +
- '<li>' + 'Visits: ' + "<span class='font-weight-semibold float-right'>" + pro + '</span>' + '</li>' +
- '<li>' + 'Time: ' + '<span class="font-weight-semibold float-right">' + formatDate(d.values[mousedate].date) + '</span>' + '</li>' +
- '</ul>'
- )
- .style('display', 'block');
- // Tooltip arrow
- tooltip.append('div').attr('class', 'd3-tip-arrow');
- })
- .on('mouseout', function (d, i) {
- // Revert full opacity to all paths
- svg.selectAll('.streamgraph-layer')
- .transition()
- .duration(250)
- .style('opacity', 1);
- // Hide cursor pointer
- hoverPointer.style('opacity', 0);
- // Hide tooltip
- tooltip.style('display', 'none');
- hoverLine.style('opacity', 0);
- });
- });
- // Append events to the chart container
- // ------------------------------
- d3Container
- .on('mousemove', function (d, i) {
- mouse = d3.mouse(this);
- mousex = mouse[0];
- mousey = mouse[1];
- // Move tooltip vertically
- tooltip.style('top', (mousey - ($('.d3-tip').outerHeight() / 2)) - 2 + 'px') // Half tooltip height - half arrow width
- // Move tooltip horizontally
- if(mousex >= ($(element).outerWidth() - $('.d3-tip').outerWidth() - margin.right - (tooltipOffset * 2))) {
- tooltip
- .style('left', (mousex - $('.d3-tip').outerWidth() - tooltipOffset) + 'px') // Change tooltip direction from right to left to keep it inside graph area
- .attr('class', 'd3-tip w');
- }
- else {
- tooltip
- .style('left', (mousex + tooltipOffset) + 'px' )
- .attr('class', 'd3-tip e');
- }
- });
- });
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', resizeStream);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', resizeStream);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function resizeStream() {
- // Layout
- // -------------------------
- // Define width
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Horizontal range
- x.range([0, width]);
- // Chart elements
- // -------------------------
- // Horizontal axis
- svg.selectAll('.d3-axis-horizontal').call(xAxis);
- // Horizontal axis subticks
- svg.selectAll('.d3-axis-subticks').attr('x1', x).attr('x2', x);
- // Grid lines width
- svg.selectAll('.d3-grid-dashed').call(gridAxis.tickSize(-width, 0, 0))
- // Right vertical axis
- svg.selectAll('.d3-axis-right').attr('transform', 'translate(' + width + ', 0)');
- // Area paths
- svg.selectAll('.streamgraph-layer').attr('d', function(d) { return area(d.values); });
- }
- }
- };
- // App sales line chart
- var _AppSalesLinesChart = function(element, height) {
- if (typeof d3 == 'undefined' || typeof d3.tip == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element),
- margin = {top: 5, right: 30, bottom: 30, left: 50},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom;
- // Tooltip
- var tooltip = d3.tip()
- .attr('class', 'd3-tip')
- .html(function (d) {
- return '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base my-1"><i class="icon-circle-left2 mr-2"></i>' + d.name + ' app' + '</div>' + '</li>' +
- '<li>' + 'Sales: ' + '<span class="font-weight-semibold float-right">' + d.value + '</span>' + '</li>' +
- '<li>' + 'Revenue: ' + '<span class="font-weight-semibold float-right">' + '$' + (d.value * 25).toFixed(2) + '</span>' + '</li>' +
- '</ul>';
- });
- // Format date
- var parseDate = d3.time.format('%Y/%m/%d').parse,
- formatDate = d3.time.format('%b %d, %y');
- // Line colors
- var scale = ['#4CAF50', '#FF5722', '#5C6BC0'],
- color = d3.scale.ordinal().range(scale);
- // Create chart
- // ------------------------------
- // Container
- var container = d3Container.append('svg');
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
- .call(tooltip);
- // Add date range switcher
- // ------------------------------
- // Menu
- var menu = $('#select_date').multiselect({
- buttonClass: 'text-default font-weight-semibold bg-transparent border-0 cursor-pointer outline-0 py-0 pl-0',
- enableHTML: true,
- dropRight: $('html').attr('dir') == 'rtl' ? false : true,
- onChange: function() {
- change();
- },
- buttonText: function (options, element) {
- var selected = '';
- options.each(function() {
- selected += $(this).html() + ', ';
- });
- return '<span class="badge badge-mark border-warning mr-2"></span>' + selected.substr(0, selected.length -2);
- }
- });
- // Load data
- // ------------------------------
- d3.csv('../../../../global_assets/demo_data/dashboard/app_sales.csv', function(error, data) {
- formatted = data;
- redraw();
- });
- // Construct layout
- // ------------------------------
- // Add events
- var altKey;
- d3.select(window)
- .on('keydown', function() { altKey = d3.event.altKey; })
- .on('keyup', function() { altKey = false; });
- // Set terms of transition on date change
- function change() {
- d3.transition()
- .duration(altKey ? 7500 : 500)
- .each(redraw);
- }
- // Main chart drawing function
- // ------------------------------
- function redraw() {
- // Construct chart layout
- // ------------------------------
- // Create data nests
- var nested = d3.nest()
- .key(function(d) { return d.type; })
- .map(formatted)
- // Get value from menu selection
- // the option values correspond
- //to the [type] value we used to nest the data
- var series = menu.val();
- // Only retrieve data from the selected series using nest
- var data = nested[series];
- // For object constancy we will need to set 'keys', one for each type of data (column name) exclude all others.
- color.domain(d3.keys(data[0]).filter(function(key) { return (key !== 'date' && key !== 'type'); }));
- // Setting up color map
- var linedata = color.domain().map(function(name) {
- return {
- name: name,
- values: data.map(function(d) {
- return {name: name, date: parseDate(d.date), value: parseFloat(d[name], 10)};
- })
- };
- });
- // Draw the line
- var line = d3.svg.line()
- .x(function(d) { return x(d.date); })
- .y(function(d) { return y(d.value); })
- .interpolate('cardinal');
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale()
- .domain([
- d3.min(linedata, function(c) { return d3.min(c.values, function(v) { return v.date; }); }),
- d3.max(linedata, function(c) { return d3.max(c.values, function(v) { return v.date; }); })
- ])
- .range([0, width]);
- // Vertical
- var y = d3.scale.linear()
- .domain([
- d3.min(linedata, function(c) { return d3.min(c.values, function(v) { return v.value; }); }),
- d3.max(linedata, function(c) { return d3.max(c.values, function(v) { return v.value; }); })
- ])
- .range([height, 0]);
- // Create axes
- // ------------------------------
- // Horizontal
- var xAxis = d3.svg.axis()
- .scale(x)
- .orient('bottom')
- .tickPadding(8)
- .ticks(d3.time.days)
- .innerTickSize(4)
- .tickFormat(d3.time.format('%a')); // Display hours and minutes in 24h format
- // Vertical
- var yAxis = d3.svg.axis()
- .scale(y)
- .orient('left')
- .ticks(6)
- .tickSize(0 -width)
- .tickPadding(8);
- //
- // Append chart elements
- //
- // Append axes
- // ------------------------------
- // Horizontal
- svg.append('g')
- .attr('class', 'd3-axis d3-axis-horizontal d3-axis-solid')
- .attr('transform', 'translate(0,' + height + ')');
- // Vertical
- svg.append('g')
- .attr('class', 'd3-axis d3-axis-vertical d3-axis-transparent');
- // Append lines
- // ------------------------------
- // Bind the data
- var lines = svg.selectAll('.lines')
- .data(linedata)
- // Append a group tag for each line
- var lineGroup = lines
- .enter()
- .append('g')
- .attr('class', 'lines')
- .attr('id', function(d){ return d.name + '-line'; });
- // Append the line to the graph
- lineGroup.append('path')
- .attr('class', 'd3-line d3-line-medium')
- .style('stroke', function(d) { return color(d.name); })
- .style('opacity', 0)
- .attr('d', function(d) { return line(d.values[0]); })
- .transition()
- .duration(500)
- .delay(function(d, i) { return i * 200; })
- .style('opacity', 1);
- // Append circles
- // ------------------------------
- var circles = lines.selectAll('circle')
- .data(function(d) { return d.values; })
- .enter()
- .append('circle')
- .attr('class', 'd3-line-circle d3-line-circle-medium')
- .attr('cx', function(d,i){return x(d.date)})
- .attr('cy',function(d,i){return y(d.value)})
- .attr('r', 3)
- .style('fill', '#fff')
- .style('stroke', function(d) { return color(d.name); });
- // Add transition
- circles
- .style('opacity', 0)
- .transition()
- .duration(500)
- .delay(500)
- .style('opacity', 1);
- // Append tooltip
- // ------------------------------
- // Add tooltip on circle hover
- circles
- .on('mouseover', function (d) {
- tooltip.offset([-15, 0]).show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- .on('mouseout', function (d) {
- tooltip.hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- });
- // Change tooltip direction of first point
- // to always keep it inside chart, useful on mobiles
- lines.each(function (d) {
- d3.select(d3.select(this).selectAll('circle')[0][0])
- .on('mouseover', function (d) {
- tooltip.offset([0, 15]).direction('e').show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- .on('mouseout', function (d) {
- tooltip.direction('n').hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- });
- })
- // Change tooltip direction of last point
- // to always keep it inside chart, useful on mobiles
- lines.each(function (d) {
- d3.select(d3.select(this).selectAll('circle')[0][d3.select(this).selectAll('circle').size() - 1])
- .on('mouseover', function (d) {
- tooltip.offset([0, -15]).direction('w').show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- .on('mouseout', function (d) {
- tooltip.direction('n').hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- })
- })
- // Update chart on date change
- // ------------------------------
- // Set variable for updating visualization
- var lineUpdate = d3.transition(lines);
- // Update lines
- lineUpdate.select('path')
- .attr('d', function(d, i) { return line(d.values); });
- // Update circles
- lineUpdate.selectAll('circle')
- .attr('cy',function(d,i){return y(d.value)})
- .attr('cx', function(d,i){return x(d.date)});
- // Update vertical axes
- d3.transition(svg)
- .select('.d3-axis-vertical')
- .call(yAxis);
- // Update horizontal axes
- d3.transition(svg)
- .select('.d3-axis-horizontal')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xAxis);
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', appSalesResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', appSalesResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function appSalesResize() {
- // Layout
- // -------------------------
- // Define width
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Horizontal range
- x.range([0, width]);
- // Vertical range
- y.range([height, 0]);
- // Chart elements
- // -------------------------
- // Horizontal axis
- svg.select('.d3-axis-horizontal').call(xAxis);
- // Vertical axis
- svg.select('.d3-axis-vertical').call(yAxis.tickSize(0-width));
- // Lines
- svg.selectAll('.d3-line').attr('d', function(d, i) { return line(d.values); });
- // Circles
- svg.selectAll('.d3-line-circle').attr('cx', function(d,i){return x(d.date)})
- }
- }
- }
- };
- // Monthly sales area chart
- var _MonthlySalesAreaChart = function(element, height, color) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element),
- margin = {top: 20, right: 35, bottom: 40, left: 35},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom;
- // Date and time format
- var parseDate = d3.time.format('%Y-%m-%d').parse,
- bisectDate = d3.bisector(function(d) { return d.date; }).left,
- formatDate = d3.time.format('%b %d');
- // Create SVG
- // ------------------------------
- // Container
- var container = d3Container.append('svg');
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
- // Construct chart layout
- // ------------------------------
- // Area
- var area = d3.svg.area()
- .x(function(d) { return x(d.date); })
- .y0(height)
- .y1(function(d) { return y(d.value); })
- .interpolate('monotone')
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale().range([0, width ]);
- // Vertical
- var y = d3.scale.linear().range([height, 0]);
- // Create axes
- // ------------------------------
- // Horizontal
- var xAxis = d3.svg.axis()
- .scale(x)
- .orient('bottom')
- .ticks(d3.time.days, 6)
- .innerTickSize(4)
- .tickPadding(8)
- .tickFormat(d3.time.format('%b %d'));
- // Load data
- // ------------------------------
- d3.json('../../../../global_assets/demo_data/dashboard/monthly_sales.json', function (error, data) {
- // Show what's wrong if error
- if (error) return console.error(error);
- // Pull out values
- data.forEach(function (d) {
- d.date = parseDate(d.date);
- d.value = +d.value;
- });
- // Get the maximum value in the given array
- var maxY = d3.max(data, function(d) { return d.value; });
- // Reset start data for animation
- var startData = data.map(function(datum) {
- return {
- date: datum.date,
- value: 0
- };
- });
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain(d3.extent(data, function(d, i) { return d.date; }));
- // Vertical
- y.domain([0, d3.max( data, function(d) { return d.value; })]);
- //
- // Append chart elements
- //
- // Append axes
- // -------------------------
- // Horizontal
- var horizontalAxis = svg.append('g')
- .attr('class', 'd3-axis d3-axis-horizontal d3-axis-solid')
- .attr('transform', 'translate(0,' + height + ')')
- .call(xAxis);
- // Add extra subticks for hidden hours
- horizontalAxis.selectAll('.d3-axis-subticks')
- .data(x.ticks(d3.time.days), function(d) { return d; })
- .enter()
- .append('line')
- .attr('class', 'd3-axis-subticks')
- .attr('y1', 0)
- .attr('y2', 4)
- .attr('x1', x)
- .attr('x2', x);
- // Append area
- // -------------------------
- // Add area path
- svg.append('path')
- .datum(data)
- .attr('class', 'd3-area')
- .attr('d', area)
- .style('fill', color)
- .transition() // begin animation
- .duration(1000)
- .attrTween('d', function() {
- var interpolator = d3.interpolateArray(startData, data);
- return function (t) {
- return area(interpolator (t));
- }
- });
- // Append crosshair and tooltip
- // -------------------------
- //
- // Line
- //
- // Line group
- var focusLine = svg.append('g')
- .attr('class', 'd3-crosshair-line')
- .style('display', 'none');
- // Line element
- focusLine.append('line')
- .attr('class', 'vertical-crosshair')
- .attr('y1', 0)
- .attr('y2', -maxY)
- .style('stroke', '#e5e5e5')
- .style('shape-rendering', 'crispEdges')
- //
- // Pointer
- //
- // Pointer group
- var focusPointer = svg.append('g')
- .attr('class', 'd3-crosshair-pointer')
- .style('display', 'none');
- // Pointer element
- focusPointer.append('circle')
- .attr('r', 3)
- .style('fill', '#fff')
- .style('stroke', color)
- .style('stroke-width', 1)
- //
- // Text
- //
- // Text group
- var focusText = svg.append('g')
- .attr('class', 'd3-crosshair-text')
- .style('display', 'none');
- // Text element
- focusText.append('text')
- .attr('dy', -10)
- .style('font-size', 12);
- //
- // Overlay with events
- //
- svg.append('rect')
- .attr('class', 'd3-crosshair-overlay')
- .style('fill', 'none')
- .style('pointer-events', 'all')
- .attr('width', width)
- .attr('height', height)
- .on('mouseover', function() {
- focusPointer.style('display', null);
- focusLine.style('display', null)
- focusText.style('display', null);
- })
- .on('mouseout', function() {
- focusPointer.style('display', 'none');
- focusLine.style('display', 'none');
- focusText.style('display', 'none');
- })
- .on('mousemove', mousemove);
- // Display tooltip on mousemove
- function mousemove() {
- // Define main variables
- var mouse = d3.mouse(this),
- mousex = mouse[0],
- mousey = mouse[1],
- x0 = x.invert(mousex),
- i = bisectDate(data, x0),
- d0 = data[i - 1],
- d1 = data[i],
- d = x0 - d0.date > d1.date - x0 ? d1 : d0;
- // Move line
- focusLine.attr('transform', 'translate(' + x(d.date) + ',' + height + ')');
- // Move pointer
- focusPointer.attr('transform', 'translate(' + x(d.date) + ',' + y(d.value) + ')');
- // Reverse tooltip at the end point
- if(mousex >= (d3Container.node().getBoundingClientRect().width - focusText.select('text').node().getBoundingClientRect().width - margin.right - margin.left)) {
- focusText.select('text').attr('text-anchor', 'end').attr('x', function () { return (x(d.date) - 15) + 'px' }).text(formatDate(d.date) + ' - ' + d.value + ' sales');
- }
- else {
- focusText.select('text').attr('text-anchor', 'start').attr('x', function () { return (x(d.date) + 15) + 'px' }).text(formatDate(d.date) + ' - ' + d.value + ' sales');
- }
- }
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', monthlySalesAreaResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', monthlySalesAreaResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function monthlySalesAreaResize() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Axes
- // -------------------------
- // Horizontal range
- x.range([0, width]);
- // Horizontal axis
- svg.selectAll('.d3-axis-horizontal').call(xAxis);
- // Horizontal axis subticks
- svg.selectAll('.d3-axis-subticks').attr('x1', x).attr('x2', x);
- // Chart elements
- // -------------------------
- // Area path
- svg.selectAll('.d3-area').datum(data).attr('d', area);
- // Crosshair
- svg.selectAll('.d3-crosshair-overlay').attr('width', width);
- }
- });
- }
- };
- // Daily sales heatmap
- var _AppSalesHeatmap = function(element) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Load data
- d3.csv('../../../../global_assets/demo_data/dashboard/app_sales_heatmap.csv', function(error, data) {
- // Bind data
- // ------------------------------
- // Nest data
- var nested_data = d3.nest().key(function(d) { return d.app; }),
- nest = nested_data.entries(data);
- // Format date
- var format = d3.time.format('%Y/%m/%d %H:%M'),
- formatTime = d3.time.format('%H:%M');
- // Pull out values
- data.forEach(function(d, i) {
- d.date = format.parse(d.date),
- d.value = +d.value
- });
- // Layout setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element);
- margin = { top: 20, right: 0, bottom: 30, left: 0 },
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- gridSize = width / new Date(data[data.length - 1].date).getHours(), // dynamically set grid size
- rowGap = 40, // vertical gap between rows
- height = (rowGap + gridSize) * (d3.max(nest, function(d,i) {return i+1})) - margin.top,
- buckets = 5, // number of colors in range
- colors = ['#DCEDC8','#C5E1A5','#9CCC65','#7CB342','#558B2F'];
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale().range([0, width]);
- // Vertical
- var y = d3.scale.linear().range([height, 0]);
- // Colors
- var colorScale = d3.scale.quantile()
- .domain([0, buckets - 1, d3.max(data, function (d) { return d.value; })])
- .range(colors);
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain([new Date(data[0].date), d3.time.hour.offset(new Date(data[data.length - 1].date), 1)]);
- // Vertical
- y.domain([0, d3.max(data, function(d) { return d.app; })]);
- // Create chart
- // ------------------------------
- // Container
- var container = d3Container.append('svg');
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');
- //
- // Append chart elements
- //
- // App groups
- // ------------------------------
- // Add groups for each app
- var hourGroup = svg.selectAll('.hour-group')
- .data(nest)
- .enter()
- .append('g')
- .attr('class', 'hour-group')
- .attr('transform', function(d, i) { return 'translate(0, ' + ((gridSize + rowGap) * i) +')'; });
- // Add app name
- hourGroup
- .append('text')
- .attr('class', 'app-label')
- .attr('x', 0)
- .attr('y', -(margin.top/2))
- .text(function (d, i) { return d.key; });
- // Sales count text
- hourGroup
- .append('text')
- .attr('class', 'sales-count')
- .attr('x', width)
- .attr('y', -(margin.top/2))
- .style('text-anchor', 'end')
- .text(function (d, i) { return d3.sum(d.values, function(d) { return d.value; }) + ' sales today' });
- // Add map elements
- // ------------------------------
- // Add map squares
- var heatMap = hourGroup.selectAll('.heatmap-hour')
- .data(function(d) {return d.values})
- .enter()
- .append('rect')
- .attr('x', function(d,i) { return x(d.date); })
- .attr('y', 0)
- .attr('class', 'heatmap-hour')
- .attr('width', gridSize)
- .attr('height', gridSize)
- .style('fill', '#fff')
- .style('stroke', '#fff')
- .style('cursor', 'pointer')
- .style('shape-rendering', 'crispEdges');
- // Add loading transition
- heatMap.transition()
- .duration(250)
- .delay(function(d, i) { return i * 20; })
- .style('fill', function(d) { return colorScale(d.value); })
- // Add user interaction
- hourGroup.each(function(d) {
- heatMap
- .on('mouseover', function (d, i) {
- d3.select(this).style('opacity', 0.75);
- d3.select(this.parentNode).select('.sales-count').text(function(d) { return d.values[i].value + ' sales at ' + formatTime(d.values[i].date); })
- })
- .on('mouseout', function (d, i) {
- d3.select(this).style('opacity', 1);
- d3.select(this.parentNode).select('.sales-count').text(function (d, i) { return d3.sum(d.values, function(d) { return d.value; }) + ' sales today' })
- })
- })
- // Add legend
- // ------------------------------
- // Get min and max values
- var minValue, maxValue;
- data.forEach(function(d, i) {
- maxValue = d3.max(data, function (d) { return d.value; });
- minValue = d3.min(data, function (d) { return d.value; });
- });
- // Place legend inside separate group
- var legendGroup = svg.append('g')
- .attr('class', 'legend-group')
- .attr('width', width)
- .attr('transform', 'translate(' + ((width/2) - ((buckets * gridSize))/2) + ',' + (height + (margin.bottom - margin.top)) + ')');
- // Then group legend elements
- var legend = legendGroup.selectAll('.heatmap-legend')
- .data([0].concat(colorScale.quantiles()), function(d) { return d; })
- .enter()
- .append('g')
- .attr('class', 'heatmap-legend');
- // Add legend items
- legend.append('rect')
- .attr('class', 'heatmap-legend-item')
- .attr('x', function(d, i) { return gridSize * i; })
- .attr('y', -8)
- .attr('width', gridSize)
- .attr('height', 5)
- .style('stroke', '#fff')
- .style('shape-rendering', 'crispEdges')
- .style('fill', function(d, i) { return colors[i]; });
- // Add min value text label
- legendGroup.append('text')
- .attr('class', 'min-legend-value')
- .attr('x', -10)
- .attr('y', -2)
- .style('text-anchor', 'end')
- .style('font-size', 11)
- .style('fill', '#999')
- .text(minValue);
- // Add max value text label
- legendGroup.append('text')
- .attr('class', 'max-legend-value')
- .attr('x', (buckets * gridSize) + 10)
- .attr('y', -2)
- .style('font-size', 11)
- .style('fill', '#999')
- .text(maxValue);
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', resizeHeatmap);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', resizeHeatmap);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function resizeHeatmap() {
- // Layout
- // -------------------------
- // Width
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- // Grid size
- gridSize = width / new Date(data[data.length - 1].date).getHours(),
- // Height
- height = (rowGap + gridSize) * (d3.max(nest, function(d,i) {return i+1})) - margin.top,
- // Main svg width
- container.attr('width', width + margin.left + margin.right).attr('height', height + margin.bottom);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right).attr('height', height + margin.bottom);
- // Horizontal range
- x.range([0, width]);
- // Chart elements
- // -------------------------
- // Groups for each app
- svg.selectAll('.hour-group')
- .attr('transform', function(d, i) { return 'translate(0, ' + ((gridSize + rowGap) * i) +')'; });
- // Map squares
- svg.selectAll('.heatmap-hour')
- .attr('width', gridSize)
- .attr('height', gridSize)
- .attr('x', function(d,i) { return x(d.date); });
- // Legend group
- svg.selectAll('.legend-group')
- .attr('transform', 'translate(' + ((width/2) - ((buckets * gridSize))/2) + ',' + (height + margin.bottom - margin.top) + ')');
- // Sales count text
- svg.selectAll('.sales-count')
- .attr('x', width);
- // Legend item
- svg.selectAll('.heatmap-legend-item')
- .attr('width', gridSize)
- .attr('x', function(d, i) { return gridSize * i; });
- // Max value text label
- svg.selectAll('.max-legend-value')
- .attr('x', (buckets * gridSize) + 10);
- }
- });
- }
- };
- // Messages area chart
- var _MessagesAreaChart = function(element, height, color) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element),
- margin = {top: 0, right: 0, bottom: 0, left: 0},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom;
- // Date and time format
- var parseDate = d3.time.format('%Y-%m-%d').parse;
- // Create SVG
- // ------------------------------
- // Container
- var container = d3Container.append('svg');
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
- // Construct chart layout
- // ------------------------------
- // Area
- var area = d3.svg.area()
- .x(function(d) { return x(d.date); })
- .y0(height)
- .y1(function(d) { return y(d.value); })
- .interpolate('monotone')
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale().range([0, width ]);
- // Vertical
- var y = d3.scale.linear().range([height, 0]);
- // Load data
- // ------------------------------
- d3.json('../../../../global_assets/demo_data/dashboard/monthly_sales.json', function (error, data) {
- // Show what's wrong if error
- if (error) return console.error(error);
- // Pull out values
- data.forEach(function (d) {
- d.date = parseDate(d.date);
- d.value = +d.value;
- });
- // Get the maximum value in the given array
- var maxY = d3.max(data, function(d) { return d.value; });
- // Reset start data for animation
- var startData = data.map(function(datum) {
- return {
- date: datum.date,
- value: 0
- };
- });
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain(d3.extent(data, function(d, i) { return d.date; }));
- // Vertical
- y.domain([0, d3.max( data, function(d) { return d.value; })]);
- //
- // Append chart elements
- //
- // Add area path
- svg.append('path')
- .datum(data)
- .attr('class', 'd3-area')
- .style('fill', color)
- .attr('d', area)
- .transition() // begin animation
- .duration(1000)
- .attrTween('d', function() {
- var interpolator = d3.interpolateArray(startData, data);
- return function (t) {
- return area(interpolator (t));
- }
- });
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', messagesAreaResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', messagesAreaResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function messagesAreaResize() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Horizontal range
- x.range([0, width]);
- // Chart elements
- // -------------------------
- // Area path
- svg.selectAll('.d3-area').datum( data ).attr('d', area);
- }
- });
- }
- };
- // Sparklines chart
- var _chartSparkline = function(element, chartType, qty, height, interpolation, duration, interval, color) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Define main variables
- var d3Container = d3.select(element),
- margin = {top: 0, right: 0, bottom: 0, left: 0},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom;
- // Generate random data (for demo only)
- var data = [];
- for (var i=0; i < qty; i++) {
- data.push(Math.floor(Math.random() * qty) + 5)
- }
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.scale.linear().range([0, width]);
- // Vertical
- var y = d3.scale.linear().range([height - 5, 5]);
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain([1, qty - 3])
- // Vertical
- y.domain([0, qty])
- // Construct chart layout
- // ------------------------------
- // Line
- var line = d3.svg.line()
- .interpolate(interpolation)
- .x(function(d, i) { return x(i); })
- .y(function(d, i) { return y(d); });
- // Area
- var area = d3.svg.area()
- .interpolate(interpolation)
- .x(function(d, i) {
- return x(i);
- })
- .y0(height)
- .y1(function(d) {
- return y(d);
- });
- // Create SVG
- // ------------------------------
- // Container
- var container = d3Container.append('svg');
- // SVG element
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append("g")
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');
- // Add mask for animation
- // ------------------------------
- // Add clip path
- var clip = svg.append('defs')
- .append('clipPath')
- .attr('id', function(d, i) { return 'load-clip-' + element.substring(1) })
- // Add clip shape
- var clips = clip.append('rect')
- .attr('class', 'load-clip')
- .attr('width', 0)
- .attr('height', height);
- // Animate mask
- clips
- .transition()
- .duration(1000)
- .ease('linear')
- .attr('width', width);
- //
- // Append chart elements
- //
- // Main path
- var path = svg.append('g')
- .attr('clip-path', function(d, i) { return 'url(#load-clip-' + element.substring(1) + ')'})
- .append('path')
- .datum(data)
- .attr('transform', 'translate(' + x(0) + ',0)');
- // Add path based on chart type
- if(chartType == 'area') {
- path.attr('d', area).attr('class', 'd3-area').style('fill', color); // area
- }
- else {
- path.attr('d', line).attr('class', 'd3-line d3-line-medium').style('stroke', color); // line
- }
- // Animate path
- path
- .style('opacity', 0)
- .transition()
- .duration(750)
- .style('opacity', 1);
- // Set update interval. For demo only
- // ------------------------------
- setInterval(function() {
- // push a new data point onto the back
- data.push(Math.floor(Math.random() * qty) + 5);
- // pop the old data point off the front
- data.shift();
- update();
- }, interval);
- // Update random data. For demo only
- // ------------------------------
- function update() {
- // Redraw the path and slide it to the left
- path
- .attr('transform', null)
- .transition()
- .duration(duration)
- .ease('linear')
- .attr('transform', 'translate(' + x(0) + ',0)');
- // Update path type
- if(chartType == 'area') {
- path.attr('d', area).attr('class', 'd3-area').style('fill', color)
- }
- else {
- path.attr('d', line).attr('class', 'd3-line d3-line-medium').style('stroke', color);
- }
- }
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', resizeSparklines);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', resizeSparklines);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function resizeSparklines() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Horizontal range
- x.range([0, width]);
- // Chart elements
- // -------------------------
- // Clip mask
- clips.attr('width', width);
- // Line
- svg.select('.d3-line').attr('d', line);
- // Area
- svg.select('.d3-area').attr('d', area);
- }
- }
- };
- // Daily revenue line chart
- var _DailyRevenueLineChart = function(element, height) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Add data set
- var dataset = [
- {
- 'date': '04/13/14',
- 'alpha': '60'
- }, {
- 'date': '04/14/14',
- 'alpha': '35'
- }, {
- 'date': '04/15/14',
- 'alpha': '65'
- }, {
- 'date': '04/16/14',
- 'alpha': '50'
- }, {
- 'date': '04/17/14',
- 'alpha': '65'
- }, {
- 'date': '04/18/14',
- 'alpha': '20'
- }, {
- 'date': '04/19/14',
- 'alpha': '60'
- }
- ];
- // Main variables
- var d3Container = d3.select(element),
- margin = {top: 0, right: 0, bottom: 0, left: 0},
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom,
- padding = 20;
- // Format date
- var parseDate = d3.time.format('%m/%d/%y').parse,
- formatDate = d3.time.format('%a, %B %e');
- // Add tooltip
- // ------------------------------
- var tooltip = d3.tip()
- .attr('class', 'd3-tip')
- .html(function (d) {
- return '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base my-1"><i class="icon-check2 mr-2"></i>' + formatDate(d.date) + '</div>' + '</li>' +
- '<li>' + 'Sales: ' + '<span class="font-weight-semibold float-right">' + d.alpha + '</span>' + '</li>' +
- '<li>' + 'Revenue: ' + '<span class="font-weight-semibold float-right">' + '$' + (d.alpha * 25).toFixed(2) + '</span>' + '</li>' +
- '</ul>';
- });
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg');
- // Add SVG group
- var svg = container
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
- .call(tooltip);
- // Load data
- // ------------------------------
- dataset.forEach(function (d) {
- d.date = parseDate(d.date);
- d.alpha = +d.alpha;
- });
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.time.scale()
- .range([padding, width - padding]);
- // Vertical
- var y = d3.scale.linear()
- .range([height, 5]);
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain(d3.extent(dataset, function (d) {
- return d.date;
- }));
- // Vertical
- y.domain([0, d3.max(dataset, function (d) {
- return Math.max(d.alpha);
- })]);
- // Construct chart layout
- // ------------------------------
- // Line
- var line = d3.svg.line()
- .x(function(d) {
- return x(d.date);
- })
- .y(function(d) {
- return y(d.alpha)
- });
- //
- // Append chart elements
- //
- // Add mask for animation
- // ------------------------------
- // Add clip path
- var clip = svg.append('defs')
- .append('clipPath')
- .attr('id', 'clip-line-small');
- // Add clip shape
- var clipRect = clip.append('rect')
- .attr('class', 'clip')
- .attr('width', 0)
- .attr('height', height);
- // Animate mask
- clipRect
- .transition()
- .duration(1000)
- .ease('linear')
- .attr('width', width);
- // Line
- // ------------------------------
- // Path
- var path = svg.append('path')
- .attr({
- 'd': line(dataset),
- 'clip-path': 'url(#clip-line-small)',
- 'class': 'd3-line d3-line-medium'
- })
- .style('stroke', '#fff');
- // Animate path
- svg.select('.line-tickets')
- .transition()
- .duration(1000)
- .ease('linear');
- // Add vertical guide lines
- // ------------------------------
- // Bind data
- var guide = svg.append('g')
- .selectAll('.d3-line-guides-group')
- .data(dataset);
- // Append lines
- guide
- .enter()
- .append('line')
- .attr('class', 'd3-line-guides')
- .attr('x1', function (d, i) {
- return x(d.date);
- })
- .attr('y1', function (d, i) {
- return height;
- })
- .attr('x2', function (d, i) {
- return x(d.date);
- })
- .attr('y2', function (d, i) {
- return height;
- })
- .style('stroke', 'rgba(255,255,255,0.3)')
- .style('stroke-dasharray', '4,2')
- .style('shape-rendering', 'crispEdges');
- // Animate guide lines
- guide
- .transition()
- .duration(1000)
- .delay(function(d, i) { return i * 150; })
- .attr('y2', function (d, i) {
- return y(d.alpha);
- });
- // Alpha app points
- // ------------------------------
- // Add points
- var points = svg.insert('g')
- .selectAll('.d3-line-circle')
- .data(dataset)
- .enter()
- .append('circle')
- .attr('class', 'd3-line-circle d3-line-circle-medium')
- .attr('cx', line.x())
- .attr('cy', line.y())
- .attr('r', 3)
- .style('stroke', '#fff')
- .style('fill', '#29B6F6');
- // Animate points on page load
- points
- .style('opacity', 0)
- .transition()
- .duration(250)
- .ease('linear')
- .delay(1000)
- .style('opacity', 1);
- // Add user interaction
- points
- .on('mouseover', function (d) {
- tooltip.offset([-10, 0]).show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- // Hide tooltip
- .on('mouseout', function (d) {
- tooltip.hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- });
- // Change tooltip direction of first point
- d3.select(points[0][0])
- .on('mouseover', function (d) {
- tooltip.offset([0, 10]).direction('e').show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- .on('mouseout', function (d) {
- tooltip.direction('n').hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- });
- // Change tooltip direction of last point
- d3.select(points[0][points.size() - 1])
- .on('mouseover', function (d) {
- tooltip.offset([0, -10]).direction('w').show(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 4);
- })
- .on('mouseout', function (d) {
- tooltip.direction('n').hide(d);
- // Animate circle radius
- d3.select(this).transition().duration(250).attr('r', 3);
- })
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', revenueResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', revenueResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function revenueResize() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Horizontal range
- x.range([padding, width - padding]);
- // Chart elements
- // -------------------------
- // Mask
- clipRect.attr('width', width);
- // Line path
- svg.selectAll('.d3-line').attr('d', line(dataset));
- // Circles
- svg.selectAll('.d3-line-circle').attr('cx', line.x());
- // Guide lines
- svg.selectAll('.d3-line-guides')
- .attr('x1', function (d, i) {
- return x(d.date);
- })
- .attr('x2', function (d, i) {
- return x(d.date);
- });
- }
- }
- };
- // Small progress pie chart
- var _ProgressPieChart = function(element, width, height, color) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Main variables
- var d3Container = d3.select(element),
- border = 2,
- radius = Math.min(width / 2, height / 2) - border,
- twoPi = 2 * Math.PI,
- progress = $(element).data('progress'),
- total = 100;
- // Construct chart layout
- // ------------------------------
- // Arc
- var arc = d3.svg.arc()
- .startAngle(0)
- .innerRadius(0)
- .outerRadius(radius)
- .endAngle(function(d) {
- return (d.value / d.size) * 2 * Math.PI;
- })
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg');
- // Add SVG group
- var svg = container
- .attr('width', width)
- .attr('height', height)
- .append('g')
- .attr('transform', 'translate(' + width / 2 + ',' + height / 2 + ')');
- //
- // Append chart elements
- //
- // Progress group
- var meter = svg.append('g')
- .attr('class', 'progress-meter');
- // Background
- meter.append('path')
- .attr('d', arc.endAngle(twoPi))
- .style('fill', '#fff')
- .style('stroke', color)
- .style('stroke-width', 1.5);
- // Foreground
- var foreground = meter.append('path')
- .style('fill', color);
- // Animate foreground path
- foreground
- .transition()
- .ease('cubic-out')
- .duration(2500)
- .attrTween('d', arcTween);
- // Tween arcs
- function arcTween() {
- var i = d3.interpolate(0, progress);
- return function(t) {
- var currentProgress = progress / (100/t);
- var endAngle = arc.endAngle(twoPi * (currentProgress));
- return arc(i(endAngle));
- };
- }
- }
- };
- // Marketing campaigns donut chart
- var _MarketingCampaignsDonutChart = function(element, size) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Add data set
- var data = [
- {
- "browser": "Google Adwords",
- "icon": "<i class='icon-google mr-2'></i>",
- "value": 1047,
- "color" : "#66BB6A"
- }, {
- "browser": "Social media",
- "icon": "<i class='icon-share4 mr-2'></i>",
- "value": 2948,
- "color": "#9575CD"
- }, {
- "browser": "Youtube video",
- "icon": "<i class='icon-youtube mr-2'></i>",
- "value": 3909,
- "color": "#FF7043"
- }
- ];
- // Main variables
- var d3Container = d3.select(element),
- distance = 2, // reserve 2px space for mouseover arc moving
- radius = (size/2) - distance,
- sum = d3.sum(data, function(d) { return d.value; });
- // Tooltip
- // ------------------------------
- var tip = d3.tip()
- .attr('class', 'd3-tip')
- .offset([-10, 0])
- .direction('e')
- .html(function (d) {
- return '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base mb-1 mt-1">' + d.data.icon + d.data.browser + '</div>' + '</li>' +
- '<li>' + 'Visits: ' + '<span class="font-weight-semibold float-right">' + d.value + '</span>' + '</li>' +
- '<li>' + 'Share: ' + '<span class="font-weight-semibold float-right">' + (100 / (sum / d.value)).toFixed(2) + '%' + '</span>' + '</li>' +
- '</ul>';
- });
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg').call(tip);
- // Add SVG group
- var svg = container
- .attr('width', size)
- .attr('height', size)
- .append('g')
- .attr('transform', 'translate(' + (size / 2) + ',' + (size / 2) + ')');
- // Construct chart layout
- // ------------------------------
- // Pie
- var pie = d3.layout.pie()
- .sort(null)
- .startAngle(Math.PI)
- .endAngle(3 * Math.PI)
- .value(function (d) {
- return d.value;
- });
- // Arc
- var arc = d3.svg.arc()
- .outerRadius(radius)
- .innerRadius(radius / 2);
- //
- // Append chart elements
- //
- // Group chart elements
- var arcGroup = svg.selectAll('.d3-arc')
- .data(pie(data))
- .enter()
- .append('g')
- .attr('class', 'd3-arc')
- .style('stroke', '#fff')
- .style('cursor', 'pointer');
- // Append path
- var arcPath = arcGroup
- .append('path')
- .style('fill', function (d) { return d.data.color; });
- // Add tooltip
- arcPath
- .on('mouseover', function (d, i) {
- // Transition on mouseover
- d3.select(this)
- .transition()
- .duration(500)
- .ease('elastic')
- .attr('transform', function (d) {
- d.midAngle = ((d.endAngle - d.startAngle) / 2) + d.startAngle;
- var x = Math.sin(d.midAngle) * distance;
- var y = -Math.cos(d.midAngle) * distance;
- return 'translate(' + x + ',' + y + ')';
- });
- })
- .on('mousemove', function (d) {
- // Show tooltip on mousemove
- tip.show(d)
- .style('top', (d3.event.pageY - 40) + 'px')
- .style('left', (d3.event.pageX + 30) + 'px');
- })
- .on('mouseout', function (d, i) {
- // Mouseout transition
- d3.select(this)
- .transition()
- .duration(500)
- .ease('bounce')
- .attr('transform', 'translate(0,0)');
- // Hide tooltip
- tip.hide(d);
- });
- // Animate chart on load
- arcPath
- .transition()
- .delay(function(d, i) { return i * 500; })
- .duration(500)
- .attrTween('d', function(d) {
- var interpolate = d3.interpolate(d.startAngle,d.endAngle);
- return function(t) {
- d.endAngle = interpolate(t);
- return arc(d);
- };
- });
- }
- };
- // Campaign status donut chart
- var _CampaignStatusDonutChart = function(element, size) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Add data set
- var data = [
- {
- "status": "Active campaigns",
- "icon": "<span class='status-mark border-blue-300 mr-2'></span>",
- "value": 439,
- "color": "#29B6F6"
- }, {
- "status": "Closed campaigns",
- "icon": "<span class='status-mark border-danger-300 mr-2'></span>",
- "value": 290,
- "color": "#EF5350"
- }, {
- "status": "Pending campaigns",
- "icon": "<span class='status-mark border-success-300 mr-2'></span>",
- "value": 190,
- "color": "#81C784"
- }, {
- "status": "Campaigns on hold",
- "icon": "<span class='status-mark border-grey-300 mr-2'></span>",
- "value": 148,
- "color": "#999"
- }
- ];
- // Main variables
- var d3Container = d3.select(element),
- distance = 2, // reserve 2px space for mouseover arc moving
- radius = (size/2) - distance,
- sum = d3.sum(data, function(d) { return d.value; })
- // Tooltip
- // ------------------------------
- var tip = d3.tip()
- .attr('class', 'd3-tip')
- .offset([-10, 0])
- .direction('e')
- .html(function (d) {
- return '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base mb-1 mt-1">' + d.data.icon + d.data.status + '</div>' + '</li>' +
- '<li>' + 'Total: ' + '<span class="font-weight-semibold float-right">' + d.value + '</span>' + '</li>' +
- '<li>' + 'Share: ' + '<span class="font-weight-semibold float-right">' + (100 / (sum / d.value)).toFixed(2) + '%' + '</span>' + '</li>' +
- '</ul>';
- });
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg').call(tip);
- // Add SVG group
- var svg = container
- .attr('width', size)
- .attr('height', size)
- .append('g')
- .attr('transform', 'translate(' + (size / 2) + ',' + (size / 2) + ')');
- // Construct chart layout
- // ------------------------------
- // Pie
- var pie = d3.layout.pie()
- .sort(null)
- .startAngle(Math.PI)
- .endAngle(3 * Math.PI)
- .value(function (d) {
- return d.value;
- });
- // Arc
- var arc = d3.svg.arc()
- .outerRadius(radius)
- .innerRadius(radius / 2);
- //
- // Append chart elements
- //
- // Group chart elements
- var arcGroup = svg.selectAll('.d3-arc')
- .data(pie(data))
- .enter()
- .append('g')
- .attr('class', 'd3-arc')
- .style('stroke', '#fff')
- .style('cursor', 'pointer');
- // Append path
- var arcPath = arcGroup
- .append('path')
- .style('fill', function (d) { return d.data.color; });
- // Add tooltip
- arcPath
- .on('mouseover', function (d, i) {
- // Transition on mouseover
- d3.select(this)
- .transition()
- .duration(500)
- .ease('elastic')
- .attr('transform', function (d) {
- d.midAngle = ((d.endAngle - d.startAngle) / 2) + d.startAngle;
- var x = Math.sin(d.midAngle) * distance;
- var y = -Math.cos(d.midAngle) * distance;
- return 'translate(' + x + ',' + y + ')';
- });
- })
- .on('mousemove', function (d) {
- // Show tooltip on mousemove
- tip.show(d)
- .style('top', (d3.event.pageY - 40) + 'px')
- .style('left', (d3.event.pageX + 30) + 'px');
- })
- .on('mouseout', function (d, i) {
- // Mouseout transition
- d3.select(this)
- .transition()
- .duration(500)
- .ease('bounce')
- .attr('transform', 'translate(0,0)');
- // Hide tooltip
- tip.hide(d);
- });
- // Animate chart on load
- arcPath
- .transition()
- .delay(function(d, i) { return i * 500; })
- .duration(500)
- .attrTween('d', function(d) {
- var interpolate = d3.interpolate(d.startAngle,d.endAngle);
- return function(t) {
- d.endAngle = interpolate(t);
- return arc(d);
- };
- });
- }
- };
- // Tickets status donut chart
- var _TicketStatusDonutChart = function(element, size) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Add data set
- var data = [
- {
- "status": "Pending tickets",
- "icon": "<i class='status-mark border-blue-300 mr-2'></i>",
- "value": 295,
- "color": "#29B6F6"
- }, {
- "status": "Resolved tickets",
- "icon": "<i class='status-mark border-success-300 mr-2'></i>",
- "value": 189,
- "color": "#66BB6A"
- }, {
- "status": "Closed tickets",
- "icon": "<i class='status-mark border-danger-300 mr-2'></i>",
- "value": 277,
- "color": "#EF5350"
- }
- ];
- // Main variables
- var d3Container = d3.select(element),
- distance = 2, // reserve 2px space for mouseover arc moving
- radius = (size/2) - distance,
- sum = d3.sum(data, function(d) { return d.value; })
- // Tooltip
- // ------------------------------
- var tip = d3.tip()
- .attr('class', 'd3-tip')
- .offset([-10, 0])
- .direction('e')
- .html(function (d) {
- return '<ul class="list-unstyled mb-1">' +
- '<li>' + '<div class="font-size-base mb-1 mt-1">' + d.data.icon + d.data.status + '</div>' + '</li>' +
- '<li>' + 'Total: ' + '<span class="font-weight-semibold float-right">' + d.value + '</span>' + '</li>' +
- '<li>' + 'Share: ' + '<span class="font-weight-semibold float-right">' + (100 / (sum / d.value)).toFixed(2) + '%' + '</span>' + '</li>' +
- '</ul>';
- })
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg').call(tip);
- // Add SVG group
- var svg = container
- .attr('width', size)
- .attr('height', size)
- .append('g')
- .attr('transform', 'translate(' + (size / 2) + ',' + (size / 2) + ')');
- // Construct chart layout
- // ------------------------------
- // Pie
- var pie = d3.layout.pie()
- .sort(null)
- .startAngle(Math.PI)
- .endAngle(3 * Math.PI)
- .value(function (d) {
- return d.value;
- });
- // Arc
- var arc = d3.svg.arc()
- .outerRadius(radius)
- .innerRadius(radius / 2);
- //
- // Append chart elements
- //
- // Group chart elements
- var arcGroup = svg.selectAll('.d3-arc')
- .data(pie(data))
- .enter()
- .append('g')
- .attr('class', 'd3-arc')
- .style('stroke', '#fff')
- .style('cursor', 'pointer');
- // Append path
- var arcPath = arcGroup
- .append('path')
- .style('fill', function (d) { return d.data.color; });
- // Add tooltip
- arcPath
- .on('mouseover', function (d, i) {
- // Transition on mouseover
- d3.select(this)
- .transition()
- .duration(500)
- .ease('elastic')
- .attr('transform', function (d) {
- d.midAngle = ((d.endAngle - d.startAngle) / 2) + d.startAngle;
- var x = Math.sin(d.midAngle) * distance;
- var y = -Math.cos(d.midAngle) * distance;
- return 'translate(' + x + ',' + y + ')';
- });
- })
- .on('mousemove', function (d) {
- // Show tooltip on mousemove
- tip.show(d)
- .style('top', (d3.event.pageY - 40) + 'px')
- .style('left', (d3.event.pageX + 30) + 'px');
- })
- .on('mouseout', function (d, i) {
- // Mouseout transition
- d3.select(this)
- .transition()
- .duration(500)
- .ease('bounce')
- .attr('transform', 'translate(0,0)');
- // Hide tooltip
- tip.hide(d);
- });
- // Animate chart on load
- arcPath
- .transition()
- .delay(function(d, i) { return i * 500; })
- .duration(500)
- .attrTween('d', function(d) {
- var interpolate = d3.interpolate(d.startAngle,d.endAngle);
- return function(t) {
- d.endAngle = interpolate(t);
- return arc(d);
- };
- });
- }
- };
- // Bar charts
- var _BarChart = function(element, barQty, height, animate, easing, duration, delay, color, tooltip) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Add data set
- var bardata = [];
- for (var i=0; i < barQty; i++) {
- bardata.push(Math.round(Math.random()*10) + 10);
- }
- // Main variables
- var d3Container = d3.select(element),
- width = d3Container.node().getBoundingClientRect().width;
- // Construct scales
- // ------------------------------
- // Horizontal
- var x = d3.scale.ordinal()
- .rangeBands([0, width], 0.3);
- // Vertical
- var y = d3.scale.linear()
- .range([0, height]);
- // Set input domains
- // ------------------------------
- // Horizontal
- x.domain(d3.range(0, bardata.length));
- // Vertical
- y.domain([0, d3.max(bardata)]);
- // Create chart
- // ------------------------------
- // Add svg element
- var container = d3Container.append('svg');
- // Add SVG group
- var svg = container
- .attr('width', width)
- .attr('height', height)
- .append('g');
- //
- // Append chart elements
- //
- // Bars
- var bars = svg.selectAll('rect')
- .data(bardata)
- .enter()
- .append('rect')
- .attr('class', 'd3-random-bars')
- .attr('width', x.rangeBand())
- .attr('x', function(d,i) {
- return x(i);
- })
- .style('fill', color);
- // Tooltip
- // ------------------------------
- var tip = d3.tip()
- .attr('class', 'd3-tip')
- .offset([-10, 0]);
- // Show and hide
- if(tooltip == 'hours' || tooltip == 'goal' || tooltip == 'members') {
- bars.call(tip)
- .on('mouseover', tip.show)
- .on('mouseout', tip.hide);
- }
- // Daily meetings tooltip content
- if(tooltip == 'hours') {
- tip.html(function (d, i) {
- return '<div class="text-center">' +
- '<h6 class="m-0">' + d + '</h6>' +
- '<span class="font-size-sm">meetings</span>' +
- '<div class="font-size-sm">' + i + ':00' + '</div>' +
- '</div>'
- });
- }
- // Statements tooltip content
- if(tooltip == 'goal') {
- tip.html(function (d, i) {
- return '<div class="text-center">' +
- '<h6 class="m-0">' + d + '</h6>' +
- '<span class="font-size-sm">statements</span>' +
- '<div class="font-size-sm">' + i + ':00' + '</div>' +
- '</div>'
- });
- }
- // Online members tooltip content
- if(tooltip == 'members') {
- tip.html(function (d, i) {
- return '<div class="text-center">' +
- '<h6 class="m-0">' + d + '0' + '</h6>' +
- '<span class="font-size-sm">members</span>' +
- '<div class="font-size-sm">' + i + ':00' + '</div>' +
- '</div>'
- });
- }
- // Bar loading animation
- // ------------------------------
- // Choose between animated or static
- if(animate) {
- withAnimation();
- } else {
- withoutAnimation();
- }
- // Animate on load
- function withAnimation() {
- bars
- .attr('height', 0)
- .attr('y', height)
- .transition()
- .attr('height', function(d) {
- return y(d);
- })
- .attr('y', function(d) {
- return height - y(d);
- })
- .delay(function(d, i) {
- return i * delay;
- })
- .duration(duration)
- .ease(easing);
- }
- // Load without animateion
- function withoutAnimation() {
- bars
- .attr('height', function(d) {
- return y(d);
- })
- .attr('y', function(d) {
- return height - y(d);
- })
- }
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', barsResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', barsResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function barsResize() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width);
- // Width of appended group
- svg.attr('width', width);
- // Horizontal range
- x.rangeBands([0, width], 0.3);
- // Chart elements
- // -------------------------
- // Bars
- svg.selectAll('.d3-random-bars')
- .attr('width', x.rangeBand())
- .attr('x', function(d,i) {
- return x(i);
- });
- }
- }
- };
- // Rounded progress charts
- var _RoundedProgressChart = function(element, radius, border, color, end, iconClass, textTitle, textAverage) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Basic setup
- // ------------------------------
- // Main variables
- var d3Container = d3.select(element),
- startPercent = 0,
- iconSize = 32,
- endPercent = end,
- twoPi = Math.PI * 2,
- formatPercent = d3.format('.0%'),
- boxSize = radius * 2;
- // Values count
- var count = Math.abs((endPercent - startPercent) / 0.01);
- // Values step
- var step = endPercent < startPercent ? -0.01 : 0.01;
- // Create chart
- // ------------------------------
- // Add SVG element
- var container = d3Container.append('svg');
- // Add SVG group
- var svg = container
- .attr('width', boxSize)
- .attr('height', boxSize)
- .append('g')
- .attr('transform', 'translate(' + (boxSize / 2) + ',' + (boxSize / 2) + ')');
- // Construct chart layout
- // ------------------------------
- // Arc
- var arc = d3.svg.arc()
- .startAngle(0)
- .innerRadius(radius)
- .outerRadius(radius - border);
- //
- // Append chart elements
- //
- // Paths
- // ------------------------------
- // Background path
- svg.append('path')
- .attr('class', 'd3-progress-background')
- .attr('d', arc.endAngle(twoPi))
- .style('fill', '#eee');
- // Foreground path
- var foreground = svg.append('path')
- .attr('class', 'd3-progress-foreground')
- .attr('filter', 'url(#blur)')
- .style('fill', color)
- .style('stroke', color);
- // Front path
- var front = svg.append('path')
- .attr('class', 'd3-progress-front')
- .style('fill', color)
- .style('fill-opacity', 1);
- // Text
- // ------------------------------
- // Percentage text value
- var numberText = d3.select(element)
- .append('h2')
- .attr('class', 'pt-1 mt-2 mb-1')
- // Icon
- d3.select(element)
- .append('i')
- .attr('class', iconClass + ' counter-icon')
- .attr('style', 'top: ' + ((boxSize - iconSize) / 2) + 'px');
- // Title
- d3.select(element)
- .append('div')
- .text(textTitle);
- // Subtitle
- d3.select(element)
- .append('div')
- .attr('class', 'font-size-sm text-muted mb-3')
- .text(textAverage);
- // Animation
- // ------------------------------
- // Animate path
- function updateProgress(progress) {
- foreground.attr('d', arc.endAngle(twoPi * progress));
- front.attr('d', arc.endAngle(twoPi * progress));
- numberText.text(formatPercent(progress));
- }
- // Animate text
- var progress = startPercent;
- (function loops() {
- updateProgress(progress);
- if (count > 0) {
- count--;
- progress += step;
- setTimeout(loops, 10);
- }
- })();
- }
- };
- // Bullet chart
- var _BulletChart = function(element, height) {
- if (typeof d3 == 'undefined') {
- console.warn('Warning - d3.min.js is not loaded.');
- return;
- }
- // Initialize chart only if element exsists in the DOM
- if($(element).length > 0) {
- // Bullet chart core
- // ------------------------------
- function bulletCore() {
- // Construct
- d3.bullet = function() {
- // Default layout variables
- var orient = 'left',
- reverse = false,
- duration = 750,
- ranges = bulletRanges,
- markers = bulletMarkers,
- measures = bulletMeasures,
- height = 30,
- tickFormat = null;
- // For each small multiple…
- function bullet(g) {
- g.each(function(d, i) {
- // Define variables
- var rangez = ranges.call(this, d, i).slice().sort(d3.descending),
- markerz = markers.call(this, d, i).slice().sort(d3.descending),
- measurez = measures.call(this, d, i).slice().sort(d3.descending),
- g = d3.select(this);
- // Compute the new x-scale.
- var x1 = d3.scale.linear()
- .domain([0, Math.max(rangez[0], markerz[0], measurez[0])])
- .range(reverse ? [width, 0] : [0, width]);
- // Retrieve the old x-scale, if this is an update.
- var x0 = this.__chart__ || d3.scale.linear()
- .domain([0, Infinity])
- .range(x1.range());
- // Stash the new scale.
- this.__chart__ = x1;
- // Derive width-scales from the x-scales.
- var w0 = bulletWidth(x0),
- w1 = bulletWidth(x1);
- // Setup range
- // ------------------------------
- // Update the range rects
- var range = g.selectAll('.bullet-range')
- .data(rangez);
- // Append range rect
- range.enter()
- .append('rect')
- .attr('class', function(d, i) { return 'bullet-range bullet-range-' + (i + 1); })
- .attr('width', w0)
- .attr('height', height)
- .attr('rx', 2)
- .attr('x', reverse ? x0 : 0)
- // Add loading animation
- .transition()
- .duration(duration)
- .attr('width', w1)
- .attr('x', reverse ? x1 : 0);
- // Add update animation
- range.transition()
- .duration(duration)
- .attr('x', reverse ? x1 : 0)
- .attr('width', w1)
- .attr('height', height);
- // Setup measures
- // ------------------------------
- // Update the measure rects
- var measure = g.selectAll('.bullet-measure')
- .data(measurez);
- // Append measure rect
- measure.enter()
- .append('rect')
- .attr('class', function(d, i) { return 'bullet-measure bullet-measure-' + (i + 1); })
- .attr('width', w0)
- .attr('height', height / 5)
- .attr('x', reverse ? x0 : 0)
- .attr('y', height / 2.5)
- .style('shape-rendering', 'crispEdges');
- // Add loading animation
- measure.transition()
- .duration(duration)
- .attr('width', w1)
- .attr('x', reverse ? x1 : 0);
- // Add update animation
- measure.transition()
- .duration(duration)
- .attr('width', w1)
- .attr('height', height / 5)
- .attr('x', reverse ? x1 : 0)
- .attr('y', height / 2.5);
- // Setup markers
- // ------------------------------
- // Update the marker lines
- var marker = g.selectAll('.bullet-marker')
- .data(markerz);
- // Append marker line
- marker.enter()
- .append('line')
- .attr('class', function(d, i) { return 'bullet-marker bullet-marker-' + (i + 1); })
- .attr('x1', x0)
- .attr('x2', x0)
- .attr('y1', height / 6)
- .attr('y2', height * 5 / 6);
- // Add loading animation
- marker.transition()
- .duration(duration)
- .attr('x1', x1)
- .attr('x2', x1);
- // Add update animation
- marker.transition()
- .duration(duration)
- .attr('x1', x1)
- .attr('x2', x1)
- .attr('y1', height / 6)
- .attr('y2', height * 5 / 6);
- // Setup axes
- // ------------------------------
- // Compute the tick format.
- var format = tickFormat || x1.tickFormat(8);
- // Update the tick groups.
- var tick = g.selectAll('.bullet-tick')
- .data(x1.ticks(8), function(d) {
- return this.textContent || format(d);
- });
- // Initialize the ticks with the old scale, x0.
- var tickEnter = tick.enter()
- .append('g')
- .attr('class', 'bullet-tick')
- .attr('transform', bulletTranslate(x0))
- .style('opacity', 1e-6);
- // Append line
- tickEnter.append('line')
- .attr('y1', height)
- .attr('y2', (height * 7 / 6) + 3);
- // Append text
- tickEnter.append('text')
- .attr('text-anchor', 'middle')
- .attr('dy', '1em')
- .attr('y', (height * 7 / 6) + 4)
- .text(format);
- // Transition the entering ticks to the new scale, x1.
- tickEnter.transition()
- .duration(duration)
- .attr('transform', bulletTranslate(x1))
- .style('opacity', 1);
- // Transition the updating ticks to the new scale, x1.
- var tickUpdate = tick.transition()
- .duration(duration)
- .attr('transform', bulletTranslate(x1))
- .style('opacity', 1);
- // Update tick line
- tickUpdate.select('line')
- .attr('y1', height + 3)
- .attr('y2', (height * 7 / 6) + 3);
- // Update tick text
- tickUpdate.select('text')
- .attr('y', (height * 7 / 6) + 4);
- // Transition the exiting ticks to the new scale, x1.
- tick.exit()
- .transition()
- .duration(duration)
- .attr('transform', bulletTranslate(x1))
- .style('opacity', 1e-6)
- .remove();
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', resizeBulletsCore);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', resizeBulletsCore);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function resizeBulletsCore() {
- // Layout variables
- width = d3.select('#bullets').node().getBoundingClientRect().width - margin.left - margin.right;
- w1 = bulletWidth(x1);
- // Layout
- // -------------------------
- // Horizontal range
- x1.range(reverse ? [width, 0] : [0, width]);
- // Chart elements
- // -------------------------
- // Measures
- g.selectAll('.bullet-measure').attr('width', w1).attr('x', reverse ? x1 : 0);
- // Ranges
- g.selectAll('.bullet-range').attr('width', w1).attr('x', reverse ? x1 : 0);
- // Markers
- g.selectAll('.bullet-marker').attr('x1', x1).attr('x2', x1)
- // Ticks
- g.selectAll('.bullet-tick').attr('transform', bulletTranslate(x1))
- }
- });
- d3.timer.flush();
- }
- // Constructor functions
- // ------------------------------
- // Left, right, top, bottom
- bullet.orient = function(x) {
- if (!arguments.length) return orient;
- orient = x;
- reverse = orient == 'right' || orient == 'bottom';
- return bullet;
- };
- // Ranges (bad, satisfactory, good)
- bullet.ranges = function(x) {
- if (!arguments.length) return ranges;
- ranges = x;
- return bullet;
- };
- // Markers (previous, goal)
- bullet.markers = function(x) {
- if (!arguments.length) return markers;
- markers = x;
- return bullet;
- };
- // Measures (actual, forecast)
- bullet.measures = function(x) {
- if (!arguments.length) return measures;
- measures = x;
- return bullet;
- };
- // Width
- bullet.width = function(x) {
- if (!arguments.length) return width;
- width = x;
- return bullet;
- };
- // Height
- bullet.height = function(x) {
- if (!arguments.length) return height;
- height = x;
- return bullet;
- };
- // Axex tick format
- bullet.tickFormat = function(x) {
- if (!arguments.length) return tickFormat;
- tickFormat = x;
- return bullet;
- };
- // Transition duration
- bullet.duration = function(x) {
- if (!arguments.length) return duration;
- duration = x;
- return bullet;
- };
- return bullet;
- };
- // Ranges
- function bulletRanges(d) {
- return d.ranges;
- }
- // Markers
- function bulletMarkers(d) {
- return d.markers;
- }
- // Measures
- function bulletMeasures(d) {
- return d.measures;
- }
- // Positioning
- function bulletTranslate(x) {
- return function(d) {
- return 'translate(' + x(d) + ',0)';
- };
- }
- // Width
- function bulletWidth(x) {
- var x0 = x(0);
- return function(d) {
- return Math.abs(x(d) - x0);
- };
- }
- }
- bulletCore();
- // Basic setup
- // ------------------------------
- // Main variables
- var d3Container = d3.select(element),
- margin = {top: 20, right: 10, bottom: 35, left: 10},
- width = width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right,
- height = height - margin.top - margin.bottom;
- // Construct chart layout
- // ------------------------------
- var chart = d3.bullet()
- .width(width)
- .height(height);
- // Load data
- // ------------------------------
- d3.json('../../../../global_assets/demo_data/dashboard/bullets.json', function(error, data) {
- // Show what's wrong if error
- if (error) return console.error(error);
- // Create SVG
- // ------------------------------
- // SVG container
- var container = d3Container.selectAll('svg')
- .data(data)
- .enter()
- .append('svg');
- // SVG group
- var svg = container
- .attr('class', function(d, i) { return 'bullet-' + (i + 1); })
- .attr('width', width + margin.left + margin.right)
- .attr('height', height + margin.top + margin.bottom)
- .append('g')
- .attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
- .call(chart);
- // Add title
- // ------------------------------
- // Title group
- var title = svg.append('g')
- .style('text-anchor', 'start');
- // Bullet title text
- title.append('text')
- .attr('class', 'bullet-title')
- .attr('y', -10)
- .text(function(d) { return d.title; });
- // Bullet subtitle text
- title.append('text')
- .attr('class', 'bullet-subtitle')
- .attr('x', width)
- .attr('y', -10)
- .style('text-anchor', 'end')
- .text(function(d) { return d.subtitle; })
- .style('opacity', 0)
- .transition()
- .duration(500)
- .delay(500)
- .style('opacity', 1);
- // Add random transition for demo
- // ------------------------------
- // Bind data
- var interval = function() {
- svg.datum(randomize).call(chart.duration(750));
- }
- // Set interval
- var intervalIds = [];
- intervalIds.push(
- setInterval(function() {
- interval()
- }, 5000)
- );
- // Enable or disable real time update
- document.getElementById('realtime').onchange = function() {
- if(realtime.checked) {
- intervalIds.push(setInterval(function() { interval() }, 5000));
- }
- else {
- for (var i=0; i < intervalIds.length; i++) {
- clearInterval(intervalIds[i]);
- }
- }
- };
- // Resize chart
- // ------------------------------
- // Call function on window resize
- $(window).on('resize', bulletResize);
- // Call function on sidebar width change
- $(document).on('click', '.sidebar-control', bulletResize);
- // Resize function
- //
- // Since D3 doesn't support SVG resize by default,
- // we need to manually specify parts of the graph that need to
- // be updated on window resize
- function bulletResize() {
- // Layout variables
- width = d3Container.node().getBoundingClientRect().width - margin.left - margin.right;
- // Layout
- // -------------------------
- // Main svg width
- container.attr('width', width + margin.left + margin.right);
- // Width of appended group
- svg.attr('width', width + margin.left + margin.right);
- // Chart elements
- // -------------------------
- // Subtitle
- svg.selectAll('.bullet-subtitle').attr('x', width);
- }
- });
- // Randomizers
- // ------------------------------
- function randomize(d) {
- if (!d.randomizer) d.randomizer = randomizer(d);
- d.ranges = d.ranges.map(d.randomizer);
- d.markers = d.markers.map(d.randomizer);
- d.measures = d.measures.map(d.randomizer);
- return d;
- }
- function randomizer(d) {
- var k = d3.max(d.ranges) * .2;
- return function(d) {
- return Math.max(0, d + k * (Math.random() - .5));
- };
- }
- }
- };
- //
- // Return objects assigned to module
- //
- return {
- initComponents: function() {
- _componentSwitchery();
- _componentDaterange();
- _componentIconLetter();
- },
- initCharts: function() {
- // Sparklines
- _chartSparkline('#new-visitors', 'line', 30, 35, 'basis', 750, 2000, '#26A69A');
- _chartSparkline('#new-sessions', 'line', 30, 35, 'basis', 750, 2000, '#FF7043');
- _chartSparkline('#total-online', 'line', 30, 35, 'basis', 750, 2000, '#5C6BC0');
- _chartSparkline('#server-load', 'area', 30, 50, 'basis', 750, 2000, 'rgba(255,255,255,0.5)');
- // Streamgraph
- _TrafficSourcesStreamChart('#traffic-sources', 330);
- // Line charts
- _AppSalesLinesChart('#app_sales', 255);
- _DailyRevenueLineChart('#today-revenue', 50);
- // Area charts
- _MonthlySalesAreaChart('#monthly-sales-stats', 100, '#4DB6AC');
- _MessagesAreaChart('#messages-stats', 40, '#5C6BC0');
- // Progress charts
- _ProgressPieChart('#today-progress', 20, 20, '#7986CB');
- _ProgressPieChart('#yesterday-progress', 20, 20, '#7986CB');
- _RoundedProgressChart('#hours-available-progress', 38, 2, '#F06292', 0.68, 'icon-watch text-pink-400', 'Hours available', '64% average');
- _RoundedProgressChart('#goal-progress', 38, 2, '#5C6BC0', 0.82, 'icon-trophy3 text-indigo-400', 'Productivity goal', '87% average');
- // Donut charts
- _MarketingCampaignsDonutChart('#campaigns-donut', 42);
- _CampaignStatusDonutChart('#campaign-status-pie', 42);
- _TicketStatusDonutChart('#tickets-status', 42);
- // Bar charts
- _BarChart('#hours-available-bars', 24, 40, true, 'elastic', 1200, 50, '#EC407A', 'hours');
- _BarChart('#goal-bars', 24, 40, true, 'elastic', 1200, 50, '#5C6BC0', 'goal');
- _BarChart('#members-online', 24, 50, true, 'elastic', 1200, 50, 'rgba(255,255,255,0.5)', 'members');
- // Heatmap
- _AppSalesHeatmap('#sales-heatmap');
- // Bullet charts
- _BulletChart("#bullets", 80);
- }
- }
- }();
- // Initialize module
- // ------------------------------
- document.addEventListener('DOMContentLoaded', function() {
- Dashboard.initComponents();
- Dashboard.initCharts();
- });
|