| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447744874497450745174527453745474557456745774587459746074617462746374647465746674677468746974707471747274737474747574767477747874797480748174827483748474857486748774887489749074917492749374947495749674977498749975007501750275037504750575067507750875097510751175127513751475157516751775187519752075217522752375247525752675277528752975307531753275337534753575367537753875397540754175427543754475457546754775487549755075517552755375547555755675577558755975607561756275637564756575667567756875697570757175727573757475757576757775787579758075817582758375847585758675877588758975907591759275937594759575967597759875997600760176027603760476057606760776087609761076117612761376147615761676177618761976207621762276237624762576267627762876297630763176327633763476357636763776387639764076417642764376447645764676477648764976507651765276537654765576567657765876597660766176627663766476657666766776687669767076717672767376747675767676777678767976807681768276837684768576867687768876897690769176927693769476957696769776987699770077017702770377047705770677077708770977107711771277137714771577167717771877197720772177227723772477257726772777287729773077317732773377347735773677377738773977407741774277437744774577467747774877497750775177527753775477557756775777587759776077617762776377647765776677677768776977707771777277737774777577767777777877797780778177827783778477857786778777887789779077917792779377947795779677977798779978007801780278037804780578067807780878097810781178127813781478157816781778187819782078217822782378247825782678277828782978307831783278337834783578367837783878397840784178427843784478457846784778487849785078517852785378547855785678577858785978607861786278637864786578667867786878697870787178727873787478757876787778787879788078817882788378847885788678877888788978907891789278937894789578967897789878997900790179027903790479057906790779087909791079117912791379147915791679177918791979207921792279237924792579267927792879297930793179327933793479357936793779387939794079417942794379447945794679477948794979507951795279537954795579567957795879597960796179627963796479657966796779687969797079717972797379747975797679777978797979807981798279837984798579867987798879897990799179927993799479957996799779987999800080018002800380048005800680078008800980108011801280138014801580168017801880198020802180228023802480258026802780288029803080318032803380348035803680378038803980408041804280438044804580468047804880498050805180528053805480558056805780588059806080618062806380648065806680678068806980708071807280738074807580768077807880798080808180828083808480858086808780888089809080918092809380948095809680978098809981008101810281038104810581068107810881098110811181128113811481158116811781188119812081218122812381248125812681278128812981308131813281338134813581368137813881398140814181428143814481458146814781488149815081518152815381548155815681578158815981608161816281638164816581668167816881698170817181728173817481758176817781788179818081818182818381848185818681878188818981908191819281938194819581968197819881998200820182028203820482058206820782088209821082118212821382148215821682178218821982208221822282238224822582268227822882298230823182328233823482358236823782388239824082418242824382448245824682478248824982508251825282538254825582568257825882598260826182628263826482658266826782688269827082718272827382748275827682778278827982808281828282838284828582868287828882898290829182928293829482958296829782988299830083018302830383048305830683078308830983108311831283138314831583168317831883198320832183228323832483258326832783288329833083318332833383348335833683378338833983408341834283438344834583468347834883498350835183528353835483558356835783588359836083618362836383648365836683678368836983708371837283738374837583768377837883798380838183828383838483858386838783888389839083918392839383948395839683978398839984008401840284038404840584068407840884098410841184128413841484158416841784188419842084218422842384248425842684278428842984308431843284338434843584368437843884398440844184428443844484458446844784488449845084518452845384548455845684578458845984608461846284638464846584668467846884698470847184728473847484758476847784788479848084818482848384848485848684878488848984908491849284938494849584968497849884998500850185028503850485058506850785088509851085118512851385148515851685178518851985208521852285238524852585268527852885298530853185328533853485358536853785388539854085418542854385448545854685478548854985508551855285538554855585568557855885598560856185628563856485658566856785688569857085718572857385748575857685778578857985808581858285838584858585868587858885898590859185928593859485958596859785988599860086018602860386048605860686078608860986108611861286138614861586168617861886198620862186228623862486258626862786288629863086318632863386348635863686378638863986408641864286438644864586468647864886498650865186528653865486558656865786588659866086618662866386648665866686678668866986708671867286738674867586768677867886798680868186828683868486858686868786888689869086918692869386948695869686978698869987008701870287038704870587068707870887098710871187128713871487158716871787188719872087218722872387248725872687278728872987308731873287338734873587368737873887398740874187428743874487458746874787488749875087518752875387548755875687578758875987608761876287638764876587668767876887698770877187728773877487758776877787788779878087818782878387848785878687878788878987908791879287938794879587968797879887998800880188028803880488058806880788088809881088118812881388148815881688178818881988208821882288238824882588268827882888298830883188328833883488358836883788388839884088418842884388448845884688478848884988508851885288538854885588568857885888598860886188628863886488658866886788688869887088718872887388748875887688778878887988808881888288838884888588868887888888898890889188928893889488958896889788988899890089018902890389048905890689078908890989108911891289138914891589168917891889198920892189228923892489258926892789288929893089318932893389348935893689378938893989408941894289438944894589468947894889498950895189528953895489558956895789588959896089618962896389648965896689678968896989708971897289738974897589768977897889798980898189828983898489858986898789888989899089918992899389948995899689978998899990009001900290039004900590069007900890099010901190129013901490159016901790189019902090219022902390249025902690279028902990309031903290339034903590369037903890399040904190429043904490459046904790489049905090519052905390549055905690579058905990609061906290639064906590669067906890699070907190729073907490759076907790789079908090819082908390849085908690879088908990909091909290939094909590969097909890999100910191029103910491059106910791089109911091119112911391149115911691179118911991209121912291239124912591269127912891299130913191329133913491359136913791389139914091419142914391449145914691479148914991509151915291539154915591569157915891599160916191629163916491659166916791689169917091719172917391749175917691779178917991809181918291839184918591869187918891899190919191929193919491959196919791989199920092019202920392049205920692079208920992109211921292139214921592169217921892199220922192229223922492259226922792289229923092319232923392349235923692379238923992409241924292439244924592469247924892499250925192529253925492559256925792589259926092619262926392649265926692679268926992709271927292739274927592769277927892799280928192829283928492859286928792889289929092919292929392949295929692979298929993009301930293039304930593069307930893099310931193129313931493159316931793189319932093219322932393249325932693279328932993309331933293339334933593369337933893399340934193429343934493459346934793489349935093519352935393549355935693579358935993609361936293639364936593669367936893699370937193729373937493759376937793789379938093819382938393849385938693879388938993909391939293939394939593969397939893999400940194029403940494059406940794089409941094119412941394149415941694179418941994209421942294239424942594269427942894299430943194329433943494359436943794389439944094419442944394449445944694479448944994509451945294539454945594569457945894599460946194629463946494659466946794689469947094719472947394749475947694779478947994809481948294839484948594869487948894899490949194929493949494959496949794989499950095019502950395049505950695079508950995109511951295139514951595169517951895199520952195229523952495259526952795289529953095319532953395349535953695379538953995409541954295439544954595469547954895499550955195529553955495559556955795589559956095619562956395649565956695679568956995709571957295739574957595769577957895799580958195829583958495859586958795889589959095919592959395949595959695979598959996009601960296039604960596069607960896099610961196129613961496159616961796189619962096219622962396249625962696279628962996309631963296339634963596369637963896399640964196429643964496459646964796489649965096519652965396549655965696579658965996609661966296639664966596669667966896699670967196729673967496759676967796789679968096819682968396849685968696879688968996909691969296939694969596969697969896999700970197029703970497059706970797089709971097119712971397149715971697179718971997209721972297239724972597269727972897299730973197329733973497359736973797389739974097419742974397449745974697479748974997509751975297539754975597569757975897599760976197629763976497659766976797689769977097719772977397749775977697779778977997809781978297839784978597869787978897899790979197929793979497959796979797989799980098019802980398049805980698079808980998109811981298139814981598169817981898199820982198229823982498259826982798289829983098319832983398349835983698379838983998409841984298439844984598469847984898499850985198529853985498559856985798589859986098619862986398649865986698679868986998709871987298739874987598769877987898799880988198829883988498859886988798889889989098919892989398949895989698979898989999009901990299039904990599069907990899099910991199129913991499159916991799189919992099219922992399249925992699279928992999309931993299339934993599369937993899399940994199429943994499459946994799489949995099519952995399549955995699579958995999609961996299639964996599669967996899699970997199729973997499759976997799789979998099819982998399849985998699879988998999909991999299939994999599969997999899991000010001100021000310004100051000610007100081000910010100111001210013100141001510016100171001810019100201002110022100231002410025100261002710028100291003010031100321003310034100351003610037100381003910040100411004210043100441004510046100471004810049100501005110052100531005410055100561005710058100591006010061100621006310064100651006610067100681006910070100711007210073100741007510076100771007810079100801008110082100831008410085100861008710088100891009010091100921009310094100951009610097100981009910100101011010210103101041010510106101071010810109101101011110112101131011410115101161011710118101191012010121101221012310124101251012610127101281012910130101311013210133101341013510136101371013810139101401014110142101431014410145101461014710148101491015010151101521015310154101551015610157101581015910160101611016210163101641016510166101671016810169101701017110172101731017410175101761017710178101791018010181101821018310184101851018610187101881018910190101911019210193101941019510196101971019810199102001020110202102031020410205102061020710208102091021010211102121021310214102151021610217102181021910220102211022210223102241022510226102271022810229102301023110232102331023410235102361023710238102391024010241102421024310244102451024610247102481024910250102511025210253102541025510256102571025810259102601026110262102631026410265102661026710268102691027010271102721027310274102751027610277102781027910280102811028210283102841028510286102871028810289102901029110292102931029410295102961029710298102991030010301103021030310304103051030610307103081030910310103111031210313103141031510316103171031810319103201032110322103231032410325103261032710328103291033010331103321033310334103351033610337103381033910340103411034210343103441034510346103471034810349103501035110352103531035410355103561035710358103591036010361103621036310364103651036610367103681036910370103711037210373103741037510376103771037810379103801038110382103831038410385103861038710388103891039010391103921039310394103951039610397103981039910400104011040210403104041040510406104071040810409104101041110412104131041410415104161041710418104191042010421104221042310424104251042610427104281042910430104311043210433104341043510436104371043810439104401044110442104431044410445104461044710448104491045010451104521045310454104551045610457104581045910460104611046210463104641046510466104671046810469104701047110472104731047410475104761047710478104791048010481104821048310484104851048610487104881048910490104911049210493104941049510496104971049810499105001050110502105031050410505105061050710508105091051010511105121051310514105151051610517105181051910520105211052210523105241052510526105271052810529105301053110532105331053410535105361053710538105391054010541105421054310544105451054610547105481054910550105511055210553105541055510556105571055810559105601056110562105631056410565105661056710568105691057010571105721057310574105751057610577105781057910580105811058210583105841058510586105871058810589105901059110592105931059410595105961059710598105991060010601106021060310604106051060610607106081060910610106111061210613106141061510616106171061810619106201062110622106231062410625106261062710628106291063010631106321063310634106351063610637106381063910640106411064210643106441064510646106471064810649106501065110652106531065410655106561065710658106591066010661106621066310664106651066610667106681066910670106711067210673106741067510676106771067810679106801068110682106831068410685106861068710688106891069010691106921069310694106951069610697106981069910700107011070210703107041070510706107071070810709107101071110712107131071410715107161071710718107191072010721107221072310724107251072610727107281072910730107311073210733107341073510736107371073810739107401074110742107431074410745107461074710748107491075010751107521075310754107551075610757107581075910760107611076210763107641076510766107671076810769107701077110772107731077410775107761077710778107791078010781107821078310784107851078610787107881078910790107911079210793107941079510796107971079810799108001080110802108031080410805108061080710808108091081010811108121081310814108151081610817108181081910820108211082210823108241082510826108271082810829108301083110832108331083410835108361083710838108391084010841108421084310844108451084610847108481084910850108511085210853108541085510856108571085810859108601086110862108631086410865108661086710868108691087010871108721087310874108751087610877108781087910880108811088210883108841088510886108871088810889108901089110892108931089410895108961089710898108991090010901109021090310904109051090610907109081090910910109111091210913109141091510916109171091810919109201092110922109231092410925109261092710928109291093010931109321093310934109351093610937109381093910940109411094210943109441094510946109471094810949109501095110952109531095410955109561095710958109591096010961109621096310964109651096610967109681096910970109711097210973109741097510976109771097810979109801098110982109831098410985109861098710988109891099010991109921099310994109951099610997109981099911000110011100211003110041100511006110071100811009110101101111012110131101411015110161101711018110191102011021110221102311024110251102611027110281102911030110311103211033110341103511036110371103811039110401104111042110431104411045110461104711048110491105011051110521105311054110551105611057110581105911060110611106211063110641106511066110671106811069110701107111072110731107411075110761107711078110791108011081110821108311084110851108611087110881108911090110911109211093110941109511096110971109811099111001110111102111031110411105111061110711108111091111011111111121111311114111151111611117111181111911120111211112211123111241112511126111271112811129111301113111132111331113411135111361113711138111391114011141111421114311144111451114611147111481114911150111511115211153111541115511156111571115811159111601116111162111631116411165111661116711168111691117011171111721117311174111751117611177111781117911180111811118211183111841118511186111871118811189111901119111192111931119411195111961119711198111991120011201112021120311204112051120611207112081120911210112111121211213112141121511216112171121811219112201122111222112231122411225112261122711228112291123011231112321123311234112351123611237112381123911240112411124211243112441124511246112471124811249112501125111252112531125411255112561125711258112591126011261112621126311264112651126611267112681126911270112711127211273112741127511276112771127811279112801128111282112831128411285112861128711288112891129011291112921129311294112951129611297112981129911300113011130211303113041130511306113071130811309113101131111312113131131411315113161131711318113191132011321113221132311324113251132611327113281132911330113311133211333113341133511336113371133811339113401134111342113431134411345113461134711348113491135011351113521135311354113551135611357113581135911360113611136211363113641136511366113671136811369113701137111372113731137411375113761137711378113791138011381113821138311384113851138611387113881138911390113911139211393113941139511396113971139811399114001140111402114031140411405114061140711408114091141011411114121141311414114151141611417114181141911420114211142211423114241142511426114271142811429114301143111432114331143411435114361143711438114391144011441114421144311444114451144611447114481144911450114511145211453114541145511456114571145811459114601146111462114631146411465114661146711468114691147011471114721147311474114751147611477114781147911480114811148211483114841148511486114871148811489114901149111492114931149411495114961149711498114991150011501115021150311504115051150611507115081150911510115111151211513115141151511516115171151811519115201152111522115231152411525115261152711528115291153011531115321153311534115351153611537115381153911540115411154211543115441154511546115471154811549115501155111552115531155411555115561155711558115591156011561115621156311564115651156611567115681156911570115711157211573115741157511576115771157811579115801158111582115831158411585115861158711588115891159011591115921159311594115951159611597115981159911600116011160211603116041160511606116071160811609116101161111612116131161411615116161161711618116191162011621116221162311624116251162611627116281162911630116311163211633116341163511636116371163811639116401164111642116431164411645116461164711648116491165011651116521165311654116551165611657116581165911660116611166211663116641166511666116671166811669116701167111672116731167411675116761167711678116791168011681116821168311684116851168611687116881168911690116911169211693116941169511696116971169811699117001170111702117031170411705117061170711708117091171011711117121171311714117151171611717117181171911720117211172211723117241172511726117271172811729117301173111732117331173411735117361173711738117391174011741117421174311744117451174611747117481174911750117511175211753117541175511756117571175811759117601176111762117631176411765117661176711768117691177011771117721177311774117751177611777117781177911780117811178211783117841178511786117871178811789117901179111792117931179411795117961179711798117991180011801118021180311804118051180611807118081180911810118111181211813118141181511816118171181811819118201182111822118231182411825118261182711828118291183011831118321183311834118351183611837118381183911840118411184211843118441184511846118471184811849118501185111852118531185411855118561185711858118591186011861118621186311864118651186611867118681186911870118711187211873118741187511876118771187811879118801188111882118831188411885118861188711888118891189011891118921189311894118951189611897118981189911900119011190211903119041190511906119071190811909119101191111912119131191411915119161191711918119191192011921119221192311924119251192611927119281192911930119311193211933119341193511936119371193811939119401194111942119431194411945119461194711948119491195011951119521195311954119551195611957119581195911960119611196211963119641196511966119671196811969119701197111972119731197411975119761197711978119791198011981119821198311984119851198611987119881198911990119911199211993119941199511996119971199811999120001200112002120031200412005120061200712008120091201012011120121201312014120151201612017120181201912020120211202212023120241202512026120271202812029120301203112032120331203412035120361203712038120391204012041120421204312044120451204612047120481204912050120511205212053120541205512056120571205812059120601206112062120631206412065120661206712068120691207012071120721207312074120751207612077120781207912080120811208212083120841208512086120871208812089120901209112092120931209412095120961209712098120991210012101121021210312104121051210612107121081210912110121111211212113121141211512116121171211812119121201212112122121231212412125121261212712128121291213012131121321213312134121351213612137121381213912140121411214212143121441214512146121471214812149121501215112152121531215412155121561215712158121591216012161121621216312164121651216612167121681216912170121711217212173121741217512176121771217812179121801218112182121831218412185121861218712188121891219012191121921219312194121951219612197121981219912200122011220212203122041220512206122071220812209122101221112212122131221412215122161221712218122191222012221122221222312224122251222612227122281222912230122311223212233122341223512236122371223812239122401224112242122431224412245122461224712248122491225012251122521225312254122551225612257122581225912260122611226212263122641226512266122671226812269122701227112272122731227412275122761227712278122791228012281122821228312284122851228612287122881228912290122911229212293122941229512296122971229812299123001230112302123031230412305123061230712308123091231012311123121231312314123151231612317123181231912320123211232212323123241232512326123271232812329123301233112332123331233412335123361233712338123391234012341123421234312344123451234612347123481234912350123511235212353123541235512356123571235812359123601236112362123631236412365123661236712368123691237012371123721237312374123751237612377123781237912380123811238212383123841238512386123871238812389123901239112392123931239412395123961239712398123991240012401124021240312404124051240612407124081240912410124111241212413124141241512416124171241812419124201242112422124231242412425124261242712428124291243012431124321243312434124351243612437124381243912440124411244212443124441244512446124471244812449124501245112452124531245412455124561245712458124591246012461124621246312464124651246612467124681246912470124711247212473124741247512476124771247812479124801248112482124831248412485124861248712488124891249012491124921249312494124951249612497124981249912500125011250212503125041250512506125071250812509125101251112512125131251412515125161251712518125191252012521125221252312524125251252612527125281252912530125311253212533125341253512536125371253812539125401254112542125431254412545125461254712548125491255012551125521255312554125551255612557125581255912560125611256212563125641256512566125671256812569125701257112572125731257412575125761257712578125791258012581125821258312584125851258612587125881258912590125911259212593125941259512596125971259812599126001260112602126031260412605126061260712608126091261012611126121261312614126151261612617126181261912620126211262212623126241262512626126271262812629126301263112632126331263412635126361263712638126391264012641126421264312644126451264612647126481264912650126511265212653126541265512656126571265812659126601266112662126631266412665126661266712668126691267012671126721267312674126751267612677126781267912680126811268212683126841268512686126871268812689126901269112692126931269412695126961269712698126991270012701127021270312704127051270612707127081270912710127111271212713127141271512716127171271812719127201272112722127231272412725127261272712728127291273012731127321273312734127351273612737127381273912740127411274212743127441274512746127471274812749127501275112752127531275412755127561275712758127591276012761127621276312764127651276612767127681276912770127711277212773127741277512776127771277812779127801278112782127831278412785127861278712788127891279012791127921279312794127951279612797127981279912800128011280212803128041280512806128071280812809128101281112812128131281412815128161281712818128191282012821128221282312824128251282612827128281282912830128311283212833128341283512836128371283812839128401284112842128431284412845128461284712848128491285012851128521285312854128551285612857128581285912860128611286212863128641286512866128671286812869128701287112872128731287412875128761287712878128791288012881128821288312884128851288612887128881288912890128911289212893128941289512896128971289812899129001290112902129031290412905129061290712908129091291012911129121291312914129151291612917129181291912920129211292212923129241292512926129271292812929129301293112932129331293412935129361293712938129391294012941129421294312944129451294612947129481294912950129511295212953129541295512956129571295812959129601296112962129631296412965129661296712968129691297012971129721297312974129751297612977129781297912980129811298212983129841298512986129871298812989129901299112992129931299412995129961299712998129991300013001130021300313004130051300613007130081300913010130111301213013130141301513016130171301813019130201302113022130231302413025130261302713028130291303013031130321303313034130351303613037130381303913040130411304213043130441304513046130471304813049130501305113052130531305413055130561305713058130591306013061130621306313064130651306613067130681306913070130711307213073130741307513076130771307813079130801308113082130831308413085130861308713088130891309013091130921309313094130951309613097130981309913100131011310213103131041310513106131071310813109131101311113112131131311413115131161311713118131191312013121131221312313124131251312613127131281312913130131311313213133131341313513136131371313813139131401314113142131431314413145131461314713148131491315013151131521315313154131551315613157131581315913160131611316213163131641316513166131671316813169131701317113172131731317413175131761317713178131791318013181131821318313184131851318613187131881318913190131911319213193131941319513196131971319813199132001320113202132031320413205132061320713208132091321013211132121321313214132151321613217132181321913220132211322213223132241322513226132271322813229132301323113232132331323413235132361323713238132391324013241132421324313244132451324613247132481324913250132511325213253132541325513256132571325813259132601326113262132631326413265132661326713268132691327013271132721327313274132751327613277132781327913280132811328213283132841328513286132871328813289132901329113292132931329413295132961329713298132991330013301133021330313304133051330613307133081330913310133111331213313133141331513316133171331813319133201332113322133231332413325133261332713328133291333013331133321333313334133351333613337133381333913340133411334213343133441334513346133471334813349133501335113352133531335413355133561335713358133591336013361133621336313364133651336613367133681336913370133711337213373133741337513376133771337813379133801338113382133831338413385133861338713388133891339013391133921339313394133951339613397133981339913400134011340213403134041340513406134071340813409134101341113412134131341413415134161341713418134191342013421134221342313424134251342613427134281342913430134311343213433134341343513436134371343813439134401344113442134431344413445134461344713448134491345013451134521345313454134551345613457134581345913460134611346213463134641346513466134671346813469134701347113472134731347413475134761347713478134791348013481134821348313484134851348613487134881348913490134911349213493134941349513496134971349813499135001350113502135031350413505135061350713508135091351013511135121351313514135151351613517135181351913520135211352213523135241352513526135271352813529135301353113532135331353413535135361353713538135391354013541135421354313544135451354613547135481354913550135511355213553135541355513556135571355813559135601356113562135631356413565135661356713568135691357013571135721357313574135751357613577135781357913580135811358213583135841358513586135871358813589135901359113592135931359413595135961359713598135991360013601136021360313604136051360613607136081360913610136111361213613136141361513616136171361813619136201362113622136231362413625136261362713628136291363013631136321363313634136351363613637136381363913640136411364213643136441364513646136471364813649136501365113652136531365413655136561365713658136591366013661136621366313664136651366613667136681366913670136711367213673136741367513676136771367813679136801368113682136831368413685136861368713688136891369013691136921369313694136951369613697136981369913700137011370213703137041370513706137071370813709137101371113712137131371413715137161371713718137191372013721137221372313724137251372613727137281372913730137311373213733137341373513736137371373813739137401374113742137431374413745137461374713748137491375013751137521375313754137551375613757137581375913760137611376213763137641376513766137671376813769137701377113772137731377413775137761377713778137791378013781137821378313784137851378613787137881378913790137911379213793137941379513796137971379813799138001380113802138031380413805138061380713808138091381013811138121381313814138151381613817138181381913820138211382213823138241382513826138271382813829138301383113832138331383413835138361383713838138391384013841138421384313844138451384613847138481384913850138511385213853138541385513856138571385813859138601386113862138631386413865138661386713868138691387013871138721387313874138751387613877138781387913880138811388213883138841388513886138871388813889138901389113892138931389413895138961389713898138991390013901139021390313904139051390613907139081390913910139111391213913139141391513916139171391813919139201392113922139231392413925139261392713928139291393013931139321393313934139351393613937139381393913940139411394213943139441394513946139471394813949139501395113952139531395413955139561395713958139591396013961139621396313964139651396613967139681396913970139711397213973139741397513976139771397813979139801398113982139831398413985139861398713988139891399013991139921399313994139951399613997139981399914000140011400214003140041400514006140071400814009140101401114012140131401414015140161401714018140191402014021140221402314024140251402614027140281402914030140311403214033140341403514036140371403814039140401404114042140431404414045140461404714048140491405014051140521405314054140551405614057140581405914060140611406214063140641406514066140671406814069140701407114072140731407414075140761407714078140791408014081140821408314084140851408614087140881408914090140911409214093140941409514096140971409814099141001410114102141031410414105141061410714108141091411014111141121411314114141151411614117141181411914120141211412214123141241412514126141271412814129141301413114132141331413414135141361413714138141391414014141141421414314144141451414614147141481414914150141511415214153141541415514156141571415814159141601416114162141631416414165141661416714168141691417014171141721417314174141751417614177141781417914180141811418214183141841418514186141871418814189141901419114192141931419414195141961419714198141991420014201142021420314204142051420614207142081420914210142111421214213142141421514216142171421814219142201422114222142231422414225142261422714228142291423014231142321423314234142351423614237142381423914240142411424214243142441424514246142471424814249142501425114252142531425414255142561425714258142591426014261142621426314264142651426614267142681426914270142711427214273142741427514276142771427814279142801428114282142831428414285142861428714288142891429014291142921429314294142951429614297142981429914300143011430214303143041430514306143071430814309143101431114312143131431414315143161431714318143191432014321143221432314324143251432614327143281432914330143311433214333143341433514336143371433814339143401434114342143431434414345143461434714348143491435014351143521435314354143551435614357143581435914360143611436214363143641436514366143671436814369143701437114372143731437414375143761437714378143791438014381143821438314384143851438614387143881438914390143911439214393143941439514396143971439814399144001440114402144031440414405144061440714408144091441014411144121441314414144151441614417144181441914420144211442214423144241442514426144271442814429144301443114432144331443414435144361443714438144391444014441144421444314444144451444614447144481444914450144511445214453144541445514456144571445814459144601446114462144631446414465144661446714468144691447014471144721447314474144751447614477144781447914480144811448214483144841448514486144871448814489144901449114492144931449414495144961449714498144991450014501145021450314504145051450614507145081450914510145111451214513145141451514516145171451814519145201452114522145231452414525145261452714528145291453014531145321453314534145351453614537145381453914540145411454214543145441454514546145471454814549145501455114552145531455414555145561455714558145591456014561145621456314564145651456614567145681456914570145711457214573145741457514576145771457814579145801458114582145831458414585145861458714588145891459014591145921459314594145951459614597145981459914600146011460214603146041460514606146071460814609146101461114612146131461414615146161461714618146191462014621146221462314624146251462614627146281462914630146311463214633146341463514636146371463814639146401464114642146431464414645146461464714648146491465014651146521465314654146551465614657146581465914660146611466214663146641466514666146671466814669146701467114672146731467414675146761467714678146791468014681146821468314684146851468614687146881468914690146911469214693146941469514696146971469814699147001470114702147031470414705147061470714708147091471014711147121471314714147151471614717147181471914720147211472214723147241472514726147271472814729147301473114732147331473414735147361473714738147391474014741147421474314744147451474614747147481474914750147511475214753147541475514756147571475814759147601476114762147631476414765147661476714768147691477014771147721477314774147751477614777147781477914780147811478214783147841478514786147871478814789147901479114792147931479414795147961479714798147991480014801148021480314804148051480614807148081480914810148111481214813148141481514816148171481814819148201482114822148231482414825148261482714828148291483014831148321483314834148351483614837148381483914840148411484214843148441484514846148471484814849148501485114852148531485414855148561485714858148591486014861148621486314864148651486614867148681486914870148711487214873148741487514876148771487814879148801488114882148831488414885148861488714888148891489014891148921489314894148951489614897148981489914900149011490214903149041490514906149071490814909149101491114912149131491414915149161491714918149191492014921149221492314924149251492614927149281492914930149311493214933149341493514936149371493814939149401494114942149431494414945149461494714948149491495014951149521495314954149551495614957149581495914960149611496214963149641496514966149671496814969149701497114972149731497414975149761497714978149791498014981149821498314984149851498614987149881498914990149911499214993149941499514996149971499814999150001500115002150031500415005150061500715008150091501015011150121501315014150151501615017150181501915020150211502215023150241502515026150271502815029150301503115032150331503415035150361503715038150391504015041150421504315044150451504615047150481504915050150511505215053150541505515056150571505815059150601506115062150631506415065150661506715068150691507015071150721507315074150751507615077150781507915080150811508215083150841508515086150871508815089150901509115092150931509415095150961509715098150991510015101151021510315104151051510615107151081510915110151111511215113151141511515116151171511815119151201512115122151231512415125151261512715128151291513015131151321513315134151351513615137151381513915140151411514215143151441514515146151471514815149151501515115152151531515415155151561515715158151591516015161151621516315164151651516615167151681516915170151711517215173151741517515176151771517815179151801518115182151831518415185151861518715188151891519015191151921519315194151951519615197151981519915200152011520215203152041520515206152071520815209152101521115212152131521415215152161521715218152191522015221152221522315224152251522615227152281522915230152311523215233152341523515236152371523815239152401524115242152431524415245152461524715248152491525015251152521525315254152551525615257152581525915260152611526215263152641526515266152671526815269152701527115272152731527415275152761527715278152791528015281152821528315284152851528615287152881528915290152911529215293152941529515296152971529815299153001530115302153031530415305153061530715308153091531015311153121531315314153151531615317153181531915320153211532215323153241532515326153271532815329153301533115332153331533415335153361533715338153391534015341153421534315344153451534615347153481534915350153511535215353153541535515356153571535815359153601536115362153631536415365153661536715368153691537015371153721537315374153751537615377153781537915380153811538215383153841538515386153871538815389153901539115392153931539415395153961539715398153991540015401154021540315404154051540615407154081540915410154111541215413154141541515416154171541815419154201542115422154231542415425154261542715428154291543015431154321543315434154351543615437154381543915440154411544215443154441544515446154471544815449154501545115452154531545415455154561545715458154591546015461154621546315464154651546615467154681546915470154711547215473154741547515476154771547815479154801548115482154831548415485154861548715488154891549015491154921549315494154951549615497154981549915500155011550215503155041550515506155071550815509155101551115512155131551415515155161551715518155191552015521155221552315524155251552615527155281552915530155311553215533155341553515536155371553815539155401554115542155431554415545155461554715548155491555015551155521555315554155551555615557155581555915560155611556215563155641556515566155671556815569155701557115572155731557415575155761557715578155791558015581155821558315584155851558615587155881558915590155911559215593155941559515596155971559815599156001560115602156031560415605156061560715608156091561015611156121561315614156151561615617156181561915620156211562215623156241562515626156271562815629156301563115632156331563415635156361563715638156391564015641156421564315644156451564615647156481564915650156511565215653156541565515656156571565815659156601566115662156631566415665156661566715668156691567015671156721567315674156751567615677156781567915680156811568215683156841568515686156871568815689156901569115692156931569415695156961569715698156991570015701157021570315704157051570615707157081570915710157111571215713157141571515716157171571815719157201572115722157231572415725157261572715728157291573015731157321573315734157351573615737157381573915740157411574215743157441574515746157471574815749157501575115752157531575415755157561575715758157591576015761157621576315764157651576615767157681576915770157711577215773157741577515776157771577815779157801578115782157831578415785157861578715788157891579015791157921579315794157951579615797157981579915800158011580215803158041580515806158071580815809158101581115812158131581415815158161581715818158191582015821158221582315824158251582615827158281582915830158311583215833158341583515836158371583815839158401584115842158431584415845158461584715848158491585015851158521585315854158551585615857158581585915860158611586215863158641586515866158671586815869158701587115872158731587415875158761587715878158791588015881158821588315884158851588615887158881588915890158911589215893158941589515896158971589815899159001590115902159031590415905159061590715908159091591015911159121591315914159151591615917159181591915920159211592215923159241592515926159271592815929159301593115932159331593415935159361593715938159391594015941159421594315944159451594615947159481594915950159511595215953159541595515956159571595815959159601596115962159631596415965159661596715968159691597015971159721597315974159751597615977159781597915980159811598215983159841598515986159871598815989159901599115992159931599415995159961599715998159991600016001160021600316004160051600616007160081600916010160111601216013160141601516016160171601816019160201602116022160231602416025160261602716028160291603016031160321603316034160351603616037160381603916040160411604216043160441604516046160471604816049160501605116052160531605416055160561605716058160591606016061160621606316064160651606616067160681606916070160711607216073160741607516076160771607816079160801608116082160831608416085160861608716088160891609016091160921609316094160951609616097160981609916100161011610216103161041610516106161071610816109161101611116112161131611416115161161611716118161191612016121161221612316124161251612616127161281612916130161311613216133161341613516136161371613816139161401614116142161431614416145161461614716148161491615016151161521615316154161551615616157161581615916160161611616216163161641616516166161671616816169161701617116172161731617416175161761617716178161791618016181161821618316184161851618616187161881618916190161911619216193161941619516196161971619816199162001620116202162031620416205162061620716208162091621016211162121621316214162151621616217162181621916220162211622216223162241622516226162271622816229162301623116232162331623416235162361623716238162391624016241162421624316244162451624616247162481624916250162511625216253162541625516256162571625816259162601626116262162631626416265162661626716268162691627016271162721627316274162751627616277162781627916280162811628216283162841628516286162871628816289162901629116292162931629416295162961629716298162991630016301163021630316304163051630616307163081630916310163111631216313163141631516316163171631816319163201632116322163231632416325163261632716328163291633016331163321633316334163351633616337163381633916340163411634216343163441634516346163471634816349163501635116352163531635416355163561635716358163591636016361163621636316364163651636616367163681636916370163711637216373163741637516376163771637816379163801638116382163831638416385163861638716388163891639016391163921639316394163951639616397163981639916400164011640216403164041640516406164071640816409164101641116412164131641416415164161641716418164191642016421164221642316424164251642616427164281642916430164311643216433164341643516436164371643816439164401644116442164431644416445164461644716448164491645016451164521645316454164551645616457164581645916460164611646216463164641646516466164671646816469164701647116472164731647416475164761647716478164791648016481164821648316484164851648616487164881648916490164911649216493164941649516496164971649816499165001650116502165031650416505165061650716508165091651016511165121651316514165151651616517165181651916520165211652216523165241652516526165271652816529165301653116532165331653416535165361653716538165391654016541165421654316544165451654616547165481654916550165511655216553165541655516556165571655816559165601656116562165631656416565165661656716568165691657016571165721657316574165751657616577165781657916580165811658216583165841658516586165871658816589165901659116592165931659416595165961659716598165991660016601166021660316604166051660616607166081660916610166111661216613166141661516616166171661816619166201662116622166231662416625166261662716628166291663016631166321663316634166351663616637166381663916640166411664216643166441664516646166471664816649166501665116652166531665416655166561665716658166591666016661166621666316664166651666616667166681666916670166711667216673166741667516676166771667816679166801668116682166831668416685166861668716688166891669016691166921669316694166951669616697166981669916700167011670216703167041670516706167071670816709167101671116712167131671416715167161671716718167191672016721167221672316724167251672616727167281672916730167311673216733167341673516736167371673816739167401674116742167431674416745167461674716748167491675016751167521675316754167551675616757167581675916760167611676216763167641676516766167671676816769167701677116772167731677416775167761677716778167791678016781167821678316784167851678616787167881678916790167911679216793167941679516796167971679816799168001680116802168031680416805168061680716808168091681016811168121681316814168151681616817168181681916820168211682216823168241682516826168271682816829168301683116832168331683416835168361683716838168391684016841168421684316844168451684616847168481684916850168511685216853168541685516856168571685816859168601686116862168631686416865168661686716868168691687016871168721687316874168751687616877168781687916880168811688216883168841688516886168871688816889168901689116892168931689416895168961689716898168991690016901169021690316904169051690616907169081690916910169111691216913169141691516916169171691816919169201692116922169231692416925169261692716928169291693016931169321693316934169351693616937169381693916940169411694216943169441694516946169471694816949169501695116952169531695416955169561695716958169591696016961169621696316964169651696616967169681696916970169711697216973169741697516976169771697816979169801698116982169831698416985169861698716988169891699016991169921699316994169951699616997169981699917000170011700217003170041700517006170071700817009170101701117012170131701417015170161701717018170191702017021170221702317024170251702617027170281702917030170311703217033170341703517036170371703817039170401704117042170431704417045170461704717048170491705017051170521705317054170551705617057170581705917060170611706217063170641706517066170671706817069170701707117072170731707417075170761707717078170791708017081170821708317084170851708617087170881708917090170911709217093170941709517096170971709817099171001710117102171031710417105171061710717108171091711017111171121711317114171151711617117171181711917120171211712217123171241712517126171271712817129171301713117132171331713417135171361713717138171391714017141171421714317144171451714617147171481714917150171511715217153171541715517156171571715817159171601716117162171631716417165171661716717168171691717017171171721717317174171751717617177171781717917180171811718217183171841718517186171871718817189171901719117192171931719417195171961719717198171991720017201172021720317204172051720617207172081720917210172111721217213172141721517216172171721817219172201722117222172231722417225172261722717228172291723017231172321723317234172351723617237172381723917240172411724217243172441724517246172471724817249172501725117252172531725417255172561725717258172591726017261172621726317264172651726617267172681726917270172711727217273172741727517276172771727817279172801728117282172831728417285172861728717288172891729017291172921729317294172951729617297172981729917300173011730217303173041730517306173071730817309173101731117312173131731417315173161731717318173191732017321173221732317324173251732617327173281732917330173311733217333173341733517336173371733817339173401734117342173431734417345173461734717348173491735017351173521735317354173551735617357173581735917360173611736217363173641736517366173671736817369173701737117372173731737417375173761737717378173791738017381173821738317384173851738617387173881738917390173911739217393173941739517396173971739817399174001740117402174031740417405174061740717408174091741017411174121741317414174151741617417174181741917420174211742217423174241742517426174271742817429174301743117432174331743417435174361743717438174391744017441174421744317444174451744617447174481744917450174511745217453174541745517456174571745817459174601746117462174631746417465174661746717468174691747017471174721747317474174751747617477174781747917480174811748217483174841748517486174871748817489174901749117492174931749417495174961749717498174991750017501175021750317504175051750617507175081750917510175111751217513175141751517516175171751817519175201752117522175231752417525175261752717528175291753017531175321753317534175351753617537175381753917540175411754217543175441754517546175471754817549175501755117552175531755417555175561755717558175591756017561175621756317564175651756617567175681756917570175711757217573175741757517576175771757817579175801758117582175831758417585175861758717588175891759017591175921759317594175951759617597175981759917600176011760217603176041760517606176071760817609176101761117612176131761417615176161761717618176191762017621176221762317624176251762617627176281762917630176311763217633176341763517636176371763817639176401764117642176431764417645176461764717648176491765017651176521765317654176551765617657176581765917660176611766217663176641766517666176671766817669176701767117672176731767417675176761767717678176791768017681176821768317684176851768617687176881768917690176911769217693176941769517696176971769817699177001770117702177031770417705177061770717708177091771017711177121771317714177151771617717177181771917720177211772217723177241772517726177271772817729177301773117732177331773417735177361773717738177391774017741177421774317744177451774617747177481774917750177511775217753177541775517756177571775817759177601776117762177631776417765177661776717768177691777017771177721777317774177751777617777177781777917780177811778217783177841778517786177871778817789177901779117792177931779417795177961779717798177991780017801178021780317804178051780617807178081780917810178111781217813178141781517816178171781817819178201782117822178231782417825178261782717828178291783017831178321783317834178351783617837178381783917840178411784217843178441784517846178471784817849178501785117852178531785417855178561785717858178591786017861178621786317864178651786617867178681786917870178711787217873178741787517876178771787817879178801788117882178831788417885178861788717888178891789017891178921789317894178951789617897178981789917900179011790217903179041790517906179071790817909179101791117912179131791417915179161791717918179191792017921179221792317924179251792617927179281792917930179311793217933179341793517936179371793817939179401794117942179431794417945179461794717948179491795017951179521795317954179551795617957179581795917960179611796217963179641796517966179671796817969179701797117972179731797417975179761797717978179791798017981179821798317984179851798617987179881798917990179911799217993179941799517996179971799817999180001800118002180031800418005180061800718008180091801018011180121801318014180151801618017180181801918020180211802218023180241802518026180271802818029180301803118032180331803418035180361803718038180391804018041180421804318044180451804618047180481804918050180511805218053180541805518056180571805818059180601806118062180631806418065180661806718068180691807018071180721807318074180751807618077180781807918080180811808218083180841808518086180871808818089180901809118092180931809418095180961809718098180991810018101181021810318104181051810618107181081810918110181111811218113181141811518116181171811818119181201812118122181231812418125181261812718128181291813018131181321813318134181351813618137181381813918140181411814218143181441814518146181471814818149181501815118152181531815418155181561815718158181591816018161181621816318164181651816618167181681816918170181711817218173181741817518176181771817818179181801818118182181831818418185181861818718188181891819018191181921819318194181951819618197181981819918200182011820218203182041820518206182071820818209182101821118212182131821418215182161821718218182191822018221182221822318224182251822618227182281822918230182311823218233182341823518236182371823818239182401824118242182431824418245182461824718248182491825018251182521825318254182551825618257182581825918260182611826218263182641826518266182671826818269182701827118272182731827418275182761827718278182791828018281182821828318284182851828618287182881828918290182911829218293182941829518296182971829818299183001830118302183031830418305183061830718308183091831018311183121831318314183151831618317183181831918320183211832218323183241832518326183271832818329183301833118332183331833418335183361833718338183391834018341183421834318344183451834618347183481834918350183511835218353183541835518356183571835818359183601836118362183631836418365183661836718368183691837018371183721837318374183751837618377183781837918380183811838218383183841838518386183871838818389183901839118392183931839418395183961839718398183991840018401184021840318404184051840618407184081840918410184111841218413184141841518416184171841818419184201842118422184231842418425184261842718428184291843018431184321843318434184351843618437184381843918440184411844218443184441844518446184471844818449184501845118452184531845418455184561845718458184591846018461184621846318464184651846618467184681846918470184711847218473184741847518476184771847818479184801848118482184831848418485184861848718488184891849018491184921849318494184951849618497184981849918500185011850218503185041850518506185071850818509185101851118512185131851418515185161851718518185191852018521185221852318524185251852618527185281852918530185311853218533185341853518536185371853818539185401854118542185431854418545185461854718548185491855018551185521855318554185551855618557185581855918560185611856218563185641856518566185671856818569185701857118572185731857418575185761857718578185791858018581185821858318584185851858618587185881858918590185911859218593185941859518596185971859818599186001860118602186031860418605186061860718608186091861018611186121861318614186151861618617186181861918620186211862218623186241862518626186271862818629186301863118632186331863418635186361863718638186391864018641186421864318644186451864618647186481864918650186511865218653186541865518656186571865818659186601866118662186631866418665186661866718668186691867018671186721867318674186751867618677186781867918680186811868218683186841868518686186871868818689186901869118692186931869418695186961869718698186991870018701187021870318704187051870618707187081870918710187111871218713187141871518716187171871818719187201872118722187231872418725187261872718728187291873018731187321873318734187351873618737187381873918740187411874218743187441874518746187471874818749187501875118752187531875418755187561875718758187591876018761187621876318764187651876618767187681876918770187711877218773187741877518776187771877818779187801878118782187831878418785187861878718788187891879018791187921879318794187951879618797187981879918800188011880218803188041880518806188071880818809188101881118812188131881418815188161881718818188191882018821188221882318824188251882618827188281882918830188311883218833188341883518836188371883818839188401884118842188431884418845188461884718848188491885018851188521885318854188551885618857188581885918860188611886218863188641886518866188671886818869188701887118872188731887418875188761887718878188791888018881188821888318884188851888618887188881888918890188911889218893188941889518896188971889818899189001890118902189031890418905189061890718908189091891018911189121891318914189151891618917189181891918920189211892218923189241892518926189271892818929189301893118932189331893418935189361893718938189391894018941189421894318944189451894618947189481894918950189511895218953189541895518956189571895818959189601896118962189631896418965189661896718968189691897018971189721897318974189751897618977189781897918980189811898218983189841898518986189871898818989189901899118992189931899418995189961899718998189991900019001190021900319004190051900619007190081900919010190111901219013190141901519016190171901819019190201902119022190231902419025190261902719028190291903019031190321903319034190351903619037190381903919040190411904219043190441904519046190471904819049190501905119052190531905419055190561905719058190591906019061190621906319064190651906619067190681906919070190711907219073190741907519076190771907819079190801908119082190831908419085190861908719088190891909019091190921909319094190951909619097190981909919100191011910219103191041910519106191071910819109191101911119112191131911419115191161911719118191191912019121191221912319124191251912619127191281912919130191311913219133191341913519136191371913819139191401914119142191431914419145191461914719148191491915019151191521915319154191551915619157191581915919160191611916219163191641916519166191671916819169191701917119172191731917419175191761917719178191791918019181191821918319184191851918619187191881918919190191911919219193191941919519196191971919819199192001920119202192031920419205192061920719208192091921019211192121921319214192151921619217192181921919220192211922219223192241922519226192271922819229192301923119232192331923419235192361923719238192391924019241192421924319244192451924619247192481924919250192511925219253192541925519256192571925819259192601926119262192631926419265192661926719268192691927019271192721927319274192751927619277192781927919280192811928219283192841928519286192871928819289192901929119292192931929419295192961929719298192991930019301193021930319304193051930619307193081930919310193111931219313193141931519316193171931819319193201932119322193231932419325193261932719328193291933019331193321933319334193351933619337193381933919340193411934219343193441934519346193471934819349193501935119352193531935419355193561935719358193591936019361193621936319364193651936619367193681936919370193711937219373193741937519376193771937819379193801938119382193831938419385193861938719388193891939019391193921939319394193951939619397193981939919400194011940219403194041940519406194071940819409194101941119412194131941419415194161941719418194191942019421194221942319424194251942619427194281942919430194311943219433194341943519436194371943819439194401944119442194431944419445194461944719448194491945019451194521945319454194551945619457194581945919460194611946219463194641946519466194671946819469194701947119472194731947419475194761947719478194791948019481194821948319484194851948619487194881948919490194911949219493194941949519496194971949819499195001950119502195031950419505195061950719508195091951019511195121951319514195151951619517195181951919520195211952219523195241952519526195271952819529195301953119532195331953419535195361953719538195391954019541195421954319544195451954619547195481954919550195511955219553195541955519556195571955819559195601956119562195631956419565195661956719568195691957019571195721957319574195751957619577195781957919580195811958219583195841958519586195871958819589195901959119592195931959419595195961959719598195991960019601196021960319604196051960619607196081960919610196111961219613196141961519616196171961819619196201962119622196231962419625196261962719628196291963019631196321963319634196351963619637196381963919640196411964219643196441964519646196471964819649196501965119652196531965419655196561965719658196591966019661196621966319664196651966619667196681966919670196711967219673196741967519676196771967819679196801968119682196831968419685196861968719688196891969019691196921969319694196951969619697196981969919700197011970219703197041970519706197071970819709197101971119712197131971419715197161971719718197191972019721197221972319724197251972619727197281972919730197311973219733197341973519736197371973819739197401974119742197431974419745197461974719748197491975019751197521975319754197551975619757197581975919760197611976219763197641976519766197671976819769197701977119772197731977419775197761977719778197791978019781197821978319784197851978619787197881978919790197911979219793197941979519796197971979819799198001980119802198031980419805198061980719808198091981019811198121981319814198151981619817198181981919820198211982219823198241982519826198271982819829198301983119832198331983419835198361983719838198391984019841198421984319844198451984619847198481984919850198511985219853198541985519856198571985819859198601986119862198631986419865198661986719868198691987019871198721987319874198751987619877198781987919880198811988219883198841988519886198871988819889198901989119892198931989419895198961989719898198991990019901199021990319904199051990619907199081990919910199111991219913199141991519916199171991819919199201992119922199231992419925199261992719928199291993019931199321993319934199351993619937199381993919940199411994219943199441994519946199471994819949199501995119952199531995419955199561995719958199591996019961199621996319964199651996619967199681996919970199711997219973199741997519976199771997819979199801998119982199831998419985199861998719988199891999019991199921999319994199951999619997199981999920000200012000220003200042000520006200072000820009200102001120012200132001420015200162001720018200192002020021200222002320024200252002620027200282002920030200312003220033200342003520036200372003820039200402004120042200432004420045200462004720048200492005020051200522005320054200552005620057200582005920060200612006220063200642006520066200672006820069200702007120072200732007420075200762007720078200792008020081200822008320084200852008620087200882008920090200912009220093200942009520096200972009820099201002010120102201032010420105201062010720108201092011020111201122011320114201152011620117201182011920120201212012220123201242012520126201272012820129201302013120132201332013420135201362013720138201392014020141201422014320144201452014620147201482014920150201512015220153201542015520156201572015820159201602016120162201632016420165201662016720168201692017020171201722017320174201752017620177201782017920180201812018220183201842018520186201872018820189201902019120192201932019420195201962019720198201992020020201202022020320204202052020620207202082020920210202112021220213202142021520216202172021820219202202022120222202232022420225202262022720228202292023020231202322023320234202352023620237202382023920240202412024220243202442024520246202472024820249202502025120252202532025420255202562025720258202592026020261202622026320264202652026620267202682026920270202712027220273202742027520276202772027820279202802028120282202832028420285202862028720288202892029020291202922029320294202952029620297202982029920300203012030220303203042030520306203072030820309203102031120312203132031420315203162031720318203192032020321203222032320324203252032620327203282032920330203312033220333203342033520336203372033820339203402034120342203432034420345203462034720348203492035020351203522035320354203552035620357203582035920360203612036220363203642036520366203672036820369203702037120372203732037420375203762037720378203792038020381203822038320384203852038620387203882038920390203912039220393203942039520396203972039820399204002040120402204032040420405204062040720408204092041020411204122041320414204152041620417204182041920420204212042220423204242042520426204272042820429204302043120432204332043420435204362043720438204392044020441204422044320444204452044620447204482044920450204512045220453204542045520456204572045820459204602046120462204632046420465204662046720468204692047020471204722047320474204752047620477204782047920480204812048220483204842048520486204872048820489204902049120492204932049420495204962049720498204992050020501205022050320504205052050620507205082050920510205112051220513205142051520516205172051820519205202052120522205232052420525205262052720528205292053020531205322053320534205352053620537205382053920540205412054220543205442054520546205472054820549205502055120552205532055420555205562055720558205592056020561205622056320564205652056620567205682056920570205712057220573205742057520576205772057820579205802058120582205832058420585205862058720588205892059020591205922059320594205952059620597205982059920600206012060220603206042060520606206072060820609206102061120612206132061420615206162061720618206192062020621206222062320624206252062620627206282062920630206312063220633206342063520636206372063820639206402064120642206432064420645206462064720648206492065020651206522065320654206552065620657206582065920660206612066220663206642066520666206672066820669206702067120672206732067420675206762067720678206792068020681206822068320684206852068620687206882068920690206912069220693206942069520696206972069820699207002070120702207032070420705207062070720708207092071020711207122071320714207152071620717207182071920720207212072220723207242072520726207272072820729207302073120732207332073420735207362073720738207392074020741207422074320744207452074620747207482074920750207512075220753207542075520756207572075820759207602076120762207632076420765207662076720768207692077020771207722077320774207752077620777207782077920780207812078220783207842078520786207872078820789207902079120792207932079420795207962079720798207992080020801208022080320804208052080620807208082080920810208112081220813208142081520816208172081820819208202082120822208232082420825208262082720828208292083020831208322083320834208352083620837208382083920840208412084220843208442084520846208472084820849208502085120852208532085420855208562085720858208592086020861208622086320864208652086620867208682086920870208712087220873208742087520876208772087820879208802088120882208832088420885208862088720888208892089020891208922089320894208952089620897208982089920900209012090220903209042090520906209072090820909209102091120912209132091420915209162091720918209192092020921209222092320924209252092620927209282092920930209312093220933209342093520936209372093820939209402094120942209432094420945209462094720948209492095020951209522095320954209552095620957209582095920960209612096220963209642096520966209672096820969209702097120972209732097420975209762097720978209792098020981209822098320984209852098620987209882098920990209912099220993209942099520996209972099820999210002100121002210032100421005210062100721008210092101021011210122101321014210152101621017210182101921020210212102221023210242102521026210272102821029210302103121032210332103421035210362103721038210392104021041210422104321044210452104621047210482104921050210512105221053210542105521056210572105821059210602106121062210632106421065210662106721068210692107021071210722107321074210752107621077210782107921080210812108221083210842108521086210872108821089210902109121092210932109421095210962109721098210992110021101211022110321104211052110621107211082110921110211112111221113211142111521116211172111821119211202112121122211232112421125211262112721128211292113021131211322113321134211352113621137211382113921140211412114221143211442114521146211472114821149211502115121152211532115421155211562115721158211592116021161211622116321164211652116621167211682116921170211712117221173211742117521176211772117821179211802118121182211832118421185211862118721188211892119021191211922119321194211952119621197211982119921200212012120221203212042120521206212072120821209212102121121212212132121421215212162121721218212192122021221212222122321224212252122621227212282122921230212312123221233212342123521236212372123821239212402124121242212432124421245212462124721248212492125021251212522125321254212552125621257212582125921260212612126221263212642126521266212672126821269212702127121272212732127421275212762127721278212792128021281212822128321284212852128621287212882128921290212912129221293212942129521296212972129821299213002130121302213032130421305213062130721308213092131021311213122131321314213152131621317213182131921320213212132221323213242132521326213272132821329213302133121332213332133421335213362133721338213392134021341213422134321344213452134621347213482134921350213512135221353213542135521356213572135821359213602136121362213632136421365213662136721368213692137021371213722137321374213752137621377213782137921380213812138221383213842138521386213872138821389213902139121392213932139421395213962139721398213992140021401214022140321404214052140621407214082140921410214112141221413214142141521416214172141821419214202142121422214232142421425214262142721428214292143021431214322143321434214352143621437214382143921440214412144221443214442144521446214472144821449214502145121452214532145421455214562145721458214592146021461214622146321464214652146621467214682146921470214712147221473214742147521476214772147821479214802148121482214832148421485214862148721488214892149021491214922149321494214952149621497214982149921500215012150221503215042150521506215072150821509215102151121512215132151421515215162151721518215192152021521215222152321524215252152621527215282152921530215312153221533215342153521536215372153821539215402154121542215432154421545215462154721548215492155021551215522155321554215552155621557215582155921560215612156221563215642156521566215672156821569215702157121572215732157421575215762157721578215792158021581215822158321584215852158621587215882158921590215912159221593215942159521596215972159821599216002160121602216032160421605216062160721608216092161021611216122161321614216152161621617216182161921620216212162221623216242162521626216272162821629216302163121632216332163421635216362163721638216392164021641216422164321644216452164621647216482164921650216512165221653216542165521656216572165821659216602166121662216632166421665216662166721668216692167021671216722167321674216752167621677216782167921680216812168221683216842168521686216872168821689216902169121692216932169421695216962169721698216992170021701217022170321704217052170621707217082170921710217112171221713217142171521716217172171821719217202172121722217232172421725217262172721728217292173021731217322173321734217352173621737217382173921740217412174221743217442174521746217472174821749217502175121752217532175421755217562175721758217592176021761217622176321764217652176621767217682176921770217712177221773217742177521776217772177821779217802178121782217832178421785217862178721788217892179021791217922179321794217952179621797217982179921800218012180221803218042180521806218072180821809218102181121812218132181421815218162181721818218192182021821218222182321824218252182621827218282182921830218312183221833218342183521836218372183821839218402184121842218432184421845218462184721848218492185021851218522185321854218552185621857218582185921860218612186221863218642186521866218672186821869218702187121872218732187421875218762187721878218792188021881218822188321884218852188621887218882188921890218912189221893218942189521896218972189821899219002190121902219032190421905219062190721908219092191021911219122191321914219152191621917219182191921920219212192221923219242192521926219272192821929219302193121932219332193421935219362193721938219392194021941219422194321944219452194621947219482194921950219512195221953219542195521956219572195821959219602196121962219632196421965219662196721968219692197021971219722197321974219752197621977219782197921980219812198221983219842198521986219872198821989219902199121992219932199421995219962199721998219992200022001220022200322004220052200622007220082200922010220112201222013220142201522016220172201822019220202202122022220232202422025220262202722028220292203022031220322203322034220352203622037220382203922040220412204222043220442204522046220472204822049220502205122052220532205422055220562205722058220592206022061220622206322064220652206622067220682206922070220712207222073220742207522076220772207822079220802208122082220832208422085220862208722088220892209022091220922209322094220952209622097220982209922100221012210222103221042210522106221072210822109221102211122112221132211422115221162211722118221192212022121221222212322124221252212622127221282212922130221312213222133221342213522136221372213822139221402214122142221432214422145221462214722148221492215022151221522215322154221552215622157221582215922160221612216222163221642216522166221672216822169221702217122172221732217422175221762217722178221792218022181221822218322184221852218622187221882218922190221912219222193221942219522196221972219822199222002220122202222032220422205222062220722208222092221022211222122221322214222152221622217222182221922220222212222222223222242222522226222272222822229222302223122232222332223422235222362223722238222392224022241222422224322244222452224622247222482224922250222512225222253222542225522256222572225822259222602226122262222632226422265222662226722268222692227022271222722227322274222752227622277222782227922280222812228222283222842228522286222872228822289222902229122292222932229422295222962229722298222992230022301223022230322304223052230622307223082230922310223112231222313223142231522316223172231822319223202232122322223232232422325223262232722328223292233022331223322233322334223352233622337223382233922340223412234222343223442234522346223472234822349223502235122352223532235422355223562235722358223592236022361223622236322364223652236622367223682236922370223712237222373223742237522376223772237822379223802238122382223832238422385223862238722388223892239022391223922239322394223952239622397223982239922400224012240222403224042240522406224072240822409224102241122412224132241422415224162241722418224192242022421224222242322424224252242622427224282242922430224312243222433224342243522436224372243822439224402244122442224432244422445224462244722448224492245022451224522245322454224552245622457224582245922460224612246222463224642246522466224672246822469224702247122472224732247422475224762247722478224792248022481224822248322484224852248622487224882248922490224912249222493224942249522496224972249822499225002250122502225032250422505225062250722508225092251022511225122251322514225152251622517225182251922520225212252222523225242252522526225272252822529225302253122532225332253422535225362253722538225392254022541225422254322544225452254622547225482254922550225512255222553225542255522556225572255822559225602256122562225632256422565225662256722568225692257022571225722257322574225752257622577225782257922580225812258222583225842258522586225872258822589225902259122592225932259422595225962259722598225992260022601226022260322604226052260622607226082260922610226112261222613226142261522616226172261822619226202262122622226232262422625226262262722628226292263022631226322263322634226352263622637226382263922640226412264222643226442264522646226472264822649226502265122652226532265422655226562265722658226592266022661226622266322664226652266622667226682266922670226712267222673226742267522676226772267822679226802268122682226832268422685226862268722688226892269022691226922269322694226952269622697226982269922700227012270222703227042270522706227072270822709227102271122712227132271422715227162271722718227192272022721227222272322724227252272622727227282272922730227312273222733227342273522736227372273822739227402274122742227432274422745227462274722748227492275022751227522275322754227552275622757227582275922760227612276222763227642276522766227672276822769227702277122772227732277422775227762277722778227792278022781227822278322784227852278622787227882278922790227912279222793227942279522796227972279822799228002280122802228032280422805228062280722808228092281022811228122281322814228152281622817228182281922820228212282222823228242282522826228272282822829228302283122832228332283422835228362283722838228392284022841228422284322844228452284622847228482284922850228512285222853228542285522856228572285822859228602286122862228632286422865228662286722868228692287022871228722287322874228752287622877228782287922880228812288222883228842288522886228872288822889228902289122892228932289422895228962289722898228992290022901229022290322904229052290622907229082290922910229112291222913229142291522916229172291822919229202292122922229232292422925229262292722928229292293022931229322293322934229352293622937229382293922940229412294222943229442294522946229472294822949229502295122952229532295422955229562295722958229592296022961229622296322964229652296622967229682296922970229712297222973229742297522976229772297822979229802298122982229832298422985229862298722988229892299022991229922299322994229952299622997229982299923000230012300223003230042300523006230072300823009230102301123012230132301423015230162301723018230192302023021230222302323024230252302623027230282302923030230312303223033230342303523036230372303823039230402304123042230432304423045230462304723048230492305023051230522305323054230552305623057230582305923060230612306223063230642306523066230672306823069230702307123072230732307423075230762307723078230792308023081230822308323084230852308623087230882308923090230912309223093230942309523096230972309823099231002310123102231032310423105231062310723108231092311023111231122311323114231152311623117231182311923120231212312223123231242312523126231272312823129231302313123132231332313423135231362313723138231392314023141231422314323144231452314623147231482314923150231512315223153231542315523156231572315823159231602316123162231632316423165231662316723168231692317023171231722317323174231752317623177231782317923180231812318223183231842318523186231872318823189231902319123192231932319423195231962319723198231992320023201232022320323204232052320623207232082320923210232112321223213232142321523216232172321823219232202322123222232232322423225232262322723228232292323023231232322323323234232352323623237232382323923240232412324223243232442324523246232472324823249232502325123252232532325423255232562325723258232592326023261232622326323264232652326623267232682326923270232712327223273232742327523276232772327823279232802328123282232832328423285232862328723288232892329023291232922329323294232952329623297232982329923300233012330223303233042330523306233072330823309233102331123312233132331423315233162331723318233192332023321233222332323324233252332623327233282332923330233312333223333233342333523336233372333823339233402334123342233432334423345233462334723348233492335023351233522335323354233552335623357233582335923360233612336223363233642336523366233672336823369233702337123372233732337423375233762337723378233792338023381233822338323384233852338623387233882338923390233912339223393233942339523396233972339823399234002340123402234032340423405234062340723408234092341023411234122341323414234152341623417234182341923420234212342223423234242342523426234272342823429234302343123432234332343423435234362343723438234392344023441234422344323444234452344623447234482344923450234512345223453234542345523456234572345823459234602346123462234632346423465234662346723468234692347023471234722347323474234752347623477234782347923480234812348223483234842348523486234872348823489234902349123492234932349423495234962349723498234992350023501235022350323504235052350623507235082350923510235112351223513235142351523516235172351823519235202352123522235232352423525235262352723528235292353023531235322353323534235352353623537235382353923540235412354223543235442354523546235472354823549235502355123552235532355423555235562355723558235592356023561235622356323564235652356623567235682356923570235712357223573235742357523576235772357823579235802358123582235832358423585235862358723588235892359023591235922359323594235952359623597235982359923600236012360223603236042360523606236072360823609236102361123612236132361423615236162361723618236192362023621236222362323624236252362623627236282362923630236312363223633236342363523636236372363823639236402364123642236432364423645236462364723648236492365023651236522365323654236552365623657236582365923660236612366223663236642366523666236672366823669236702367123672236732367423675236762367723678236792368023681236822368323684236852368623687236882368923690236912369223693236942369523696236972369823699237002370123702237032370423705237062370723708237092371023711237122371323714237152371623717237182371923720237212372223723237242372523726237272372823729237302373123732237332373423735237362373723738237392374023741237422374323744237452374623747237482374923750237512375223753237542375523756237572375823759237602376123762237632376423765237662376723768237692377023771237722377323774237752377623777237782377923780237812378223783237842378523786237872378823789237902379123792237932379423795237962379723798237992380023801238022380323804238052380623807238082380923810238112381223813238142381523816238172381823819238202382123822238232382423825238262382723828238292383023831238322383323834238352383623837238382383923840238412384223843238442384523846238472384823849238502385123852238532385423855238562385723858238592386023861238622386323864238652386623867238682386923870238712387223873238742387523876238772387823879238802388123882238832388423885238862388723888238892389023891238922389323894238952389623897238982389923900239012390223903239042390523906239072390823909239102391123912239132391423915239162391723918239192392023921239222392323924239252392623927239282392923930239312393223933239342393523936239372393823939239402394123942239432394423945239462394723948239492395023951239522395323954239552395623957239582395923960239612396223963239642396523966239672396823969239702397123972239732397423975239762397723978239792398023981239822398323984239852398623987239882398923990239912399223993239942399523996239972399823999240002400124002240032400424005240062400724008240092401024011240122401324014240152401624017240182401924020240212402224023240242402524026240272402824029240302403124032240332403424035240362403724038240392404024041240422404324044240452404624047240482404924050240512405224053240542405524056240572405824059240602406124062240632406424065240662406724068240692407024071240722407324074240752407624077240782407924080240812408224083240842408524086240872408824089240902409124092240932409424095240962409724098240992410024101241022410324104241052410624107241082410924110241112411224113241142411524116241172411824119241202412124122241232412424125241262412724128241292413024131241322413324134241352413624137241382413924140241412414224143241442414524146241472414824149241502415124152241532415424155241562415724158241592416024161241622416324164241652416624167241682416924170241712417224173241742417524176241772417824179241802418124182241832418424185241862418724188241892419024191241922419324194241952419624197241982419924200242012420224203242042420524206242072420824209242102421124212242132421424215242162421724218242192422024221242222422324224242252422624227242282422924230242312423224233242342423524236242372423824239242402424124242242432424424245242462424724248242492425024251242522425324254242552425624257242582425924260242612426224263242642426524266242672426824269242702427124272242732427424275242762427724278242792428024281242822428324284242852428624287242882428924290242912429224293242942429524296242972429824299243002430124302243032430424305243062430724308243092431024311243122431324314243152431624317243182431924320243212432224323243242432524326243272432824329243302433124332243332433424335243362433724338243392434024341243422434324344243452434624347243482434924350243512435224353243542435524356243572435824359243602436124362243632436424365243662436724368243692437024371243722437324374243752437624377243782437924380243812438224383243842438524386243872438824389243902439124392243932439424395243962439724398243992440024401244022440324404244052440624407244082440924410244112441224413244142441524416244172441824419244202442124422244232442424425244262442724428244292443024431244322443324434244352443624437244382443924440244412444224443244442444524446244472444824449244502445124452244532445424455244562445724458244592446024461244622446324464244652446624467244682446924470244712447224473244742447524476244772447824479244802448124482244832448424485244862448724488244892449024491244922449324494244952449624497244982449924500245012450224503245042450524506245072450824509245102451124512245132451424515245162451724518245192452024521245222452324524245252452624527245282452924530245312453224533245342453524536245372453824539245402454124542245432454424545245462454724548245492455024551245522455324554245552455624557245582455924560245612456224563245642456524566245672456824569245702457124572245732457424575245762457724578245792458024581245822458324584245852458624587245882458924590245912459224593245942459524596245972459824599246002460124602246032460424605246062460724608246092461024611246122461324614246152461624617246182461924620246212462224623246242462524626246272462824629246302463124632246332463424635246362463724638246392464024641246422464324644246452464624647246482464924650246512465224653246542465524656246572465824659246602466124662246632466424665246662466724668246692467024671246722467324674246752467624677246782467924680246812468224683246842468524686246872468824689246902469124692246932469424695246962469724698246992470024701247022470324704247052470624707247082470924710247112471224713247142471524716247172471824719247202472124722247232472424725247262472724728247292473024731247322473324734247352473624737247382473924740247412474224743247442474524746247472474824749247502475124752247532475424755247562475724758247592476024761247622476324764247652476624767247682476924770247712477224773247742477524776247772477824779247802478124782247832478424785247862478724788247892479024791247922479324794247952479624797247982479924800248012480224803248042480524806248072480824809248102481124812248132481424815248162481724818248192482024821248222482324824248252482624827248282482924830248312483224833248342483524836248372483824839248402484124842248432484424845248462484724848248492485024851248522485324854248552485624857248582485924860248612486224863248642486524866248672486824869248702487124872248732487424875248762487724878248792488024881248822488324884248852488624887248882488924890248912489224893248942489524896248972489824899249002490124902249032490424905249062490724908249092491024911249122491324914249152491624917249182491924920249212492224923249242492524926249272492824929249302493124932249332493424935249362493724938249392494024941249422494324944249452494624947249482494924950249512495224953249542495524956249572495824959249602496124962249632496424965249662496724968249692497024971249722497324974249752497624977249782497924980249812498224983249842498524986249872498824989249902499124992249932499424995249962499724998249992500025001250022500325004250052500625007250082500925010250112501225013250142501525016250172501825019250202502125022250232502425025250262502725028250292503025031250322503325034250352503625037250382503925040250412504225043250442504525046250472504825049250502505125052250532505425055250562505725058250592506025061250622506325064250652506625067250682506925070250712507225073250742507525076250772507825079250802508125082250832508425085250862508725088250892509025091250922509325094250952509625097250982509925100251012510225103251042510525106251072510825109251102511125112251132511425115251162511725118251192512025121251222512325124251252512625127251282512925130251312513225133251342513525136251372513825139251402514125142251432514425145251462514725148251492515025151251522515325154251552515625157251582515925160251612516225163251642516525166251672516825169251702517125172251732517425175251762517725178251792518025181251822518325184251852518625187251882518925190251912519225193251942519525196251972519825199252002520125202252032520425205252062520725208252092521025211252122521325214252152521625217252182521925220252212522225223252242522525226252272522825229252302523125232252332523425235252362523725238252392524025241252422524325244252452524625247252482524925250252512525225253252542525525256252572525825259252602526125262252632526425265252662526725268252692527025271252722527325274252752527625277252782527925280252812528225283252842528525286252872528825289252902529125292252932529425295252962529725298252992530025301253022530325304253052530625307253082530925310253112531225313253142531525316253172531825319253202532125322253232532425325253262532725328253292533025331253322533325334253352533625337253382533925340253412534225343253442534525346253472534825349253502535125352253532535425355253562535725358253592536025361253622536325364253652536625367253682536925370253712537225373253742537525376253772537825379253802538125382253832538425385253862538725388253892539025391253922539325394253952539625397253982539925400254012540225403254042540525406254072540825409254102541125412254132541425415254162541725418254192542025421254222542325424254252542625427254282542925430254312543225433254342543525436254372543825439254402544125442254432544425445254462544725448254492545025451254522545325454254552545625457254582545925460254612546225463254642546525466254672546825469254702547125472254732547425475254762547725478254792548025481254822548325484254852548625487254882548925490254912549225493254942549525496254972549825499255002550125502255032550425505255062550725508255092551025511255122551325514255152551625517255182551925520255212552225523255242552525526255272552825529255302553125532255332553425535255362553725538255392554025541255422554325544255452554625547255482554925550255512555225553255542555525556255572555825559255602556125562255632556425565255662556725568255692557025571255722557325574255752557625577255782557925580255812558225583255842558525586255872558825589255902559125592255932559425595255962559725598255992560025601256022560325604256052560625607256082560925610256112561225613256142561525616256172561825619256202562125622256232562425625256262562725628256292563025631256322563325634256352563625637256382563925640256412564225643256442564525646256472564825649256502565125652256532565425655256562565725658256592566025661256622566325664256652566625667256682566925670256712567225673256742567525676256772567825679256802568125682256832568425685256862568725688256892569025691256922569325694256952569625697256982569925700257012570225703257042570525706257072570825709257102571125712257132571425715257162571725718257192572025721257222572325724257252572625727257282572925730257312573225733257342573525736257372573825739257402574125742257432574425745257462574725748257492575025751257522575325754257552575625757257582575925760257612576225763257642576525766257672576825769257702577125772257732577425775257762577725778257792578025781257822578325784257852578625787257882578925790257912579225793257942579525796257972579825799258002580125802258032580425805258062580725808258092581025811258122581325814258152581625817258182581925820258212582225823258242582525826258272582825829258302583125832258332583425835258362583725838258392584025841258422584325844258452584625847258482584925850258512585225853258542585525856258572585825859258602586125862258632586425865258662586725868258692587025871258722587325874258752587625877258782587925880258812588225883258842588525886258872588825889258902589125892258932589425895258962589725898258992590025901259022590325904259052590625907259082590925910259112591225913259142591525916259172591825919259202592125922259232592425925259262592725928259292593025931259322593325934259352593625937259382593925940259412594225943259442594525946259472594825949259502595125952259532595425955259562595725958259592596025961259622596325964259652596625967259682596925970259712597225973259742597525976259772597825979259802598125982259832598425985259862598725988259892599025991259922599325994259952599625997259982599926000260012600226003260042600526006260072600826009260102601126012260132601426015260162601726018260192602026021260222602326024260252602626027260282602926030260312603226033260342603526036260372603826039260402604126042260432604426045260462604726048260492605026051260522605326054260552605626057260582605926060260612606226063260642606526066260672606826069260702607126072260732607426075260762607726078260792608026081260822608326084260852608626087260882608926090260912609226093260942609526096260972609826099261002610126102261032610426105261062610726108261092611026111261122611326114261152611626117261182611926120261212612226123261242612526126261272612826129261302613126132261332613426135261362613726138261392614026141261422614326144261452614626147261482614926150261512615226153261542615526156261572615826159261602616126162261632616426165261662616726168261692617026171261722617326174261752617626177261782617926180261812618226183261842618526186261872618826189261902619126192261932619426195261962619726198261992620026201262022620326204262052620626207262082620926210262112621226213262142621526216262172621826219262202622126222262232622426225262262622726228262292623026231262322623326234262352623626237262382623926240262412624226243262442624526246262472624826249262502625126252262532625426255262562625726258262592626026261262622626326264262652626626267262682626926270262712627226273262742627526276262772627826279262802628126282262832628426285262862628726288262892629026291262922629326294262952629626297262982629926300263012630226303263042630526306263072630826309263102631126312263132631426315263162631726318263192632026321263222632326324263252632626327263282632926330263312633226333263342633526336263372633826339263402634126342263432634426345263462634726348263492635026351263522635326354263552635626357263582635926360263612636226363263642636526366263672636826369263702637126372263732637426375263762637726378263792638026381263822638326384263852638626387263882638926390263912639226393263942639526396263972639826399264002640126402264032640426405264062640726408264092641026411264122641326414264152641626417264182641926420264212642226423264242642526426264272642826429264302643126432264332643426435264362643726438264392644026441264422644326444264452644626447264482644926450264512645226453264542645526456264572645826459264602646126462264632646426465264662646726468264692647026471264722647326474264752647626477264782647926480264812648226483264842648526486264872648826489264902649126492264932649426495264962649726498264992650026501265022650326504265052650626507265082650926510265112651226513265142651526516265172651826519265202652126522265232652426525265262652726528265292653026531265322653326534265352653626537265382653926540265412654226543265442654526546265472654826549265502655126552265532655426555265562655726558265592656026561265622656326564265652656626567265682656926570265712657226573265742657526576265772657826579265802658126582265832658426585265862658726588265892659026591265922659326594265952659626597265982659926600266012660226603266042660526606266072660826609266102661126612266132661426615266162661726618266192662026621266222662326624266252662626627266282662926630266312663226633266342663526636266372663826639266402664126642266432664426645266462664726648266492665026651266522665326654266552665626657266582665926660266612666226663266642666526666266672666826669266702667126672266732667426675266762667726678266792668026681266822668326684266852668626687266882668926690266912669226693266942669526696266972669826699267002670126702267032670426705267062670726708267092671026711267122671326714267152671626717267182671926720267212672226723267242672526726267272672826729267302673126732267332673426735267362673726738267392674026741267422674326744267452674626747267482674926750267512675226753267542675526756267572675826759267602676126762267632676426765267662676726768267692677026771267722677326774267752677626777267782677926780267812678226783267842678526786267872678826789267902679126792267932679426795267962679726798267992680026801268022680326804268052680626807268082680926810268112681226813268142681526816268172681826819268202682126822268232682426825268262682726828268292683026831268322683326834268352683626837268382683926840268412684226843268442684526846268472684826849268502685126852268532685426855268562685726858268592686026861268622686326864268652686626867268682686926870268712687226873268742687526876268772687826879268802688126882268832688426885268862688726888268892689026891268922689326894268952689626897268982689926900269012690226903269042690526906269072690826909269102691126912269132691426915269162691726918269192692026921269222692326924269252692626927269282692926930269312693226933269342693526936269372693826939269402694126942269432694426945269462694726948269492695026951269522695326954269552695626957269582695926960269612696226963269642696526966269672696826969269702697126972269732697426975269762697726978269792698026981269822698326984269852698626987269882698926990269912699226993269942699526996269972699826999270002700127002270032700427005270062700727008270092701027011270122701327014270152701627017270182701927020270212702227023270242702527026270272702827029270302703127032270332703427035270362703727038270392704027041270422704327044270452704627047270482704927050270512705227053270542705527056270572705827059270602706127062270632706427065270662706727068270692707027071270722707327074270752707627077270782707927080270812708227083270842708527086270872708827089270902709127092270932709427095270962709727098270992710027101271022710327104271052710627107271082710927110271112711227113271142711527116271172711827119271202712127122271232712427125271262712727128271292713027131271322713327134271352713627137271382713927140271412714227143271442714527146271472714827149271502715127152271532715427155271562715727158271592716027161271622716327164271652716627167271682716927170271712717227173271742717527176271772717827179271802718127182271832718427185271862718727188271892719027191271922719327194271952719627197271982719927200272012720227203272042720527206272072720827209272102721127212272132721427215272162721727218272192722027221272222722327224272252722627227272282722927230272312723227233272342723527236272372723827239272402724127242272432724427245272462724727248272492725027251272522725327254272552725627257272582725927260272612726227263272642726527266272672726827269272702727127272272732727427275272762727727278272792728027281272822728327284272852728627287272882728927290272912729227293272942729527296272972729827299273002730127302273032730427305273062730727308273092731027311273122731327314273152731627317273182731927320273212732227323273242732527326273272732827329273302733127332273332733427335273362733727338273392734027341273422734327344273452734627347273482734927350273512735227353273542735527356273572735827359273602736127362273632736427365273662736727368273692737027371273722737327374273752737627377273782737927380273812738227383273842738527386273872738827389273902739127392273932739427395273962739727398273992740027401274022740327404274052740627407274082740927410274112741227413274142741527416274172741827419274202742127422274232742427425274262742727428274292743027431274322743327434274352743627437274382743927440274412744227443274442744527446274472744827449274502745127452274532745427455274562745727458274592746027461274622746327464274652746627467274682746927470274712747227473274742747527476274772747827479274802748127482274832748427485274862748727488274892749027491274922749327494274952749627497274982749927500275012750227503275042750527506275072750827509275102751127512275132751427515275162751727518275192752027521275222752327524275252752627527275282752927530275312753227533275342753527536275372753827539275402754127542275432754427545275462754727548275492755027551275522755327554275552755627557275582755927560275612756227563275642756527566275672756827569275702757127572275732757427575275762757727578275792758027581275822758327584275852758627587275882758927590275912759227593275942759527596275972759827599276002760127602276032760427605276062760727608276092761027611276122761327614276152761627617276182761927620276212762227623276242762527626276272762827629276302763127632276332763427635276362763727638276392764027641276422764327644276452764627647276482764927650276512765227653276542765527656276572765827659276602766127662276632766427665276662766727668276692767027671276722767327674276752767627677276782767927680276812768227683276842768527686276872768827689276902769127692276932769427695276962769727698276992770027701277022770327704277052770627707277082770927710277112771227713277142771527716277172771827719277202772127722277232772427725277262772727728277292773027731277322773327734277352773627737277382773927740277412774227743277442774527746277472774827749277502775127752277532775427755277562775727758277592776027761277622776327764277652776627767277682776927770277712777227773277742777527776277772777827779277802778127782277832778427785277862778727788277892779027791277922779327794277952779627797277982779927800278012780227803278042780527806278072780827809278102781127812278132781427815278162781727818278192782027821278222782327824278252782627827278282782927830278312783227833278342783527836278372783827839278402784127842278432784427845278462784727848278492785027851278522785327854278552785627857278582785927860278612786227863278642786527866278672786827869278702787127872278732787427875278762787727878278792788027881278822788327884278852788627887278882788927890278912789227893278942789527896278972789827899279002790127902279032790427905279062790727908279092791027911279122791327914279152791627917279182791927920279212792227923279242792527926279272792827929279302793127932279332793427935279362793727938279392794027941279422794327944279452794627947279482794927950279512795227953279542795527956279572795827959279602796127962279632796427965279662796727968279692797027971279722797327974279752797627977279782797927980279812798227983279842798527986279872798827989279902799127992279932799427995279962799727998279992800028001280022800328004280052800628007280082800928010280112801228013280142801528016280172801828019280202802128022280232802428025280262802728028280292803028031280322803328034280352803628037280382803928040280412804228043280442804528046280472804828049280502805128052280532805428055280562805728058280592806028061280622806328064280652806628067280682806928070280712807228073280742807528076280772807828079280802808128082280832808428085280862808728088280892809028091280922809328094280952809628097280982809928100281012810228103281042810528106281072810828109281102811128112281132811428115281162811728118281192812028121281222812328124281252812628127281282812928130281312813228133281342813528136281372813828139281402814128142281432814428145281462814728148281492815028151281522815328154281552815628157281582815928160281612816228163281642816528166281672816828169281702817128172281732817428175281762817728178281792818028181281822818328184281852818628187281882818928190281912819228193281942819528196281972819828199282002820128202282032820428205282062820728208282092821028211282122821328214282152821628217282182821928220282212822228223282242822528226282272822828229282302823128232282332823428235282362823728238282392824028241282422824328244282452824628247282482824928250282512825228253282542825528256282572825828259282602826128262282632826428265282662826728268282692827028271282722827328274282752827628277282782827928280282812828228283282842828528286282872828828289282902829128292282932829428295282962829728298282992830028301283022830328304283052830628307283082830928310283112831228313283142831528316283172831828319283202832128322283232832428325283262832728328283292833028331283322833328334283352833628337283382833928340283412834228343283442834528346283472834828349283502835128352283532835428355283562835728358283592836028361283622836328364283652836628367283682836928370283712837228373283742837528376283772837828379283802838128382283832838428385283862838728388283892839028391283922839328394283952839628397283982839928400284012840228403284042840528406284072840828409284102841128412284132841428415284162841728418284192842028421284222842328424284252842628427284282842928430284312843228433284342843528436284372843828439284402844128442284432844428445284462844728448284492845028451284522845328454284552845628457284582845928460284612846228463284642846528466284672846828469284702847128472284732847428475284762847728478284792848028481284822848328484284852848628487284882848928490284912849228493284942849528496284972849828499285002850128502285032850428505285062850728508285092851028511285122851328514285152851628517285182851928520285212852228523285242852528526285272852828529285302853128532285332853428535285362853728538285392854028541285422854328544285452854628547285482854928550285512855228553285542855528556285572855828559285602856128562285632856428565285662856728568285692857028571285722857328574285752857628577285782857928580285812858228583285842858528586285872858828589285902859128592285932859428595285962859728598285992860028601286022860328604286052860628607286082860928610286112861228613286142861528616286172861828619286202862128622286232862428625286262862728628286292863028631286322863328634286352863628637286382863928640286412864228643286442864528646286472864828649286502865128652286532865428655286562865728658286592866028661286622866328664286652866628667286682866928670286712867228673286742867528676286772867828679286802868128682286832868428685286862868728688286892869028691286922869328694286952869628697286982869928700287012870228703287042870528706287072870828709287102871128712287132871428715287162871728718287192872028721287222872328724287252872628727287282872928730287312873228733287342873528736287372873828739287402874128742287432874428745287462874728748287492875028751287522875328754287552875628757287582875928760287612876228763287642876528766287672876828769287702877128772287732877428775287762877728778287792878028781287822878328784287852878628787287882878928790287912879228793287942879528796287972879828799288002880128802288032880428805288062880728808288092881028811288122881328814288152881628817288182881928820288212882228823288242882528826288272882828829288302883128832288332883428835288362883728838288392884028841288422884328844288452884628847288482884928850288512885228853288542885528856288572885828859288602886128862288632886428865288662886728868288692887028871288722887328874288752887628877288782887928880288812888228883288842888528886288872888828889288902889128892288932889428895288962889728898288992890028901289022890328904289052890628907289082890928910289112891228913289142891528916289172891828919289202892128922289232892428925289262892728928289292893028931289322893328934289352893628937289382893928940289412894228943289442894528946289472894828949289502895128952289532895428955289562895728958289592896028961289622896328964289652896628967289682896928970289712897228973289742897528976289772897828979289802898128982289832898428985289862898728988289892899028991289922899328994289952899628997289982899929000290012900229003290042900529006290072900829009290102901129012290132901429015290162901729018290192902029021290222902329024290252902629027290282902929030290312903229033290342903529036290372903829039290402904129042290432904429045290462904729048290492905029051290522905329054290552905629057290582905929060290612906229063290642906529066290672906829069290702907129072290732907429075290762907729078290792908029081290822908329084290852908629087290882908929090290912909229093290942909529096290972909829099291002910129102291032910429105291062910729108291092911029111291122911329114291152911629117291182911929120291212912229123291242912529126291272912829129291302913129132291332913429135291362913729138291392914029141291422914329144291452914629147291482914929150291512915229153291542915529156291572915829159291602916129162291632916429165291662916729168291692917029171291722917329174291752917629177291782917929180291812918229183291842918529186291872918829189291902919129192291932919429195291962919729198291992920029201292022920329204292052920629207292082920929210292112921229213292142921529216292172921829219292202922129222292232922429225292262922729228292292923029231292322923329234292352923629237292382923929240292412924229243292442924529246292472924829249292502925129252292532925429255292562925729258292592926029261292622926329264292652926629267292682926929270292712927229273292742927529276292772927829279292802928129282292832928429285292862928729288292892929029291292922929329294292952929629297292982929929300293012930229303293042930529306293072930829309293102931129312293132931429315293162931729318293192932029321293222932329324293252932629327293282932929330293312933229333293342933529336293372933829339293402934129342293432934429345293462934729348293492935029351293522935329354293552935629357293582935929360293612936229363293642936529366293672936829369293702937129372293732937429375293762937729378293792938029381293822938329384293852938629387293882938929390293912939229393293942939529396293972939829399294002940129402294032940429405294062940729408294092941029411294122941329414294152941629417294182941929420294212942229423294242942529426294272942829429294302943129432294332943429435294362943729438294392944029441294422944329444294452944629447294482944929450294512945229453294542945529456294572945829459294602946129462294632946429465294662946729468294692947029471294722947329474294752947629477294782947929480294812948229483294842948529486294872948829489294902949129492294932949429495294962949729498294992950029501295022950329504295052950629507295082950929510295112951229513295142951529516295172951829519295202952129522295232952429525295262952729528295292953029531295322953329534295352953629537295382953929540295412954229543295442954529546295472954829549295502955129552295532955429555295562955729558295592956029561295622956329564295652956629567295682956929570295712957229573295742957529576295772957829579295802958129582295832958429585295862958729588295892959029591295922959329594295952959629597295982959929600296012960229603296042960529606296072960829609296102961129612296132961429615296162961729618296192962029621296222962329624296252962629627296282962929630296312963229633296342963529636296372963829639296402964129642296432964429645296462964729648296492965029651296522965329654296552965629657296582965929660296612966229663296642966529666296672966829669296702967129672296732967429675296762967729678296792968029681296822968329684296852968629687296882968929690296912969229693296942969529696296972969829699297002970129702297032970429705297062970729708297092971029711297122971329714297152971629717297182971929720297212972229723297242972529726297272972829729297302973129732297332973429735297362973729738297392974029741297422974329744297452974629747297482974929750297512975229753297542975529756297572975829759297602976129762297632976429765297662976729768297692977029771297722977329774297752977629777297782977929780297812978229783297842978529786297872978829789297902979129792297932979429795297962979729798297992980029801298022980329804298052980629807298082980929810298112981229813298142981529816298172981829819298202982129822298232982429825298262982729828298292983029831298322983329834298352983629837298382983929840298412984229843298442984529846298472984829849298502985129852298532985429855298562985729858298592986029861298622986329864298652986629867298682986929870298712987229873298742987529876298772987829879298802988129882298832988429885298862988729888298892989029891298922989329894298952989629897298982989929900299012990229903299042990529906299072990829909299102991129912299132991429915299162991729918299192992029921299222992329924299252992629927299282992929930299312993229933299342993529936299372993829939299402994129942299432994429945299462994729948299492995029951299522995329954299552995629957299582995929960299612996229963299642996529966299672996829969299702997129972299732997429975299762997729978299792998029981299822998329984299852998629987299882998929990299912999229993299942999529996299972999829999300003000130002300033000430005300063000730008300093001030011300123001330014300153001630017300183001930020300213002230023300243002530026300273002830029300303003130032300333003430035300363003730038300393004030041300423004330044300453004630047300483004930050300513005230053300543005530056300573005830059300603006130062300633006430065300663006730068300693007030071300723007330074300753007630077300783007930080300813008230083300843008530086300873008830089300903009130092300933009430095300963009730098300993010030101301023010330104301053010630107301083010930110301113011230113301143011530116301173011830119301203012130122301233012430125301263012730128301293013030131301323013330134301353013630137301383013930140301413014230143301443014530146301473014830149301503015130152301533015430155301563015730158301593016030161301623016330164301653016630167301683016930170301713017230173301743017530176301773017830179301803018130182301833018430185301863018730188301893019030191301923019330194301953019630197301983019930200302013020230203302043020530206302073020830209302103021130212302133021430215302163021730218302193022030221302223022330224302253022630227302283022930230302313023230233302343023530236302373023830239302403024130242302433024430245302463024730248302493025030251302523025330254302553025630257302583025930260302613026230263302643026530266302673026830269302703027130272302733027430275302763027730278302793028030281302823028330284302853028630287302883028930290302913029230293302943029530296302973029830299303003030130302303033030430305303063030730308303093031030311303123031330314303153031630317303183031930320303213032230323303243032530326303273032830329303303033130332303333033430335303363033730338303393034030341303423034330344303453034630347303483034930350303513035230353303543035530356303573035830359303603036130362303633036430365303663036730368303693037030371303723037330374303753037630377303783037930380303813038230383303843038530386303873038830389303903039130392303933039430395303963039730398303993040030401304023040330404304053040630407304083040930410304113041230413304143041530416304173041830419304203042130422304233042430425304263042730428304293043030431304323043330434304353043630437304383043930440304413044230443304443044530446304473044830449304503045130452304533045430455304563045730458304593046030461304623046330464304653046630467304683046930470304713047230473304743047530476304773047830479304803048130482304833048430485304863048730488304893049030491304923049330494304953049630497304983049930500305013050230503305043050530506305073050830509305103051130512305133051430515305163051730518305193052030521305223052330524305253052630527305283052930530305313053230533305343053530536305373053830539305403054130542305433054430545305463054730548305493055030551305523055330554305553055630557305583055930560305613056230563305643056530566305673056830569305703057130572305733057430575305763057730578305793058030581305823058330584305853058630587305883058930590305913059230593305943059530596305973059830599306003060130602306033060430605306063060730608306093061030611306123061330614306153061630617306183061930620306213062230623306243062530626306273062830629306303063130632306333063430635306363063730638306393064030641306423064330644306453064630647306483064930650306513065230653306543065530656306573065830659306603066130662306633066430665306663066730668306693067030671306723067330674306753067630677306783067930680306813068230683306843068530686306873068830689306903069130692306933069430695306963069730698306993070030701307023070330704307053070630707307083070930710307113071230713307143071530716307173071830719307203072130722307233072430725307263072730728307293073030731307323073330734307353073630737307383073930740307413074230743307443074530746307473074830749307503075130752307533075430755307563075730758307593076030761307623076330764307653076630767307683076930770307713077230773307743077530776307773077830779307803078130782307833078430785307863078730788307893079030791307923079330794307953079630797307983079930800308013080230803308043080530806308073080830809308103081130812308133081430815308163081730818308193082030821308223082330824308253082630827308283082930830308313083230833308343083530836308373083830839308403084130842308433084430845308463084730848308493085030851308523085330854308553085630857308583085930860308613086230863308643086530866308673086830869308703087130872308733087430875308763087730878308793088030881308823088330884308853088630887308883088930890308913089230893308943089530896308973089830899309003090130902309033090430905309063090730908309093091030911309123091330914309153091630917309183091930920309213092230923309243092530926309273092830929309303093130932309333093430935309363093730938309393094030941309423094330944309453094630947309483094930950309513095230953309543095530956309573095830959309603096130962309633096430965309663096730968309693097030971309723097330974309753097630977309783097930980309813098230983309843098530986309873098830989309903099130992309933099430995309963099730998309993100031001310023100331004310053100631007310083100931010310113101231013310143101531016310173101831019310203102131022310233102431025310263102731028310293103031031310323103331034310353103631037310383103931040310413104231043310443104531046310473104831049310503105131052310533105431055310563105731058310593106031061310623106331064310653106631067310683106931070310713107231073310743107531076310773107831079310803108131082310833108431085310863108731088310893109031091310923109331094310953109631097310983109931100311013110231103311043110531106311073110831109311103111131112311133111431115311163111731118311193112031121311223112331124311253112631127311283112931130311313113231133311343113531136311373113831139311403114131142311433114431145311463114731148311493115031151311523115331154311553115631157311583115931160311613116231163311643116531166311673116831169311703117131172311733117431175311763117731178311793118031181311823118331184311853118631187311883118931190311913119231193311943119531196311973119831199312003120131202312033120431205312063120731208312093121031211312123121331214312153121631217312183121931220312213122231223312243122531226312273122831229312303123131232312333123431235312363123731238312393124031241312423124331244312453124631247312483124931250312513125231253312543125531256312573125831259312603126131262312633126431265312663126731268312693127031271312723127331274312753127631277312783127931280312813128231283312843128531286312873128831289312903129131292312933129431295312963129731298312993130031301313023130331304313053130631307313083130931310313113131231313313143131531316313173131831319313203132131322313233132431325313263132731328313293133031331313323133331334313353133631337313383133931340313413134231343313443134531346313473134831349313503135131352313533135431355313563135731358313593136031361313623136331364313653136631367313683136931370313713137231373313743137531376313773137831379313803138131382313833138431385313863138731388313893139031391313923139331394313953139631397313983139931400314013140231403314043140531406314073140831409314103141131412314133141431415314163141731418314193142031421314223142331424314253142631427314283142931430314313143231433314343143531436314373143831439314403144131442314433144431445314463144731448314493145031451314523145331454314553145631457314583145931460314613146231463314643146531466314673146831469314703147131472314733147431475314763147731478314793148031481314823148331484314853148631487314883148931490314913149231493314943149531496314973149831499315003150131502315033150431505315063150731508315093151031511315123151331514315153151631517315183151931520315213152231523315243152531526315273152831529315303153131532315333153431535315363153731538315393154031541315423154331544315453154631547315483154931550315513155231553315543155531556315573155831559315603156131562315633156431565315663156731568315693157031571315723157331574315753157631577315783157931580315813158231583315843158531586315873158831589315903159131592315933159431595315963159731598315993160031601316023160331604316053160631607316083160931610316113161231613316143161531616316173161831619316203162131622316233162431625316263162731628316293163031631316323163331634316353163631637316383163931640316413164231643316443164531646316473164831649316503165131652316533165431655316563165731658316593166031661316623166331664316653166631667316683166931670316713167231673316743167531676316773167831679316803168131682316833168431685316863168731688316893169031691316923169331694316953169631697316983169931700317013170231703317043170531706317073170831709317103171131712317133171431715317163171731718317193172031721317223172331724317253172631727317283172931730317313173231733317343173531736317373173831739317403174131742317433174431745317463174731748317493175031751317523175331754317553175631757317583175931760317613176231763317643176531766317673176831769317703177131772317733177431775317763177731778317793178031781317823178331784317853178631787317883178931790317913179231793317943179531796317973179831799318003180131802318033180431805318063180731808318093181031811318123181331814318153181631817318183181931820318213182231823318243182531826318273182831829318303183131832318333183431835318363183731838318393184031841318423184331844318453184631847318483184931850318513185231853318543185531856318573185831859318603186131862318633186431865318663186731868318693187031871318723187331874318753187631877318783187931880318813188231883318843188531886318873188831889318903189131892318933189431895318963189731898318993190031901319023190331904319053190631907319083190931910319113191231913319143191531916319173191831919319203192131922319233192431925319263192731928319293193031931319323193331934319353193631937319383193931940319413194231943319443194531946319473194831949319503195131952319533195431955319563195731958319593196031961319623196331964319653196631967319683196931970319713197231973319743197531976319773197831979319803198131982319833198431985319863198731988319893199031991319923199331994319953199631997319983199932000320013200232003320043200532006320073200832009320103201132012320133201432015320163201732018320193202032021320223202332024320253202632027320283202932030320313203232033320343203532036320373203832039320403204132042320433204432045320463204732048320493205032051320523205332054320553205632057320583205932060320613206232063320643206532066320673206832069320703207132072320733207432075320763207732078320793208032081320823208332084320853208632087320883208932090320913209232093320943209532096320973209832099321003210132102321033210432105321063210732108321093211032111321123211332114321153211632117321183211932120321213212232123321243212532126321273212832129321303213132132321333213432135321363213732138321393214032141321423214332144321453214632147321483214932150321513215232153321543215532156321573215832159321603216132162321633216432165321663216732168321693217032171321723217332174321753217632177321783217932180321813218232183321843218532186321873218832189321903219132192321933219432195321963219732198321993220032201322023220332204322053220632207322083220932210322113221232213322143221532216322173221832219322203222132222322233222432225322263222732228322293223032231322323223332234322353223632237322383223932240322413224232243322443224532246322473224832249322503225132252322533225432255322563225732258322593226032261322623226332264322653226632267322683226932270322713227232273322743227532276322773227832279322803228132282322833228432285322863228732288322893229032291322923229332294322953229632297322983229932300323013230232303323043230532306323073230832309323103231132312323133231432315323163231732318323193232032321323223232332324323253232632327323283232932330323313233232333323343233532336323373233832339323403234132342323433234432345323463234732348323493235032351323523235332354323553235632357323583235932360323613236232363323643236532366323673236832369323703237132372323733237432375323763237732378323793238032381323823238332384323853238632387323883238932390323913239232393323943239532396323973239832399324003240132402324033240432405324063240732408324093241032411324123241332414324153241632417324183241932420324213242232423324243242532426324273242832429324303243132432324333243432435324363243732438324393244032441324423244332444324453244632447324483244932450324513245232453324543245532456324573245832459324603246132462324633246432465324663246732468324693247032471324723247332474324753247632477324783247932480324813248232483324843248532486324873248832489324903249132492324933249432495324963249732498324993250032501325023250332504325053250632507325083250932510325113251232513325143251532516325173251832519325203252132522325233252432525325263252732528325293253032531325323253332534325353253632537325383253932540325413254232543325443254532546325473254832549325503255132552325533255432555325563255732558325593256032561325623256332564325653256632567325683256932570325713257232573325743257532576325773257832579325803258132582325833258432585325863258732588325893259032591325923259332594325953259632597325983259932600326013260232603326043260532606326073260832609326103261132612326133261432615326163261732618326193262032621326223262332624326253262632627326283262932630326313263232633326343263532636326373263832639326403264132642326433264432645326463264732648326493265032651326523265332654326553265632657326583265932660326613266232663326643266532666326673266832669326703267132672326733267432675326763267732678326793268032681326823268332684326853268632687326883268932690326913269232693326943269532696326973269832699327003270132702327033270432705327063270732708327093271032711327123271332714327153271632717327183271932720327213272232723327243272532726327273272832729327303273132732327333273432735327363273732738327393274032741327423274332744327453274632747327483274932750327513275232753327543275532756327573275832759327603276132762327633276432765327663276732768327693277032771327723277332774327753277632777327783277932780327813278232783327843278532786327873278832789327903279132792327933279432795327963279732798327993280032801328023280332804328053280632807328083280932810328113281232813328143281532816328173281832819328203282132822328233282432825328263282732828328293283032831328323283332834328353283632837328383283932840328413284232843328443284532846328473284832849328503285132852328533285432855328563285732858328593286032861328623286332864328653286632867328683286932870328713287232873328743287532876328773287832879328803288132882328833288432885328863288732888328893289032891328923289332894328953289632897328983289932900329013290232903329043290532906329073290832909329103291132912329133291432915329163291732918329193292032921329223292332924329253292632927329283292932930329313293232933329343293532936329373293832939329403294132942329433294432945329463294732948329493295032951329523295332954329553295632957329583295932960329613296232963329643296532966329673296832969329703297132972329733297432975329763297732978329793298032981329823298332984329853298632987329883298932990329913299232993329943299532996329973299832999330003300133002330033300433005330063300733008330093301033011330123301333014330153301633017330183301933020330213302233023330243302533026330273302833029330303303133032330333303433035330363303733038330393304033041330423304333044330453304633047330483304933050330513305233053330543305533056330573305833059330603306133062330633306433065330663306733068330693307033071330723307333074330753307633077330783307933080330813308233083330843308533086330873308833089330903309133092330933309433095330963309733098330993310033101331023310333104331053310633107331083310933110331113311233113331143311533116331173311833119331203312133122331233312433125331263312733128331293313033131331323313333134331353313633137331383313933140331413314233143331443314533146331473314833149331503315133152331533315433155331563315733158331593316033161331623316333164331653316633167331683316933170331713317233173331743317533176331773317833179331803318133182331833318433185331863318733188331893319033191331923319333194331953319633197331983319933200332013320233203332043320533206332073320833209332103321133212332133321433215332163321733218332193322033221332223322333224332253322633227332283322933230332313323233233332343323533236332373323833239332403324133242332433324433245332463324733248332493325033251332523325333254332553325633257332583325933260332613326233263332643326533266332673326833269332703327133272332733327433275332763327733278332793328033281332823328333284332853328633287332883328933290332913329233293332943329533296332973329833299333003330133302333033330433305333063330733308333093331033311333123331333314333153331633317333183331933320333213332233323333243332533326333273332833329333303333133332333333333433335333363333733338333393334033341333423334333344333453334633347333483334933350333513335233353333543335533356333573335833359333603336133362333633336433365333663336733368333693337033371333723337333374333753337633377333783337933380333813338233383333843338533386333873338833389333903339133392333933339433395333963339733398333993340033401334023340333404334053340633407334083340933410334113341233413334143341533416334173341833419334203342133422334233342433425334263342733428334293343033431334323343333434334353343633437334383343933440334413344233443334443344533446334473344833449334503345133452334533345433455334563345733458334593346033461334623346333464334653346633467334683346933470334713347233473334743347533476334773347833479334803348133482334833348433485334863348733488334893349033491334923349333494334953349633497334983349933500335013350233503335043350533506335073350833509335103351133512335133351433515335163351733518335193352033521335223352333524335253352633527335283352933530335313353233533335343353533536335373353833539335403354133542335433354433545335463354733548335493355033551335523355333554335553355633557335583355933560335613356233563335643356533566335673356833569335703357133572335733357433575335763357733578335793358033581335823358333584335853358633587335883358933590335913359233593335943359533596335973359833599336003360133602336033360433605336063360733608336093361033611336123361333614336153361633617336183361933620336213362233623336243362533626336273362833629336303363133632336333363433635336363363733638336393364033641336423364333644336453364633647336483364933650336513365233653336543365533656336573365833659336603366133662336633366433665336663366733668336693367033671336723367333674336753367633677336783367933680336813368233683336843368533686336873368833689336903369133692336933369433695336963369733698336993370033701337023370333704337053370633707337083370933710337113371233713337143371533716337173371833719337203372133722337233372433725337263372733728337293373033731337323373333734337353373633737337383373933740337413374233743337443374533746337473374833749337503375133752337533375433755337563375733758337593376033761337623376333764337653376633767337683376933770337713377233773337743377533776337773377833779337803378133782337833378433785337863378733788337893379033791337923379333794337953379633797337983379933800338013380233803338043380533806338073380833809338103381133812338133381433815338163381733818338193382033821338223382333824338253382633827338283382933830338313383233833338343383533836338373383833839338403384133842338433384433845338463384733848338493385033851338523385333854338553385633857338583385933860338613386233863338643386533866338673386833869338703387133872338733387433875338763387733878338793388033881338823388333884338853388633887338883388933890338913389233893338943389533896338973389833899339003390133902339033390433905339063390733908339093391033911339123391333914339153391633917339183391933920339213392233923339243392533926339273392833929339303393133932339333393433935339363393733938339393394033941339423394333944339453394633947339483394933950339513395233953339543395533956339573395833959339603396133962339633396433965339663396733968339693397033971339723397333974339753397633977339783397933980339813398233983339843398533986339873398833989339903399133992339933399433995339963399733998339993400034001340023400334004340053400634007340083400934010340113401234013340143401534016340173401834019340203402134022340233402434025340263402734028340293403034031340323403334034340353403634037340383403934040340413404234043340443404534046340473404834049340503405134052340533405434055340563405734058340593406034061340623406334064340653406634067340683406934070340713407234073340743407534076340773407834079340803408134082340833408434085340863408734088340893409034091340923409334094340953409634097340983409934100341013410234103341043410534106341073410834109341103411134112341133411434115341163411734118341193412034121341223412334124341253412634127341283412934130341313413234133341343413534136341373413834139341403414134142341433414434145341463414734148341493415034151341523415334154341553415634157341583415934160341613416234163341643416534166341673416834169341703417134172341733417434175341763417734178341793418034181341823418334184341853418634187341883418934190341913419234193341943419534196341973419834199342003420134202342033420434205342063420734208342093421034211342123421334214342153421634217342183421934220342213422234223342243422534226342273422834229342303423134232342333423434235342363423734238342393424034241342423424334244342453424634247342483424934250342513425234253342543425534256342573425834259342603426134262342633426434265342663426734268342693427034271342723427334274342753427634277342783427934280342813428234283342843428534286342873428834289342903429134292342933429434295342963429734298342993430034301343023430334304343053430634307343083430934310343113431234313343143431534316343173431834319343203432134322343233432434325343263432734328343293433034331343323433334334343353433634337343383433934340343413434234343343443434534346343473434834349343503435134352343533435434355343563435734358343593436034361343623436334364343653436634367343683436934370343713437234373343743437534376343773437834379343803438134382343833438434385343863438734388343893439034391343923439334394343953439634397343983439934400344013440234403344043440534406344073440834409344103441134412344133441434415344163441734418344193442034421344223442334424344253442634427344283442934430344313443234433344343443534436344373443834439344403444134442344433444434445344463444734448344493445034451344523445334454344553445634457344583445934460344613446234463344643446534466344673446834469344703447134472344733447434475344763447734478344793448034481344823448334484344853448634487344883448934490344913449234493344943449534496344973449834499345003450134502345033450434505345063450734508345093451034511345123451334514345153451634517345183451934520345213452234523345243452534526345273452834529345303453134532345333453434535345363453734538345393454034541345423454334544345453454634547345483454934550345513455234553345543455534556345573455834559345603456134562345633456434565345663456734568345693457034571345723457334574345753457634577345783457934580345813458234583345843458534586345873458834589345903459134592345933459434595345963459734598345993460034601346023460334604346053460634607346083460934610346113461234613346143461534616346173461834619346203462134622346233462434625346263462734628346293463034631346323463334634346353463634637346383463934640346413464234643346443464534646346473464834649346503465134652346533465434655346563465734658346593466034661346623466334664346653466634667346683466934670346713467234673346743467534676346773467834679346803468134682346833468434685346863468734688346893469034691346923469334694346953469634697346983469934700347013470234703347043470534706347073470834709347103471134712347133471434715347163471734718347193472034721347223472334724347253472634727347283472934730347313473234733347343473534736347373473834739347403474134742347433474434745347463474734748347493475034751347523475334754347553475634757347583475934760347613476234763347643476534766347673476834769347703477134772347733477434775347763477734778347793478034781347823478334784347853478634787347883478934790347913479234793347943479534796347973479834799348003480134802348033480434805348063480734808348093481034811348123481334814348153481634817348183481934820348213482234823348243482534826348273482834829348303483134832348333483434835348363483734838348393484034841348423484334844348453484634847348483484934850348513485234853348543485534856348573485834859348603486134862348633486434865348663486734868348693487034871348723487334874348753487634877348783487934880348813488234883348843488534886348873488834889348903489134892348933489434895348963489734898348993490034901349023490334904349053490634907349083490934910349113491234913349143491534916349173491834919349203492134922349233492434925349263492734928349293493034931349323493334934349353493634937349383493934940349413494234943349443494534946349473494834949349503495134952349533495434955349563495734958349593496034961349623496334964349653496634967349683496934970349713497234973349743497534976349773497834979349803498134982349833498434985349863498734988349893499034991349923499334994349953499634997349983499935000350013500235003350043500535006350073500835009350103501135012350133501435015350163501735018350193502035021350223502335024350253502635027350283502935030350313503235033350343503535036350373503835039350403504135042350433504435045350463504735048350493505035051350523505335054350553505635057350583505935060350613506235063350643506535066350673506835069350703507135072350733507435075350763507735078350793508035081350823508335084350853508635087350883508935090350913509235093350943509535096350973509835099351003510135102351033510435105351063510735108351093511035111351123511335114351153511635117351183511935120351213512235123351243512535126351273512835129351303513135132351333513435135351363513735138351393514035141351423514335144351453514635147351483514935150351513515235153351543515535156351573515835159351603516135162351633516435165351663516735168351693517035171351723517335174351753517635177351783517935180351813518235183351843518535186351873518835189351903519135192351933519435195351963519735198351993520035201352023520335204352053520635207352083520935210352113521235213352143521535216352173521835219352203522135222352233522435225352263522735228352293523035231352323523335234352353523635237352383523935240352413524235243352443524535246352473524835249352503525135252352533525435255352563525735258352593526035261352623526335264352653526635267352683526935270352713527235273352743527535276352773527835279352803528135282352833528435285352863528735288352893529035291352923529335294352953529635297352983529935300353013530235303353043530535306353073530835309353103531135312353133531435315353163531735318353193532035321353223532335324353253532635327353283532935330353313533235333353343533535336353373533835339353403534135342353433534435345353463534735348353493535035351353523535335354353553535635357353583535935360353613536235363353643536535366353673536835369353703537135372353733537435375353763537735378353793538035381353823538335384353853538635387353883538935390353913539235393353943539535396353973539835399354003540135402354033540435405354063540735408354093541035411354123541335414354153541635417354183541935420354213542235423354243542535426354273542835429354303543135432354333543435435354363543735438354393544035441354423544335444354453544635447354483544935450354513545235453354543545535456354573545835459354603546135462354633546435465354663546735468354693547035471354723547335474354753547635477354783547935480354813548235483354843548535486354873548835489354903549135492354933549435495354963549735498354993550035501355023550335504355053550635507355083550935510355113551235513355143551535516355173551835519355203552135522355233552435525355263552735528355293553035531355323553335534355353553635537355383553935540355413554235543355443554535546355473554835549355503555135552355533555435555355563555735558355593556035561355623556335564355653556635567355683556935570355713557235573355743557535576355773557835579355803558135582355833558435585355863558735588355893559035591355923559335594355953559635597355983559935600356013560235603356043560535606356073560835609356103561135612356133561435615356163561735618356193562035621356223562335624356253562635627356283562935630356313563235633356343563535636356373563835639356403564135642356433564435645356463564735648356493565035651356523565335654356553565635657356583565935660356613566235663356643566535666356673566835669356703567135672356733567435675356763567735678356793568035681356823568335684356853568635687356883568935690356913569235693356943569535696356973569835699357003570135702357033570435705357063570735708357093571035711357123571335714357153571635717357183571935720357213572235723357243572535726357273572835729357303573135732357333573435735357363573735738357393574035741357423574335744357453574635747357483574935750357513575235753357543575535756357573575835759357603576135762357633576435765357663576735768357693577035771357723577335774357753577635777357783577935780357813578235783357843578535786357873578835789357903579135792357933579435795357963579735798357993580035801358023580335804358053580635807358083580935810358113581235813358143581535816358173581835819358203582135822358233582435825358263582735828358293583035831358323583335834358353583635837358383583935840358413584235843358443584535846358473584835849358503585135852358533585435855358563585735858358593586035861358623586335864358653586635867358683586935870358713587235873358743587535876358773587835879358803588135882358833588435885358863588735888358893589035891358923589335894358953589635897358983589935900359013590235903359043590535906359073590835909359103591135912359133591435915359163591735918359193592035921359223592335924359253592635927359283592935930359313593235933359343593535936359373593835939359403594135942359433594435945359463594735948359493595035951359523595335954359553595635957359583595935960359613596235963359643596535966359673596835969359703597135972359733597435975359763597735978359793598035981359823598335984359853598635987359883598935990359913599235993359943599535996359973599835999360003600136002360033600436005360063600736008360093601036011360123601336014360153601636017360183601936020360213602236023360243602536026360273602836029360303603136032360333603436035360363603736038360393604036041360423604336044360453604636047360483604936050360513605236053360543605536056360573605836059360603606136062360633606436065360663606736068360693607036071360723607336074360753607636077360783607936080360813608236083360843608536086360873608836089360903609136092360933609436095360963609736098360993610036101361023610336104361053610636107361083610936110361113611236113361143611536116361173611836119361203612136122361233612436125361263612736128361293613036131361323613336134361353613636137361383613936140361413614236143361443614536146361473614836149361503615136152361533615436155361563615736158361593616036161361623616336164361653616636167361683616936170361713617236173361743617536176361773617836179361803618136182361833618436185361863618736188361893619036191361923619336194361953619636197361983619936200362013620236203362043620536206362073620836209362103621136212362133621436215362163621736218362193622036221362223622336224362253622636227362283622936230362313623236233362343623536236362373623836239362403624136242362433624436245362463624736248362493625036251362523625336254362553625636257362583625936260362613626236263362643626536266362673626836269362703627136272362733627436275362763627736278362793628036281362823628336284362853628636287362883628936290362913629236293362943629536296362973629836299363003630136302363033630436305363063630736308363093631036311363123631336314363153631636317363183631936320363213632236323363243632536326363273632836329363303633136332363333633436335363363633736338363393634036341363423634336344363453634636347363483634936350363513635236353363543635536356363573635836359363603636136362363633636436365363663636736368363693637036371363723637336374363753637636377363783637936380363813638236383363843638536386363873638836389363903639136392363933639436395363963639736398363993640036401364023640336404364053640636407364083640936410364113641236413364143641536416364173641836419364203642136422364233642436425364263642736428364293643036431364323643336434364353643636437364383643936440364413644236443364443644536446364473644836449364503645136452364533645436455364563645736458364593646036461364623646336464364653646636467364683646936470364713647236473364743647536476364773647836479364803648136482364833648436485364863648736488364893649036491364923649336494364953649636497364983649936500365013650236503365043650536506365073650836509365103651136512365133651436515365163651736518365193652036521365223652336524365253652636527365283652936530365313653236533365343653536536365373653836539365403654136542365433654436545365463654736548365493655036551365523655336554365553655636557365583655936560365613656236563365643656536566365673656836569365703657136572365733657436575365763657736578365793658036581365823658336584365853658636587365883658936590365913659236593365943659536596365973659836599366003660136602366033660436605366063660736608366093661036611366123661336614366153661636617366183661936620366213662236623366243662536626366273662836629366303663136632366333663436635366363663736638366393664036641366423664336644366453664636647366483664936650366513665236653366543665536656366573665836659366603666136662366633666436665366663666736668366693667036671366723667336674366753667636677366783667936680366813668236683366843668536686366873668836689366903669136692366933669436695366963669736698366993670036701367023670336704367053670636707367083670936710367113671236713367143671536716367173671836719367203672136722367233672436725367263672736728367293673036731367323673336734367353673636737367383673936740367413674236743367443674536746367473674836749367503675136752367533675436755367563675736758367593676036761367623676336764367653676636767367683676936770367713677236773367743677536776367773677836779367803678136782367833678436785367863678736788367893679036791367923679336794367953679636797367983679936800368013680236803368043680536806368073680836809368103681136812368133681436815368163681736818368193682036821368223682336824368253682636827368283682936830368313683236833368343683536836368373683836839368403684136842368433684436845368463684736848368493685036851368523685336854368553685636857368583685936860368613686236863368643686536866368673686836869368703687136872368733687436875368763687736878368793688036881368823688336884368853688636887368883688936890368913689236893368943689536896368973689836899369003690136902369033690436905369063690736908369093691036911369123691336914369153691636917369183691936920369213692236923369243692536926369273692836929369303693136932369333693436935369363693736938369393694036941369423694336944369453694636947369483694936950369513695236953369543695536956369573695836959369603696136962369633696436965369663696736968369693697036971369723697336974369753697636977369783697936980369813698236983369843698536986369873698836989369903699136992369933699436995369963699736998369993700037001370023700337004370053700637007370083700937010370113701237013370143701537016370173701837019370203702137022370233702437025370263702737028370293703037031370323703337034370353703637037370383703937040370413704237043370443704537046370473704837049370503705137052370533705437055370563705737058370593706037061370623706337064370653706637067370683706937070370713707237073370743707537076370773707837079370803708137082370833708437085370863708737088370893709037091370923709337094370953709637097370983709937100371013710237103371043710537106371073710837109371103711137112371133711437115371163711737118371193712037121371223712337124371253712637127371283712937130371313713237133371343713537136371373713837139371403714137142371433714437145371463714737148371493715037151371523715337154371553715637157371583715937160371613716237163371643716537166371673716837169371703717137172371733717437175371763717737178371793718037181371823718337184371853718637187371883718937190371913719237193371943719537196371973719837199372003720137202372033720437205372063720737208372093721037211372123721337214372153721637217372183721937220372213722237223372243722537226372273722837229372303723137232372333723437235372363723737238372393724037241372423724337244372453724637247372483724937250372513725237253372543725537256372573725837259372603726137262372633726437265372663726737268372693727037271372723727337274372753727637277372783727937280372813728237283372843728537286372873728837289372903729137292372933729437295372963729737298372993730037301373023730337304373053730637307373083730937310373113731237313373143731537316373173731837319373203732137322373233732437325373263732737328373293733037331373323733337334373353733637337373383733937340373413734237343373443734537346373473734837349373503735137352373533735437355373563735737358373593736037361373623736337364373653736637367373683736937370373713737237373373743737537376373773737837379373803738137382373833738437385373863738737388373893739037391373923739337394373953739637397373983739937400374013740237403374043740537406374073740837409374103741137412374133741437415374163741737418374193742037421374223742337424374253742637427374283742937430374313743237433374343743537436374373743837439374403744137442374433744437445374463744737448374493745037451374523745337454374553745637457374583745937460374613746237463374643746537466374673746837469374703747137472374733747437475374763747737478374793748037481374823748337484374853748637487374883748937490374913749237493374943749537496374973749837499375003750137502375033750437505375063750737508375093751037511375123751337514375153751637517375183751937520375213752237523375243752537526375273752837529375303753137532375333753437535375363753737538375393754037541375423754337544375453754637547375483754937550375513755237553375543755537556375573755837559375603756137562375633756437565375663756737568375693757037571375723757337574375753757637577375783757937580375813758237583375843758537586375873758837589375903759137592375933759437595375963759737598375993760037601376023760337604376053760637607376083760937610376113761237613376143761537616376173761837619376203762137622376233762437625376263762737628376293763037631376323763337634376353763637637376383763937640376413764237643376443764537646376473764837649376503765137652376533765437655376563765737658376593766037661376623766337664376653766637667376683766937670376713767237673376743767537676376773767837679376803768137682376833768437685376863768737688376893769037691376923769337694376953769637697376983769937700377013770237703377043770537706377073770837709377103771137712377133771437715377163771737718377193772037721377223772337724377253772637727377283772937730377313773237733377343773537736377373773837739377403774137742377433774437745377463774737748377493775037751377523775337754377553775637757377583775937760377613776237763377643776537766377673776837769377703777137772377733777437775377763777737778377793778037781377823778337784377853778637787377883778937790377913779237793377943779537796377973779837799378003780137802378033780437805378063780737808378093781037811378123781337814378153781637817378183781937820378213782237823378243782537826378273782837829378303783137832378333783437835378363783737838378393784037841378423784337844378453784637847378483784937850378513785237853378543785537856378573785837859378603786137862378633786437865378663786737868378693787037871378723787337874378753787637877378783787937880378813788237883378843788537886378873788837889378903789137892378933789437895378963789737898378993790037901379023790337904379053790637907379083790937910379113791237913379143791537916379173791837919379203792137922379233792437925379263792737928379293793037931379323793337934379353793637937379383793937940379413794237943379443794537946379473794837949379503795137952379533795437955379563795737958379593796037961379623796337964379653796637967379683796937970379713797237973379743797537976379773797837979379803798137982379833798437985379863798737988379893799037991379923799337994379953799637997379983799938000380013800238003380043800538006380073800838009380103801138012380133801438015380163801738018380193802038021380223802338024380253802638027380283802938030380313803238033380343803538036380373803838039380403804138042380433804438045380463804738048380493805038051380523805338054380553805638057380583805938060380613806238063380643806538066380673806838069380703807138072380733807438075380763807738078380793808038081380823808338084380853808638087380883808938090380913809238093380943809538096380973809838099381003810138102381033810438105381063810738108381093811038111381123811338114381153811638117381183811938120381213812238123381243812538126381273812838129381303813138132381333813438135381363813738138381393814038141381423814338144381453814638147381483814938150381513815238153381543815538156381573815838159381603816138162381633816438165381663816738168381693817038171381723817338174381753817638177381783817938180381813818238183381843818538186381873818838189381903819138192381933819438195381963819738198381993820038201382023820338204382053820638207382083820938210382113821238213382143821538216382173821838219382203822138222382233822438225382263822738228382293823038231382323823338234382353823638237382383823938240382413824238243382443824538246382473824838249382503825138252382533825438255382563825738258382593826038261382623826338264382653826638267382683826938270382713827238273382743827538276382773827838279382803828138282382833828438285382863828738288382893829038291382923829338294382953829638297382983829938300383013830238303383043830538306383073830838309383103831138312383133831438315383163831738318383193832038321383223832338324383253832638327383283832938330383313833238333383343833538336383373833838339383403834138342383433834438345383463834738348383493835038351383523835338354383553835638357383583835938360383613836238363383643836538366383673836838369383703837138372383733837438375383763837738378383793838038381383823838338384383853838638387383883838938390383913839238393383943839538396383973839838399384003840138402384033840438405384063840738408384093841038411384123841338414384153841638417384183841938420384213842238423384243842538426384273842838429384303843138432384333843438435384363843738438384393844038441384423844338444384453844638447384483844938450384513845238453384543845538456384573845838459384603846138462384633846438465384663846738468384693847038471384723847338474384753847638477384783847938480384813848238483384843848538486384873848838489384903849138492384933849438495384963849738498384993850038501385023850338504385053850638507385083850938510385113851238513385143851538516385173851838519385203852138522385233852438525385263852738528385293853038531385323853338534385353853638537385383853938540385413854238543385443854538546385473854838549385503855138552385533855438555385563855738558385593856038561385623856338564385653856638567385683856938570385713857238573385743857538576385773857838579385803858138582385833858438585385863858738588385893859038591385923859338594385953859638597385983859938600386013860238603386043860538606386073860838609386103861138612386133861438615386163861738618386193862038621386223862338624386253862638627386283862938630386313863238633386343863538636386373863838639386403864138642386433864438645386463864738648386493865038651386523865338654386553865638657386583865938660386613866238663386643866538666386673866838669386703867138672386733867438675386763867738678386793868038681386823868338684386853868638687386883868938690386913869238693386943869538696386973869838699387003870138702387033870438705387063870738708387093871038711387123871338714387153871638717387183871938720387213872238723387243872538726387273872838729387303873138732387333873438735387363873738738387393874038741387423874338744387453874638747387483874938750387513875238753387543875538756387573875838759387603876138762387633876438765387663876738768387693877038771387723877338774387753877638777387783877938780387813878238783387843878538786387873878838789387903879138792387933879438795387963879738798387993880038801388023880338804388053880638807388083880938810388113881238813388143881538816388173881838819388203882138822388233882438825388263882738828388293883038831388323883338834388353883638837388383883938840388413884238843388443884538846388473884838849388503885138852388533885438855388563885738858388593886038861388623886338864388653886638867388683886938870388713887238873388743887538876388773887838879388803888138882388833888438885388863888738888388893889038891388923889338894388953889638897388983889938900389013890238903389043890538906389073890838909389103891138912389133891438915389163891738918389193892038921389223892338924389253892638927389283892938930389313893238933389343893538936389373893838939389403894138942389433894438945389463894738948389493895038951389523895338954389553895638957389583895938960389613896238963389643896538966389673896838969389703897138972389733897438975389763897738978389793898038981389823898338984389853898638987389883898938990389913899238993389943899538996389973899838999390003900139002390033900439005390063900739008390093901039011390123901339014390153901639017390183901939020390213902239023390243902539026390273902839029390303903139032390333903439035390363903739038390393904039041390423904339044390453904639047390483904939050390513905239053390543905539056390573905839059390603906139062390633906439065390663906739068390693907039071390723907339074390753907639077390783907939080390813908239083390843908539086390873908839089390903909139092390933909439095390963909739098390993910039101391023910339104391053910639107391083910939110391113911239113391143911539116391173911839119391203912139122391233912439125391263912739128391293913039131391323913339134391353913639137391383913939140391413914239143391443914539146391473914839149391503915139152391533915439155391563915739158391593916039161391623916339164391653916639167391683916939170391713917239173391743917539176391773917839179391803918139182391833918439185391863918739188391893919039191391923919339194391953919639197391983919939200392013920239203392043920539206392073920839209392103921139212392133921439215392163921739218392193922039221392223922339224392253922639227392283922939230392313923239233392343923539236392373923839239392403924139242392433924439245392463924739248392493925039251392523925339254392553925639257392583925939260392613926239263392643926539266392673926839269392703927139272392733927439275392763927739278392793928039281392823928339284392853928639287392883928939290392913929239293392943929539296392973929839299393003930139302393033930439305393063930739308393093931039311393123931339314393153931639317393183931939320393213932239323393243932539326393273932839329393303933139332393333933439335393363933739338393393934039341393423934339344393453934639347393483934939350393513935239353393543935539356393573935839359393603936139362393633936439365393663936739368393693937039371393723937339374393753937639377393783937939380393813938239383393843938539386393873938839389393903939139392393933939439395393963939739398393993940039401394023940339404394053940639407394083940939410394113941239413394143941539416394173941839419394203942139422394233942439425394263942739428394293943039431394323943339434394353943639437394383943939440394413944239443394443944539446394473944839449394503945139452394533945439455394563945739458394593946039461394623946339464394653946639467394683946939470394713947239473394743947539476394773947839479394803948139482394833948439485394863948739488394893949039491394923949339494394953949639497394983949939500395013950239503395043950539506395073950839509395103951139512395133951439515395163951739518395193952039521395223952339524395253952639527395283952939530395313953239533395343953539536395373953839539395403954139542395433954439545395463954739548395493955039551395523955339554395553955639557395583955939560395613956239563395643956539566395673956839569395703957139572395733957439575395763957739578395793958039581395823958339584395853958639587395883958939590395913959239593395943959539596395973959839599396003960139602396033960439605396063960739608396093961039611396123961339614396153961639617396183961939620396213962239623396243962539626396273962839629396303963139632396333963439635396363963739638396393964039641396423964339644396453964639647396483964939650396513965239653396543965539656396573965839659396603966139662396633966439665396663966739668396693967039671396723967339674396753967639677396783967939680396813968239683396843968539686396873968839689396903969139692396933969439695396963969739698396993970039701397023970339704397053970639707397083970939710397113971239713397143971539716397173971839719397203972139722397233972439725397263972739728397293973039731397323973339734397353973639737397383973939740397413974239743397443974539746397473974839749397503975139752397533975439755397563975739758397593976039761397623976339764397653976639767397683976939770397713977239773397743977539776397773977839779397803978139782397833978439785397863978739788397893979039791397923979339794397953979639797397983979939800398013980239803398043980539806398073980839809398103981139812398133981439815398163981739818398193982039821398223982339824398253982639827398283982939830398313983239833398343983539836398373983839839398403984139842398433984439845398463984739848398493985039851398523985339854398553985639857398583985939860398613986239863398643986539866398673986839869398703987139872398733987439875398763987739878398793988039881398823988339884398853988639887398883988939890398913989239893398943989539896398973989839899399003990139902399033990439905399063990739908399093991039911399123991339914399153991639917399183991939920399213992239923399243992539926399273992839929399303993139932399333993439935399363993739938399393994039941399423994339944399453994639947399483994939950399513995239953399543995539956399573995839959399603996139962399633996439965399663996739968399693997039971399723997339974399753997639977399783997939980399813998239983399843998539986399873998839989399903999139992399933999439995399963999739998399994000040001400024000340004400054000640007400084000940010400114001240013400144001540016400174001840019400204002140022400234002440025400264002740028400294003040031400324003340034400354003640037400384003940040400414004240043400444004540046400474004840049400504005140052400534005440055400564005740058400594006040061400624006340064400654006640067400684006940070400714007240073400744007540076400774007840079400804008140082400834008440085400864008740088400894009040091400924009340094400954009640097400984009940100401014010240103401044010540106401074010840109401104011140112401134011440115401164011740118401194012040121401224012340124401254012640127401284012940130401314013240133401344013540136401374013840139401404014140142401434014440145401464014740148401494015040151401524015340154401554015640157401584015940160401614016240163401644016540166401674016840169401704017140172401734017440175401764017740178401794018040181401824018340184401854018640187401884018940190401914019240193401944019540196401974019840199402004020140202402034020440205402064020740208402094021040211402124021340214402154021640217402184021940220402214022240223402244022540226402274022840229402304023140232402334023440235402364023740238402394024040241402424024340244402454024640247402484024940250402514025240253402544025540256402574025840259402604026140262402634026440265402664026740268402694027040271402724027340274402754027640277402784027940280402814028240283402844028540286402874028840289402904029140292402934029440295402964029740298402994030040301403024030340304403054030640307403084030940310403114031240313403144031540316403174031840319403204032140322403234032440325403264032740328403294033040331403324033340334403354033640337403384033940340403414034240343403444034540346403474034840349403504035140352403534035440355403564035740358403594036040361403624036340364403654036640367403684036940370403714037240373403744037540376403774037840379403804038140382403834038440385403864038740388403894039040391403924039340394403954039640397403984039940400404014040240403404044040540406404074040840409404104041140412404134041440415404164041740418404194042040421404224042340424404254042640427404284042940430404314043240433404344043540436404374043840439404404044140442404434044440445404464044740448404494045040451404524045340454404554045640457404584045940460404614046240463404644046540466404674046840469404704047140472404734047440475404764047740478404794048040481404824048340484404854048640487404884048940490404914049240493404944049540496404974049840499405004050140502405034050440505405064050740508405094051040511405124051340514405154051640517405184051940520405214052240523405244052540526405274052840529405304053140532405334053440535405364053740538405394054040541405424054340544405454054640547405484054940550405514055240553405544055540556405574055840559405604056140562405634056440565405664056740568405694057040571405724057340574405754057640577405784057940580405814058240583405844058540586405874058840589405904059140592405934059440595405964059740598405994060040601406024060340604406054060640607406084060940610406114061240613406144061540616406174061840619406204062140622406234062440625406264062740628406294063040631406324063340634406354063640637406384063940640406414064240643406444064540646406474064840649406504065140652406534065440655406564065740658406594066040661406624066340664406654066640667406684066940670406714067240673406744067540676406774067840679406804068140682406834068440685406864068740688406894069040691406924069340694406954069640697406984069940700407014070240703407044070540706407074070840709407104071140712407134071440715407164071740718407194072040721407224072340724407254072640727407284072940730407314073240733407344073540736407374073840739407404074140742407434074440745407464074740748407494075040751407524075340754407554075640757407584075940760407614076240763407644076540766407674076840769407704077140772407734077440775407764077740778407794078040781407824078340784407854078640787407884078940790407914079240793407944079540796407974079840799408004080140802408034080440805408064080740808408094081040811408124081340814408154081640817408184081940820408214082240823408244082540826408274082840829408304083140832408334083440835408364083740838408394084040841408424084340844408454084640847408484084940850408514085240853408544085540856408574085840859408604086140862408634086440865408664086740868408694087040871408724087340874408754087640877408784087940880408814088240883408844088540886408874088840889408904089140892408934089440895408964089740898408994090040901409024090340904409054090640907409084090940910409114091240913409144091540916409174091840919409204092140922409234092440925409264092740928409294093040931409324093340934409354093640937409384093940940409414094240943409444094540946409474094840949409504095140952409534095440955409564095740958409594096040961409624096340964409654096640967409684096940970409714097240973409744097540976409774097840979409804098140982409834098440985409864098740988409894099040991409924099340994409954099640997409984099941000410014100241003410044100541006410074100841009410104101141012410134101441015410164101741018410194102041021410224102341024410254102641027410284102941030410314103241033410344103541036410374103841039410404104141042410434104441045410464104741048410494105041051410524105341054410554105641057410584105941060410614106241063410644106541066410674106841069410704107141072410734107441075410764107741078410794108041081410824108341084410854108641087410884108941090410914109241093410944109541096410974109841099411004110141102411034110441105411064110741108411094111041111411124111341114411154111641117411184111941120411214112241123411244112541126411274112841129411304113141132411334113441135411364113741138411394114041141411424114341144411454114641147411484114941150411514115241153411544115541156411574115841159411604116141162411634116441165411664116741168411694117041171411724117341174411754117641177411784117941180411814118241183411844118541186411874118841189411904119141192411934119441195411964119741198411994120041201412024120341204412054120641207412084120941210412114121241213412144121541216412174121841219412204122141222412234122441225412264122741228412294123041231412324123341234412354123641237412384123941240412414124241243412444124541246412474124841249412504125141252412534125441255412564125741258412594126041261412624126341264412654126641267412684126941270412714127241273412744127541276412774127841279412804128141282412834128441285412864128741288412894129041291412924129341294412954129641297412984129941300413014130241303413044130541306413074130841309413104131141312413134131441315413164131741318413194132041321413224132341324413254132641327413284132941330413314133241333413344133541336413374133841339413404134141342413434134441345413464134741348413494135041351413524135341354413554135641357413584135941360413614136241363413644136541366413674136841369413704137141372413734137441375413764137741378413794138041381413824138341384413854138641387413884138941390413914139241393413944139541396413974139841399414004140141402414034140441405414064140741408414094141041411414124141341414414154141641417414184141941420414214142241423414244142541426414274142841429414304143141432414334143441435414364143741438414394144041441414424144341444414454144641447414484144941450414514145241453414544145541456414574145841459414604146141462414634146441465414664146741468414694147041471414724147341474414754147641477414784147941480414814148241483414844148541486414874148841489414904149141492414934149441495414964149741498414994150041501415024150341504415054150641507415084150941510415114151241513415144151541516415174151841519415204152141522415234152441525415264152741528415294153041531415324153341534415354153641537415384153941540415414154241543415444154541546415474154841549415504155141552415534155441555415564155741558415594156041561415624156341564415654156641567415684156941570415714157241573415744157541576415774157841579415804158141582415834158441585415864158741588415894159041591415924159341594415954159641597415984159941600416014160241603416044160541606416074160841609416104161141612416134161441615416164161741618416194162041621416224162341624416254162641627416284162941630416314163241633416344163541636416374163841639416404164141642416434164441645416464164741648416494165041651416524165341654416554165641657416584165941660416614166241663416644166541666416674166841669416704167141672416734167441675416764167741678416794168041681416824168341684416854168641687416884168941690416914169241693416944169541696416974169841699417004170141702417034170441705417064170741708417094171041711417124171341714417154171641717417184171941720417214172241723417244172541726417274172841729417304173141732417334173441735417364173741738417394174041741417424174341744417454174641747417484174941750417514175241753417544175541756417574175841759417604176141762417634176441765417664176741768417694177041771417724177341774417754177641777417784177941780417814178241783417844178541786417874178841789417904179141792417934179441795417964179741798417994180041801418024180341804418054180641807418084180941810418114181241813418144181541816418174181841819418204182141822418234182441825418264182741828418294183041831418324183341834418354183641837418384183941840418414184241843418444184541846418474184841849418504185141852418534185441855418564185741858418594186041861418624186341864418654186641867418684186941870418714187241873418744187541876418774187841879418804188141882418834188441885418864188741888418894189041891418924189341894418954189641897418984189941900419014190241903419044190541906419074190841909419104191141912419134191441915419164191741918419194192041921419224192341924419254192641927419284192941930419314193241933419344193541936419374193841939419404194141942419434194441945419464194741948419494195041951419524195341954419554195641957419584195941960419614196241963419644196541966419674196841969419704197141972419734197441975419764197741978419794198041981419824198341984419854198641987419884198941990419914199241993419944199541996419974199841999420004200142002420034200442005420064200742008420094201042011420124201342014420154201642017420184201942020420214202242023420244202542026420274202842029420304203142032420334203442035420364203742038420394204042041420424204342044420454204642047420484204942050420514205242053420544205542056420574205842059420604206142062420634206442065420664206742068420694207042071420724207342074420754207642077420784207942080420814208242083420844208542086420874208842089420904209142092420934209442095420964209742098420994210042101421024210342104421054210642107421084210942110421114211242113421144211542116421174211842119421204212142122421234212442125421264212742128421294213042131421324213342134421354213642137421384213942140421414214242143421444214542146421474214842149421504215142152421534215442155421564215742158421594216042161421624216342164421654216642167421684216942170421714217242173421744217542176421774217842179421804218142182421834218442185421864218742188421894219042191421924219342194421954219642197421984219942200422014220242203422044220542206422074220842209422104221142212422134221442215422164221742218422194222042221422224222342224422254222642227422284222942230422314223242233422344223542236422374223842239422404224142242422434224442245422464224742248422494225042251422524225342254422554225642257422584225942260422614226242263422644226542266422674226842269422704227142272422734227442275422764227742278422794228042281422824228342284422854228642287422884228942290422914229242293422944229542296422974229842299423004230142302423034230442305423064230742308423094231042311423124231342314423154231642317423184231942320423214232242323423244232542326423274232842329423304233142332423334233442335423364233742338423394234042341423424234342344423454234642347423484234942350423514235242353423544235542356423574235842359423604236142362423634236442365423664236742368423694237042371423724237342374423754237642377423784237942380423814238242383423844238542386423874238842389423904239142392423934239442395423964239742398423994240042401424024240342404424054240642407424084240942410424114241242413424144241542416424174241842419424204242142422424234242442425424264242742428424294243042431424324243342434424354243642437424384243942440424414244242443424444244542446424474244842449424504245142452424534245442455424564245742458424594246042461424624246342464424654246642467424684246942470424714247242473424744247542476424774247842479424804248142482424834248442485424864248742488424894249042491424924249342494424954249642497424984249942500425014250242503425044250542506425074250842509425104251142512425134251442515425164251742518425194252042521425224252342524425254252642527425284252942530425314253242533425344253542536425374253842539425404254142542425434254442545425464254742548425494255042551425524255342554425554255642557425584255942560425614256242563425644256542566425674256842569425704257142572425734257442575425764257742578425794258042581425824258342584425854258642587425884258942590425914259242593425944259542596425974259842599426004260142602426034260442605426064260742608426094261042611426124261342614426154261642617426184261942620426214262242623426244262542626426274262842629426304263142632426334263442635426364263742638426394264042641426424264342644426454264642647426484264942650426514265242653426544265542656426574265842659426604266142662426634266442665426664266742668426694267042671426724267342674426754267642677426784267942680426814268242683426844268542686426874268842689426904269142692426934269442695426964269742698426994270042701427024270342704427054270642707427084270942710427114271242713427144271542716427174271842719427204272142722427234272442725427264272742728427294273042731427324273342734427354273642737427384273942740427414274242743427444274542746427474274842749427504275142752427534275442755427564275742758427594276042761427624276342764427654276642767427684276942770427714277242773427744277542776427774277842779427804278142782427834278442785427864278742788427894279042791427924279342794427954279642797427984279942800428014280242803428044280542806428074280842809428104281142812428134281442815428164281742818428194282042821428224282342824428254282642827428284282942830428314283242833428344283542836428374283842839428404284142842428434284442845428464284742848428494285042851428524285342854428554285642857428584285942860428614286242863428644286542866428674286842869428704287142872428734287442875428764287742878428794288042881428824288342884428854288642887428884288942890428914289242893428944289542896428974289842899429004290142902429034290442905429064290742908429094291042911429124291342914429154291642917429184291942920429214292242923429244292542926429274292842929429304293142932429334293442935429364293742938429394294042941429424294342944429454294642947429484294942950429514295242953429544295542956429574295842959429604296142962429634296442965429664296742968429694297042971429724297342974429754297642977429784297942980429814298242983429844298542986429874298842989429904299142992429934299442995429964299742998429994300043001430024300343004430054300643007430084300943010430114301243013430144301543016430174301843019430204302143022430234302443025430264302743028430294303043031430324303343034430354303643037430384303943040430414304243043430444304543046430474304843049430504305143052430534305443055430564305743058430594306043061430624306343064430654306643067430684306943070430714307243073430744307543076430774307843079430804308143082430834308443085430864308743088430894309043091430924309343094430954309643097430984309943100431014310243103431044310543106431074310843109431104311143112431134311443115431164311743118431194312043121431224312343124431254312643127431284312943130431314313243133431344313543136431374313843139431404314143142431434314443145431464314743148431494315043151431524315343154431554315643157431584315943160431614316243163431644316543166431674316843169431704317143172431734317443175431764317743178431794318043181431824318343184431854318643187431884318943190431914319243193431944319543196431974319843199432004320143202432034320443205432064320743208432094321043211432124321343214432154321643217432184321943220432214322243223432244322543226432274322843229432304323143232432334323443235432364323743238432394324043241432424324343244432454324643247432484324943250432514325243253432544325543256432574325843259432604326143262432634326443265432664326743268432694327043271432724327343274432754327643277432784327943280432814328243283432844328543286432874328843289432904329143292432934329443295432964329743298432994330043301433024330343304433054330643307433084330943310433114331243313433144331543316433174331843319433204332143322433234332443325433264332743328433294333043331433324333343334433354333643337433384333943340433414334243343433444334543346433474334843349433504335143352433534335443355433564335743358433594336043361433624336343364433654336643367433684336943370433714337243373433744337543376433774337843379433804338143382433834338443385433864338743388433894339043391433924339343394433954339643397433984339943400434014340243403434044340543406434074340843409434104341143412434134341443415434164341743418434194342043421434224342343424434254342643427434284342943430434314343243433434344343543436434374343843439434404344143442434434344443445434464344743448434494345043451434524345343454434554345643457434584345943460434614346243463434644346543466434674346843469434704347143472434734347443475434764347743478434794348043481434824348343484434854348643487434884348943490434914349243493434944349543496434974349843499435004350143502435034350443505435064350743508435094351043511435124351343514435154351643517435184351943520435214352243523435244352543526435274352843529435304353143532435334353443535435364353743538435394354043541435424354343544435454354643547435484354943550435514355243553435544355543556435574355843559435604356143562435634356443565435664356743568435694357043571435724357343574435754357643577435784357943580435814358243583435844358543586435874358843589435904359143592435934359443595435964359743598435994360043601436024360343604436054360643607436084360943610436114361243613436144361543616436174361843619436204362143622436234362443625436264362743628436294363043631436324363343634436354363643637436384363943640436414364243643436444364543646436474364843649436504365143652436534365443655436564365743658436594366043661436624366343664436654366643667436684366943670436714367243673436744367543676436774367843679436804368143682436834368443685436864368743688436894369043691436924369343694436954369643697436984369943700437014370243703437044370543706437074370843709437104371143712437134371443715437164371743718437194372043721437224372343724437254372643727437284372943730437314373243733437344373543736437374373843739437404374143742437434374443745437464374743748437494375043751437524375343754437554375643757437584375943760437614376243763437644376543766437674376843769437704377143772437734377443775437764377743778437794378043781437824378343784437854378643787437884378943790437914379243793437944379543796437974379843799438004380143802438034380443805438064380743808438094381043811438124381343814438154381643817438184381943820438214382243823438244382543826438274382843829438304383143832438334383443835438364383743838438394384043841438424384343844438454384643847438484384943850438514385243853438544385543856438574385843859438604386143862438634386443865438664386743868438694387043871438724387343874438754387643877438784387943880438814388243883438844388543886438874388843889438904389143892438934389443895438964389743898438994390043901439024390343904439054390643907439084390943910439114391243913439144391543916439174391843919439204392143922439234392443925439264392743928439294393043931439324393343934439354393643937439384393943940439414394243943439444394543946439474394843949439504395143952439534395443955439564395743958439594396043961439624396343964439654396643967439684396943970439714397243973439744397543976439774397843979439804398143982439834398443985439864398743988439894399043991439924399343994439954399643997439984399944000440014400244003440044400544006440074400844009440104401144012440134401444015440164401744018440194402044021440224402344024440254402644027440284402944030440314403244033440344403544036440374403844039440404404144042440434404444045440464404744048440494405044051440524405344054440554405644057440584405944060440614406244063440644406544066440674406844069440704407144072440734407444075440764407744078440794408044081440824408344084440854408644087440884408944090440914409244093440944409544096440974409844099441004410144102441034410444105441064410744108441094411044111441124411344114441154411644117441184411944120441214412244123441244412544126441274412844129441304413144132441334413444135441364413744138441394414044141441424414344144441454414644147441484414944150441514415244153441544415544156441574415844159441604416144162441634416444165441664416744168441694417044171441724417344174441754417644177441784417944180441814418244183441844418544186441874418844189441904419144192441934419444195441964419744198441994420044201442024420344204442054420644207442084420944210442114421244213442144421544216442174421844219442204422144222442234422444225442264422744228442294423044231442324423344234442354423644237442384423944240442414424244243442444424544246442474424844249442504425144252442534425444255442564425744258442594426044261442624426344264442654426644267442684426944270442714427244273442744427544276442774427844279442804428144282442834428444285442864428744288442894429044291442924429344294442954429644297442984429944300443014430244303443044430544306443074430844309443104431144312443134431444315443164431744318443194432044321443224432344324443254432644327443284432944330443314433244333443344433544336443374433844339443404434144342443434434444345443464434744348443494435044351443524435344354443554435644357443584435944360443614436244363443644436544366443674436844369443704437144372443734437444375443764437744378443794438044381443824438344384443854438644387443884438944390443914439244393443944439544396443974439844399444004440144402444034440444405444064440744408444094441044411444124441344414444154441644417444184441944420444214442244423444244442544426444274442844429444304443144432444334443444435444364443744438444394444044441444424444344444444454444644447444484444944450444514445244453444544445544456444574445844459444604446144462444634446444465444664446744468444694447044471444724447344474444754447644477444784447944480444814448244483444844448544486444874448844489444904449144492444934449444495444964449744498444994450044501445024450344504445054450644507445084450944510445114451244513445144451544516445174451844519445204452144522445234452444525445264452744528445294453044531445324453344534445354453644537445384453944540445414454244543445444454544546445474454844549445504455144552445534455444555445564455744558445594456044561445624456344564445654456644567445684456944570445714457244573445744457544576445774457844579445804458144582445834458444585445864458744588445894459044591445924459344594445954459644597445984459944600446014460244603446044460544606446074460844609446104461144612446134461444615446164461744618446194462044621446224462344624446254462644627446284462944630446314463244633446344463544636446374463844639446404464144642446434464444645446464464744648446494465044651446524465344654446554465644657446584465944660446614466244663446644466544666446674466844669446704467144672446734467444675446764467744678446794468044681446824468344684446854468644687446884468944690446914469244693446944469544696446974469844699447004470144702447034470444705447064470744708447094471044711447124471344714447154471644717447184471944720447214472244723447244472544726447274472844729447304473144732447334473444735447364473744738447394474044741447424474344744447454474644747447484474944750447514475244753447544475544756447574475844759447604476144762447634476444765447664476744768447694477044771447724477344774447754477644777447784477944780447814478244783447844478544786447874478844789447904479144792447934479444795447964479744798447994480044801448024480344804448054480644807448084480944810448114481244813448144481544816448174481844819448204482144822448234482444825448264482744828448294483044831448324483344834448354483644837448384483944840448414484244843448444484544846448474484844849448504485144852448534485444855448564485744858448594486044861448624486344864448654486644867448684486944870448714487244873448744487544876448774487844879448804488144882448834488444885448864488744888448894489044891448924489344894448954489644897448984489944900449014490244903449044490544906449074490844909449104491144912449134491444915449164491744918449194492044921449224492344924449254492644927449284492944930449314493244933449344493544936449374493844939449404494144942449434494444945449464494744948449494495044951449524495344954449554495644957449584495944960449614496244963449644496544966449674496844969449704497144972449734497444975449764497744978449794498044981449824498344984449854498644987449884498944990449914499244993449944499544996449974499844999450004500145002450034500445005450064500745008450094501045011450124501345014450154501645017450184501945020450214502245023450244502545026450274502845029450304503145032450334503445035450364503745038450394504045041450424504345044450454504645047450484504945050450514505245053450544505545056450574505845059450604506145062450634506445065450664506745068450694507045071450724507345074450754507645077450784507945080450814508245083450844508545086450874508845089450904509145092450934509445095450964509745098450994510045101451024510345104451054510645107451084510945110451114511245113451144511545116451174511845119451204512145122451234512445125451264512745128451294513045131451324513345134451354513645137451384513945140451414514245143451444514545146451474514845149451504515145152451534515445155451564515745158451594516045161451624516345164451654516645167451684516945170451714517245173451744517545176451774517845179451804518145182451834518445185451864518745188451894519045191451924519345194451954519645197451984519945200452014520245203452044520545206452074520845209452104521145212452134521445215452164521745218452194522045221452224522345224452254522645227452284522945230452314523245233452344523545236452374523845239452404524145242452434524445245452464524745248452494525045251452524525345254452554525645257452584525945260452614526245263452644526545266452674526845269452704527145272452734527445275452764527745278452794528045281452824528345284452854528645287452884528945290452914529245293452944529545296452974529845299453004530145302453034530445305453064530745308453094531045311453124531345314453154531645317453184531945320453214532245323453244532545326453274532845329453304533145332453334533445335453364533745338453394534045341453424534345344453454534645347453484534945350453514535245353453544535545356453574535845359453604536145362453634536445365453664536745368453694537045371453724537345374453754537645377453784537945380453814538245383453844538545386453874538845389453904539145392453934539445395453964539745398453994540045401454024540345404454054540645407454084540945410454114541245413454144541545416454174541845419454204542145422454234542445425454264542745428454294543045431454324543345434454354543645437454384543945440454414544245443454444544545446454474544845449454504545145452454534545445455454564545745458454594546045461454624546345464454654546645467454684546945470454714547245473454744547545476454774547845479454804548145482454834548445485454864548745488454894549045491454924549345494454954549645497454984549945500455014550245503455044550545506455074550845509455104551145512455134551445515455164551745518455194552045521455224552345524455254552645527455284552945530455314553245533455344553545536455374553845539455404554145542455434554445545455464554745548455494555045551455524555345554455554555645557455584555945560455614556245563455644556545566455674556845569455704557145572455734557445575455764557745578455794558045581455824558345584455854558645587455884558945590455914559245593455944559545596455974559845599456004560145602456034560445605456064560745608456094561045611456124561345614456154561645617456184561945620456214562245623456244562545626456274562845629456304563145632456334563445635456364563745638456394564045641456424564345644456454564645647456484564945650456514565245653456544565545656456574565845659456604566145662456634566445665456664566745668456694567045671456724567345674456754567645677456784567945680456814568245683456844568545686456874568845689456904569145692456934569445695456964569745698456994570045701457024570345704457054570645707457084570945710457114571245713457144571545716457174571845719457204572145722457234572445725457264572745728457294573045731457324573345734457354573645737457384573945740457414574245743457444574545746457474574845749457504575145752457534575445755457564575745758457594576045761457624576345764457654576645767457684576945770457714577245773457744577545776457774577845779457804578145782457834578445785457864578745788457894579045791457924579345794457954579645797457984579945800458014580245803458044580545806458074580845809458104581145812458134581445815458164581745818458194582045821458224582345824458254582645827458284582945830458314583245833458344583545836458374583845839458404584145842458434584445845458464584745848458494585045851458524585345854458554585645857458584585945860458614586245863458644586545866458674586845869458704587145872458734587445875458764587745878458794588045881458824588345884458854588645887458884588945890458914589245893458944589545896458974589845899459004590145902459034590445905459064590745908459094591045911459124591345914459154591645917459184591945920459214592245923459244592545926459274592845929459304593145932459334593445935459364593745938459394594045941459424594345944459454594645947459484594945950459514595245953459544595545956459574595845959459604596145962459634596445965459664596745968459694597045971459724597345974459754597645977459784597945980459814598245983459844598545986459874598845989459904599145992459934599445995459964599745998459994600046001460024600346004460054600646007460084600946010460114601246013460144601546016460174601846019460204602146022460234602446025460264602746028460294603046031460324603346034460354603646037460384603946040460414604246043460444604546046460474604846049460504605146052460534605446055460564605746058460594606046061460624606346064460654606646067460684606946070460714607246073460744607546076460774607846079460804608146082460834608446085460864608746088460894609046091460924609346094460954609646097460984609946100461014610246103461044610546106461074610846109461104611146112461134611446115461164611746118461194612046121461224612346124461254612646127461284612946130461314613246133461344613546136461374613846139461404614146142461434614446145461464614746148461494615046151461524615346154461554615646157461584615946160461614616246163461644616546166461674616846169461704617146172461734617446175461764617746178461794618046181461824618346184461854618646187461884618946190461914619246193461944619546196461974619846199462004620146202462034620446205462064620746208462094621046211462124621346214462154621646217462184621946220462214622246223462244622546226462274622846229462304623146232462334623446235462364623746238462394624046241462424624346244462454624646247462484624946250462514625246253462544625546256462574625846259462604626146262462634626446265462664626746268462694627046271462724627346274462754627646277462784627946280462814628246283462844628546286462874628846289462904629146292462934629446295462964629746298462994630046301463024630346304463054630646307463084630946310463114631246313463144631546316463174631846319463204632146322463234632446325463264632746328463294633046331463324633346334463354633646337463384633946340463414634246343463444634546346463474634846349463504635146352463534635446355463564635746358463594636046361463624636346364463654636646367463684636946370463714637246373463744637546376463774637846379463804638146382463834638446385463864638746388463894639046391463924639346394463954639646397463984639946400464014640246403464044640546406464074640846409464104641146412464134641446415464164641746418464194642046421464224642346424464254642646427464284642946430464314643246433464344643546436464374643846439464404644146442464434644446445464464644746448464494645046451464524645346454464554645646457464584645946460464614646246463464644646546466464674646846469464704647146472464734647446475464764647746478464794648046481464824648346484464854648646487464884648946490464914649246493464944649546496464974649846499465004650146502465034650446505465064650746508465094651046511465124651346514465154651646517465184651946520465214652246523465244652546526465274652846529465304653146532465334653446535465364653746538465394654046541465424654346544465454654646547465484654946550465514655246553465544655546556465574655846559465604656146562465634656446565465664656746568465694657046571465724657346574465754657646577465784657946580465814658246583465844658546586465874658846589465904659146592465934659446595465964659746598465994660046601466024660346604466054660646607466084660946610466114661246613466144661546616466174661846619466204662146622466234662446625466264662746628466294663046631466324663346634466354663646637466384663946640466414664246643466444664546646466474664846649466504665146652466534665446655466564665746658466594666046661466624666346664466654666646667466684666946670466714667246673466744667546676466774667846679466804668146682466834668446685466864668746688466894669046691466924669346694466954669646697466984669946700467014670246703467044670546706467074670846709467104671146712467134671446715467164671746718467194672046721467224672346724467254672646727467284672946730467314673246733467344673546736467374673846739467404674146742467434674446745467464674746748467494675046751467524675346754467554675646757467584675946760467614676246763467644676546766467674676846769467704677146772467734677446775467764677746778467794678046781467824678346784467854678646787467884678946790467914679246793467944679546796467974679846799468004680146802468034680446805468064680746808468094681046811468124681346814468154681646817468184681946820468214682246823468244682546826468274682846829468304683146832468334683446835468364683746838468394684046841468424684346844468454684646847468484684946850468514685246853468544685546856468574685846859468604686146862468634686446865468664686746868468694687046871468724687346874468754687646877468784687946880468814688246883468844688546886468874688846889468904689146892468934689446895468964689746898468994690046901469024690346904469054690646907469084690946910469114691246913469144691546916469174691846919469204692146922469234692446925469264692746928469294693046931469324693346934469354693646937469384693946940469414694246943469444694546946469474694846949469504695146952469534695446955469564695746958469594696046961469624696346964469654696646967469684696946970469714697246973469744697546976469774697846979469804698146982469834698446985469864698746988469894699046991469924699346994469954699646997469984699947000470014700247003470044700547006470074700847009470104701147012470134701447015470164701747018470194702047021470224702347024470254702647027470284702947030470314703247033470344703547036470374703847039470404704147042470434704447045470464704747048470494705047051470524705347054470554705647057470584705947060470614706247063470644706547066470674706847069470704707147072470734707447075470764707747078470794708047081470824708347084470854708647087470884708947090470914709247093470944709547096470974709847099471004710147102471034710447105471064710747108471094711047111471124711347114471154711647117471184711947120471214712247123471244712547126471274712847129471304713147132471334713447135471364713747138471394714047141471424714347144471454714647147471484714947150471514715247153471544715547156471574715847159471604716147162471634716447165471664716747168471694717047171471724717347174471754717647177471784717947180471814718247183471844718547186471874718847189471904719147192471934719447195471964719747198471994720047201472024720347204472054720647207472084720947210472114721247213472144721547216472174721847219472204722147222472234722447225472264722747228472294723047231472324723347234472354723647237472384723947240472414724247243472444724547246472474724847249472504725147252472534725447255472564725747258472594726047261472624726347264472654726647267472684726947270472714727247273472744727547276472774727847279472804728147282472834728447285472864728747288472894729047291472924729347294472954729647297472984729947300473014730247303473044730547306473074730847309473104731147312473134731447315473164731747318473194732047321473224732347324473254732647327473284732947330473314733247333473344733547336473374733847339473404734147342473434734447345473464734747348473494735047351473524735347354473554735647357473584735947360473614736247363473644736547366473674736847369473704737147372473734737447375473764737747378473794738047381473824738347384473854738647387473884738947390473914739247393473944739547396473974739847399474004740147402474034740447405474064740747408474094741047411474124741347414474154741647417474184741947420474214742247423474244742547426474274742847429474304743147432474334743447435474364743747438474394744047441474424744347444474454744647447474484744947450474514745247453474544745547456474574745847459474604746147462474634746447465474664746747468474694747047471474724747347474474754747647477474784747947480474814748247483474844748547486474874748847489474904749147492474934749447495474964749747498474994750047501475024750347504475054750647507475084750947510475114751247513475144751547516475174751847519475204752147522475234752447525475264752747528475294753047531475324753347534475354753647537475384753947540475414754247543475444754547546475474754847549475504755147552475534755447555475564755747558475594756047561475624756347564475654756647567475684756947570475714757247573475744757547576475774757847579475804758147582475834758447585475864758747588475894759047591475924759347594475954759647597475984759947600476014760247603476044760547606476074760847609476104761147612476134761447615476164761747618476194762047621476224762347624476254762647627476284762947630476314763247633476344763547636476374763847639476404764147642476434764447645476464764747648476494765047651476524765347654476554765647657476584765947660476614766247663476644766547666476674766847669476704767147672476734767447675476764767747678476794768047681476824768347684476854768647687476884768947690476914769247693476944769547696476974769847699477004770147702477034770447705477064770747708477094771047711477124771347714477154771647717477184771947720477214772247723477244772547726477274772847729477304773147732477334773447735477364773747738477394774047741477424774347744477454774647747477484774947750477514775247753477544775547756477574775847759477604776147762477634776447765477664776747768477694777047771477724777347774477754777647777477784777947780477814778247783477844778547786477874778847789477904779147792477934779447795477964779747798477994780047801478024780347804478054780647807478084780947810478114781247813478144781547816478174781847819478204782147822478234782447825478264782747828478294783047831478324783347834478354783647837478384783947840478414784247843478444784547846478474784847849478504785147852478534785447855478564785747858478594786047861478624786347864478654786647867478684786947870478714787247873478744787547876478774787847879478804788147882478834788447885478864788747888478894789047891478924789347894478954789647897478984789947900479014790247903479044790547906479074790847909479104791147912479134791447915479164791747918479194792047921479224792347924479254792647927479284792947930479314793247933479344793547936479374793847939479404794147942479434794447945479464794747948479494795047951479524795347954479554795647957479584795947960479614796247963479644796547966479674796847969479704797147972479734797447975479764797747978479794798047981479824798347984479854798647987479884798947990479914799247993479944799547996479974799847999480004800148002480034800448005480064800748008480094801048011480124801348014480154801648017480184801948020480214802248023480244802548026480274802848029480304803148032480334803448035480364803748038480394804048041480424804348044480454804648047480484804948050480514805248053480544805548056480574805848059480604806148062480634806448065480664806748068480694807048071480724807348074480754807648077480784807948080480814808248083480844808548086480874808848089480904809148092480934809448095480964809748098480994810048101481024810348104481054810648107481084810948110481114811248113481144811548116481174811848119481204812148122481234812448125481264812748128481294813048131481324813348134481354813648137481384813948140481414814248143481444814548146481474814848149481504815148152481534815448155481564815748158481594816048161481624816348164481654816648167481684816948170481714817248173481744817548176481774817848179481804818148182481834818448185481864818748188481894819048191481924819348194481954819648197481984819948200482014820248203482044820548206482074820848209482104821148212482134821448215482164821748218482194822048221482224822348224482254822648227482284822948230482314823248233482344823548236482374823848239482404824148242482434824448245482464824748248482494825048251482524825348254482554825648257482584825948260482614826248263482644826548266482674826848269482704827148272482734827448275482764827748278482794828048281482824828348284482854828648287482884828948290482914829248293482944829548296482974829848299483004830148302483034830448305483064830748308483094831048311483124831348314483154831648317483184831948320483214832248323483244832548326483274832848329483304833148332483334833448335483364833748338483394834048341483424834348344483454834648347483484834948350483514835248353483544835548356483574835848359483604836148362483634836448365483664836748368483694837048371483724837348374483754837648377483784837948380483814838248383483844838548386483874838848389483904839148392483934839448395483964839748398483994840048401484024840348404484054840648407484084840948410484114841248413484144841548416484174841848419484204842148422484234842448425484264842748428484294843048431484324843348434484354843648437484384843948440484414844248443484444844548446484474844848449484504845148452484534845448455484564845748458484594846048461484624846348464484654846648467484684846948470484714847248473484744847548476484774847848479484804848148482484834848448485484864848748488484894849048491484924849348494484954849648497484984849948500485014850248503485044850548506485074850848509485104851148512485134851448515485164851748518485194852048521485224852348524485254852648527485284852948530485314853248533485344853548536485374853848539485404854148542485434854448545485464854748548485494855048551485524855348554485554855648557485584855948560485614856248563485644856548566485674856848569485704857148572485734857448575485764857748578485794858048581485824858348584485854858648587485884858948590485914859248593485944859548596485974859848599486004860148602486034860448605486064860748608486094861048611486124861348614486154861648617486184861948620486214862248623486244862548626486274862848629486304863148632486334863448635486364863748638486394864048641486424864348644486454864648647486484864948650486514865248653486544865548656486574865848659486604866148662486634866448665486664866748668486694867048671486724867348674486754867648677486784867948680486814868248683486844868548686486874868848689486904869148692486934869448695486964869748698486994870048701487024870348704487054870648707487084870948710487114871248713487144871548716487174871848719487204872148722487234872448725487264872748728487294873048731487324873348734487354873648737487384873948740487414874248743487444874548746487474874848749487504875148752487534875448755487564875748758487594876048761487624876348764487654876648767487684876948770487714877248773487744877548776487774877848779487804878148782487834878448785487864878748788487894879048791487924879348794487954879648797487984879948800488014880248803488044880548806488074880848809488104881148812488134881448815488164881748818488194882048821488224882348824488254882648827488284882948830488314883248833488344883548836488374883848839488404884148842488434884448845488464884748848488494885048851488524885348854488554885648857488584885948860488614886248863488644886548866488674886848869488704887148872488734887448875488764887748878488794888048881488824888348884488854888648887488884888948890488914889248893488944889548896488974889848899489004890148902489034890448905489064890748908489094891048911489124891348914489154891648917489184891948920489214892248923489244892548926489274892848929489304893148932489334893448935489364893748938489394894048941489424894348944489454894648947489484894948950489514895248953489544895548956489574895848959489604896148962489634896448965489664896748968489694897048971489724897348974489754897648977489784897948980489814898248983489844898548986489874898848989489904899148992489934899448995489964899748998489994900049001490024900349004490054900649007490084900949010490114901249013490144901549016490174901849019490204902149022490234902449025490264902749028490294903049031490324903349034490354903649037490384903949040490414904249043490444904549046490474904849049490504905149052490534905449055490564905749058490594906049061490624906349064490654906649067490684906949070490714907249073490744907549076490774907849079490804908149082490834908449085490864908749088490894909049091490924909349094490954909649097490984909949100491014910249103491044910549106491074910849109491104911149112491134911449115491164911749118491194912049121491224912349124491254912649127491284912949130491314913249133491344913549136491374913849139491404914149142491434914449145491464914749148491494915049151491524915349154491554915649157491584915949160491614916249163491644916549166491674916849169491704917149172491734917449175491764917749178491794918049181491824918349184491854918649187491884918949190491914919249193491944919549196491974919849199492004920149202492034920449205492064920749208492094921049211492124921349214492154921649217492184921949220492214922249223492244922549226492274922849229492304923149232492334923449235492364923749238492394924049241492424924349244492454924649247492484924949250492514925249253492544925549256492574925849259492604926149262492634926449265492664926749268492694927049271492724927349274492754927649277492784927949280492814928249283492844928549286492874928849289492904929149292492934929449295492964929749298492994930049301493024930349304493054930649307493084930949310493114931249313493144931549316493174931849319493204932149322493234932449325493264932749328493294933049331493324933349334493354933649337493384933949340493414934249343493444934549346493474934849349493504935149352493534935449355493564935749358493594936049361493624936349364493654936649367493684936949370493714937249373493744937549376493774937849379493804938149382493834938449385493864938749388493894939049391493924939349394493954939649397493984939949400494014940249403494044940549406494074940849409494104941149412494134941449415494164941749418494194942049421494224942349424494254942649427494284942949430494314943249433494344943549436494374943849439494404944149442494434944449445494464944749448494494945049451494524945349454494554945649457494584945949460494614946249463494644946549466494674946849469494704947149472494734947449475494764947749478494794948049481494824948349484494854948649487494884948949490494914949249493494944949549496494974949849499495004950149502495034950449505495064950749508495094951049511495124951349514495154951649517495184951949520495214952249523495244952549526495274952849529495304953149532495334953449535495364953749538495394954049541495424954349544495454954649547495484954949550495514955249553495544955549556495574955849559495604956149562495634956449565495664956749568495694957049571495724957349574495754957649577495784957949580495814958249583495844958549586495874958849589495904959149592495934959449595495964959749598495994960049601496024960349604496054960649607496084960949610496114961249613496144961549616496174961849619496204962149622496234962449625496264962749628496294963049631496324963349634496354963649637496384963949640496414964249643496444964549646496474964849649496504965149652496534965449655496564965749658496594966049661496624966349664496654966649667496684966949670496714967249673496744967549676496774967849679496804968149682496834968449685496864968749688496894969049691496924969349694496954969649697496984969949700497014970249703497044970549706497074970849709497104971149712497134971449715497164971749718497194972049721497224972349724497254972649727497284972949730497314973249733497344973549736497374973849739497404974149742497434974449745497464974749748497494975049751497524975349754497554975649757497584975949760497614976249763497644976549766497674976849769497704977149772497734977449775497764977749778497794978049781497824978349784497854978649787497884978949790497914979249793497944979549796497974979849799498004980149802498034980449805498064980749808498094981049811498124981349814498154981649817498184981949820498214982249823498244982549826498274982849829498304983149832498334983449835498364983749838498394984049841498424984349844498454984649847498484984949850498514985249853498544985549856498574985849859498604986149862498634986449865498664986749868498694987049871498724987349874498754987649877498784987949880498814988249883498844988549886498874988849889498904989149892498934989449895498964989749898498994990049901499024990349904499054990649907499084990949910499114991249913499144991549916499174991849919499204992149922499234992449925499264992749928499294993049931499324993349934499354993649937499384993949940499414994249943499444994549946499474994849949499504995149952499534995449955499564995749958499594996049961499624996349964499654996649967499684996949970499714997249973499744997549976499774997849979499804998149982499834998449985499864998749988499894999049991499924999349994499954999649997499984999950000500015000250003500045000550006500075000850009500105001150012500135001450015500165001750018500195002050021500225002350024500255002650027500285002950030500315003250033500345003550036500375003850039500405004150042500435004450045500465004750048500495005050051500525005350054500555005650057500585005950060500615006250063500645006550066500675006850069500705007150072500735007450075500765007750078500795008050081500825008350084500855008650087500885008950090500915009250093500945009550096500975009850099501005010150102501035010450105501065010750108501095011050111501125011350114501155011650117501185011950120501215012250123501245012550126501275012850129501305013150132501335013450135501365013750138501395014050141501425014350144501455014650147501485014950150501515015250153501545015550156501575015850159501605016150162501635016450165501665016750168501695017050171501725017350174501755017650177501785017950180501815018250183501845018550186501875018850189501905019150192501935019450195501965019750198501995020050201502025020350204502055020650207502085020950210502115021250213502145021550216502175021850219502205022150222502235022450225502265022750228502295023050231502325023350234502355023650237502385023950240502415024250243502445024550246502475024850249502505025150252502535025450255502565025750258502595026050261502625026350264502655026650267502685026950270502715027250273502745027550276502775027850279502805028150282502835028450285502865028750288502895029050291502925029350294502955029650297502985029950300503015030250303503045030550306503075030850309503105031150312503135031450315503165031750318503195032050321503225032350324503255032650327503285032950330503315033250333503345033550336503375033850339503405034150342503435034450345503465034750348503495035050351503525035350354503555035650357503585035950360503615036250363503645036550366503675036850369503705037150372503735037450375503765037750378503795038050381503825038350384503855038650387503885038950390503915039250393503945039550396503975039850399504005040150402504035040450405504065040750408504095041050411504125041350414504155041650417504185041950420504215042250423504245042550426504275042850429504305043150432504335043450435504365043750438504395044050441504425044350444504455044650447504485044950450504515045250453504545045550456504575045850459504605046150462504635046450465504665046750468504695047050471504725047350474504755047650477504785047950480504815048250483504845048550486504875048850489504905049150492504935049450495504965049750498504995050050501505025050350504505055050650507505085050950510505115051250513505145051550516505175051850519505205052150522505235052450525505265052750528505295053050531505325053350534505355053650537505385053950540505415054250543505445054550546505475054850549505505055150552505535055450555505565055750558505595056050561505625056350564505655056650567505685056950570505715057250573505745057550576505775057850579505805058150582505835058450585505865058750588505895059050591505925059350594505955059650597505985059950600506015060250603506045060550606506075060850609506105061150612506135061450615506165061750618506195062050621506225062350624506255062650627506285062950630506315063250633506345063550636506375063850639506405064150642506435064450645506465064750648506495065050651506525065350654506555065650657506585065950660506615066250663506645066550666506675066850669506705067150672506735067450675506765067750678506795068050681506825068350684506855068650687506885068950690506915069250693506945069550696506975069850699507005070150702507035070450705507065070750708507095071050711507125071350714507155071650717507185071950720507215072250723507245072550726507275072850729507305073150732507335073450735507365073750738507395074050741507425074350744507455074650747507485074950750507515075250753507545075550756507575075850759507605076150762507635076450765507665076750768507695077050771507725077350774507755077650777507785077950780507815078250783507845078550786507875078850789507905079150792507935079450795507965079750798507995080050801508025080350804508055080650807508085080950810508115081250813508145081550816508175081850819508205082150822508235082450825508265082750828508295083050831508325083350834508355083650837508385083950840508415084250843508445084550846508475084850849508505085150852508535085450855508565085750858508595086050861508625086350864508655086650867508685086950870508715087250873508745087550876508775087850879508805088150882508835088450885508865088750888508895089050891508925089350894508955089650897508985089950900509015090250903509045090550906509075090850909509105091150912509135091450915509165091750918509195092050921509225092350924509255092650927509285092950930509315093250933509345093550936509375093850939509405094150942509435094450945509465094750948509495095050951509525095350954509555095650957509585095950960509615096250963509645096550966509675096850969509705097150972509735097450975509765097750978509795098050981509825098350984509855098650987509885098950990509915099250993509945099550996509975099850999510005100151002510035100451005510065100751008510095101051011510125101351014510155101651017510185101951020510215102251023510245102551026510275102851029510305103151032510335103451035510365103751038510395104051041510425104351044510455104651047510485104951050510515105251053510545105551056510575105851059510605106151062510635106451065510665106751068510695107051071510725107351074510755107651077510785107951080510815108251083510845108551086510875108851089510905109151092510935109451095510965109751098510995110051101511025110351104511055110651107511085110951110511115111251113511145111551116511175111851119511205112151122511235112451125511265112751128511295113051131511325113351134511355113651137511385113951140511415114251143511445114551146511475114851149511505115151152511535115451155511565115751158511595116051161511625116351164511655116651167511685116951170511715117251173511745117551176511775117851179511805118151182511835118451185511865118751188511895119051191511925119351194511955119651197511985119951200512015120251203512045120551206512075120851209512105121151212512135121451215512165121751218512195122051221512225122351224512255122651227512285122951230512315123251233512345123551236512375123851239512405124151242512435124451245512465124751248512495125051251512525125351254512555125651257512585125951260512615126251263512645126551266512675126851269512705127151272512735127451275512765127751278512795128051281512825128351284512855128651287512885128951290512915129251293512945129551296512975129851299513005130151302513035130451305513065130751308513095131051311513125131351314513155131651317513185131951320513215132251323513245132551326513275132851329513305133151332513335133451335513365133751338513395134051341513425134351344513455134651347513485134951350513515135251353513545135551356513575135851359513605136151362513635136451365513665136751368513695137051371513725137351374513755137651377513785137951380513815138251383513845138551386513875138851389513905139151392513935139451395513965139751398513995140051401514025140351404514055140651407514085140951410514115141251413514145141551416514175141851419514205142151422514235142451425514265142751428514295143051431514325143351434514355143651437514385143951440514415144251443514445144551446514475144851449514505145151452514535145451455514565145751458514595146051461514625146351464514655146651467514685146951470514715147251473514745147551476514775147851479514805148151482514835148451485514865148751488514895149051491514925149351494514955149651497514985149951500515015150251503515045150551506515075150851509515105151151512515135151451515515165151751518515195152051521515225152351524515255152651527515285152951530515315153251533515345153551536515375153851539515405154151542515435154451545515465154751548515495155051551515525155351554515555155651557515585155951560515615156251563515645156551566515675156851569515705157151572515735157451575515765157751578515795158051581515825158351584515855158651587515885158951590515915159251593515945159551596515975159851599516005160151602516035160451605516065160751608516095161051611516125161351614516155161651617516185161951620516215162251623516245162551626516275162851629516305163151632516335163451635516365163751638516395164051641516425164351644516455164651647516485164951650516515165251653516545165551656516575165851659516605166151662516635166451665516665166751668516695167051671516725167351674516755167651677516785167951680516815168251683516845168551686516875168851689516905169151692516935169451695516965169751698516995170051701517025170351704517055170651707517085170951710517115171251713517145171551716517175171851719517205172151722517235172451725517265172751728517295173051731517325173351734517355173651737517385173951740517415174251743517445174551746517475174851749517505175151752517535175451755517565175751758517595176051761517625176351764517655176651767517685176951770517715177251773517745177551776517775177851779517805178151782517835178451785517865178751788517895179051791517925179351794517955179651797517985179951800518015180251803518045180551806518075180851809518105181151812518135181451815518165181751818518195182051821518225182351824518255182651827518285182951830518315183251833518345183551836518375183851839518405184151842518435184451845518465184751848518495185051851518525185351854518555185651857518585185951860518615186251863518645186551866518675186851869518705187151872518735187451875518765187751878518795188051881518825188351884518855188651887518885188951890518915189251893518945189551896518975189851899519005190151902519035190451905519065190751908519095191051911519125191351914519155191651917519185191951920519215192251923519245192551926519275192851929519305193151932519335193451935519365193751938519395194051941519425194351944519455194651947519485194951950519515195251953519545195551956519575195851959519605196151962519635196451965519665196751968519695197051971519725197351974519755197651977519785197951980519815198251983519845198551986519875198851989519905199151992519935199451995519965199751998519995200052001520025200352004520055200652007520085200952010520115201252013520145201552016520175201852019520205202152022520235202452025520265202752028520295203052031520325203352034520355203652037520385203952040520415204252043520445204552046520475204852049520505205152052520535205452055520565205752058520595206052061520625206352064520655206652067520685206952070520715207252073520745207552076520775207852079520805208152082520835208452085520865208752088520895209052091520925209352094520955209652097520985209952100521015210252103521045210552106521075210852109521105211152112521135211452115521165211752118521195212052121521225212352124521255212652127521285212952130521315213252133521345213552136521375213852139521405214152142521435214452145521465214752148521495215052151521525215352154521555215652157521585215952160521615216252163521645216552166521675216852169521705217152172521735217452175521765217752178521795218052181521825218352184521855218652187521885218952190521915219252193521945219552196521975219852199522005220152202522035220452205522065220752208522095221052211522125221352214522155221652217522185221952220522215222252223522245222552226522275222852229522305223152232522335223452235522365223752238522395224052241522425224352244522455224652247522485224952250522515225252253522545225552256522575225852259522605226152262522635226452265522665226752268522695227052271522725227352274522755227652277522785227952280522815228252283522845228552286522875228852289522905229152292522935229452295522965229752298522995230052301523025230352304523055230652307523085230952310523115231252313523145231552316523175231852319523205232152322523235232452325523265232752328523295233052331523325233352334523355233652337523385233952340523415234252343523445234552346523475234852349523505235152352523535235452355523565235752358523595236052361523625236352364523655236652367523685236952370523715237252373523745237552376523775237852379523805238152382523835238452385523865238752388523895239052391523925239352394523955239652397523985239952400524015240252403524045240552406524075240852409524105241152412524135241452415524165241752418524195242052421524225242352424524255242652427524285242952430524315243252433524345243552436524375243852439524405244152442524435244452445524465244752448524495245052451524525245352454524555245652457524585245952460524615246252463524645246552466524675246852469524705247152472524735247452475524765247752478524795248052481524825248352484524855248652487524885248952490524915249252493524945249552496524975249852499525005250152502525035250452505525065250752508525095251052511525125251352514525155251652517525185251952520525215252252523525245252552526525275252852529525305253152532525335253452535525365253752538525395254052541525425254352544525455254652547525485254952550525515255252553525545255552556525575255852559525605256152562525635256452565525665256752568525695257052571525725257352574525755257652577525785257952580525815258252583525845258552586525875258852589525905259152592525935259452595525965259752598525995260052601526025260352604526055260652607526085260952610526115261252613526145261552616526175261852619526205262152622526235262452625526265262752628526295263052631526325263352634526355263652637526385263952640526415264252643526445264552646526475264852649526505265152652526535265452655526565265752658526595266052661526625266352664526655266652667526685266952670526715267252673526745267552676526775267852679526805268152682526835268452685526865268752688526895269052691526925269352694526955269652697526985269952700527015270252703527045270552706527075270852709527105271152712527135271452715527165271752718527195272052721527225272352724527255272652727527285272952730527315273252733527345273552736527375273852739527405274152742527435274452745527465274752748527495275052751527525275352754527555275652757527585275952760527615276252763527645276552766527675276852769527705277152772527735277452775527765277752778527795278052781527825278352784527855278652787527885278952790527915279252793527945279552796527975279852799528005280152802528035280452805528065280752808528095281052811528125281352814528155281652817528185281952820528215282252823528245282552826528275282852829528305283152832528335283452835528365283752838528395284052841528425284352844528455284652847528485284952850528515285252853528545285552856528575285852859528605286152862528635286452865528665286752868528695287052871528725287352874528755287652877528785287952880528815288252883528845288552886528875288852889528905289152892528935289452895528965289752898528995290052901529025290352904529055290652907529085290952910529115291252913529145291552916529175291852919529205292152922529235292452925529265292752928529295293052931529325293352934529355293652937529385293952940529415294252943529445294552946529475294852949529505295152952529535295452955529565295752958529595296052961529625296352964529655296652967529685296952970529715297252973529745297552976529775297852979529805298152982529835298452985529865298752988529895299052991529925299352994529955299652997529985299953000530015300253003530045300553006530075300853009530105301153012530135301453015530165301753018530195302053021530225302353024530255302653027530285302953030530315303253033530345303553036530375303853039530405304153042530435304453045530465304753048530495305053051530525305353054530555305653057530585305953060530615306253063530645306553066530675306853069530705307153072530735307453075530765307753078530795308053081530825308353084530855308653087530885308953090530915309253093530945309553096530975309853099531005310153102531035310453105531065310753108531095311053111531125311353114531155311653117531185311953120531215312253123531245312553126531275312853129531305313153132531335313453135531365313753138531395314053141531425314353144531455314653147531485314953150531515315253153531545315553156531575315853159531605316153162531635316453165531665316753168531695317053171531725317353174531755317653177531785317953180531815318253183531845318553186531875318853189531905319153192531935319453195531965319753198531995320053201532025320353204532055320653207532085320953210532115321253213532145321553216532175321853219532205322153222532235322453225532265322753228532295323053231532325323353234532355323653237532385323953240532415324253243532445324553246532475324853249532505325153252532535325453255532565325753258532595326053261532625326353264532655326653267532685326953270532715327253273532745327553276532775327853279532805328153282532835328453285532865328753288532895329053291532925329353294532955329653297532985329953300533015330253303533045330553306533075330853309533105331153312533135331453315533165331753318533195332053321533225332353324533255332653327533285332953330533315333253333533345333553336533375333853339533405334153342533435334453345533465334753348533495335053351533525335353354533555335653357533585335953360533615336253363533645336553366533675336853369533705337153372533735337453375533765337753378533795338053381533825338353384533855338653387533885338953390533915339253393533945339553396533975339853399534005340153402534035340453405534065340753408534095341053411534125341353414534155341653417534185341953420534215342253423534245342553426534275342853429534305343153432534335343453435534365343753438534395344053441534425344353444534455344653447534485344953450534515345253453534545345553456534575345853459534605346153462534635346453465534665346753468534695347053471534725347353474534755347653477534785347953480534815348253483534845348553486534875348853489534905349153492534935349453495534965349753498534995350053501535025350353504535055350653507535085350953510535115351253513535145351553516535175351853519535205352153522535235352453525535265352753528535295353053531535325353353534535355353653537535385353953540535415354253543535445354553546535475354853549535505355153552535535355453555535565355753558535595356053561535625356353564535655356653567535685356953570535715357253573535745357553576535775357853579535805358153582535835358453585535865358753588535895359053591535925359353594535955359653597535985359953600536015360253603536045360553606536075360853609536105361153612536135361453615536165361753618536195362053621536225362353624536255362653627536285362953630536315363253633536345363553636536375363853639536405364153642536435364453645536465364753648536495365053651536525365353654536555365653657536585365953660536615366253663536645366553666536675366853669536705367153672536735367453675536765367753678536795368053681536825368353684536855368653687536885368953690536915369253693536945369553696536975369853699537005370153702537035370453705537065370753708537095371053711537125371353714537155371653717537185371953720537215372253723537245372553726537275372853729537305373153732537335373453735537365373753738537395374053741537425374353744537455374653747537485374953750537515375253753537545375553756537575375853759537605376153762537635376453765537665376753768537695377053771537725377353774537755377653777537785377953780537815378253783537845378553786537875378853789537905379153792537935379453795537965379753798537995380053801538025380353804538055380653807538085380953810538115381253813538145381553816538175381853819538205382153822538235382453825538265382753828538295383053831538325383353834538355383653837538385383953840538415384253843538445384553846538475384853849538505385153852538535385453855538565385753858538595386053861538625386353864538655386653867538685386953870538715387253873538745387553876538775387853879538805388153882538835388453885538865388753888538895389053891538925389353894538955389653897538985389953900539015390253903539045390553906539075390853909539105391153912539135391453915539165391753918539195392053921539225392353924539255392653927539285392953930539315393253933539345393553936539375393853939539405394153942539435394453945539465394753948539495395053951539525395353954539555395653957539585395953960539615396253963539645396553966539675396853969539705397153972539735397453975539765397753978539795398053981539825398353984539855398653987539885398953990539915399253993539945399553996539975399853999540005400154002540035400454005540065400754008540095401054011540125401354014540155401654017540185401954020540215402254023540245402554026540275402854029540305403154032540335403454035540365403754038540395404054041540425404354044540455404654047540485404954050540515405254053540545405554056540575405854059540605406154062540635406454065540665406754068540695407054071540725407354074540755407654077540785407954080540815408254083540845408554086540875408854089540905409154092540935409454095540965409754098540995410054101541025410354104541055410654107541085410954110541115411254113541145411554116541175411854119541205412154122541235412454125541265412754128541295413054131541325413354134541355413654137541385413954140541415414254143541445414554146541475414854149541505415154152541535415454155541565415754158541595416054161541625416354164541655416654167541685416954170541715417254173541745417554176541775417854179541805418154182541835418454185541865418754188541895419054191541925419354194541955419654197541985419954200542015420254203542045420554206542075420854209542105421154212542135421454215542165421754218542195422054221542225422354224542255422654227542285422954230542315423254233542345423554236542375423854239542405424154242542435424454245542465424754248542495425054251542525425354254542555425654257542585425954260542615426254263542645426554266542675426854269542705427154272542735427454275542765427754278542795428054281542825428354284542855428654287542885428954290542915429254293542945429554296542975429854299543005430154302543035430454305543065430754308543095431054311543125431354314543155431654317543185431954320543215432254323543245432554326543275432854329543305433154332543335433454335543365433754338543395434054341543425434354344543455434654347543485434954350543515435254353543545435554356543575435854359543605436154362543635436454365543665436754368543695437054371543725437354374543755437654377543785437954380543815438254383543845438554386543875438854389543905439154392543935439454395543965439754398543995440054401544025440354404544055440654407544085440954410544115441254413544145441554416544175441854419544205442154422544235442454425544265442754428544295443054431544325443354434544355443654437544385443954440544415444254443544445444554446544475444854449544505445154452544535445454455544565445754458544595446054461544625446354464544655446654467544685446954470544715447254473544745447554476544775447854479544805448154482544835448454485544865448754488544895449054491544925449354494544955449654497544985449954500545015450254503545045450554506545075450854509545105451154512545135451454515545165451754518545195452054521545225452354524545255452654527545285452954530545315453254533545345453554536545375453854539545405454154542545435454454545545465454754548545495455054551545525455354554545555455654557545585455954560545615456254563545645456554566545675456854569545705457154572545735457454575545765457754578545795458054581545825458354584545855458654587545885458954590545915459254593545945459554596545975459854599546005460154602546035460454605546065460754608546095461054611546125461354614546155461654617546185461954620546215462254623546245462554626546275462854629546305463154632546335463454635546365463754638546395464054641546425464354644546455464654647546485464954650546515465254653546545465554656546575465854659546605466154662546635466454665546665466754668546695467054671546725467354674546755467654677546785467954680546815468254683546845468554686546875468854689546905469154692546935469454695546965469754698546995470054701547025470354704547055470654707547085470954710547115471254713547145471554716547175471854719547205472154722547235472454725547265472754728547295473054731547325473354734547355473654737547385473954740547415474254743547445474554746547475474854749547505475154752547535475454755547565475754758547595476054761547625476354764547655476654767547685476954770547715477254773547745477554776547775477854779547805478154782547835478454785547865478754788547895479054791547925479354794547955479654797547985479954800548015480254803548045480554806548075480854809548105481154812548135481454815548165481754818548195482054821548225482354824548255482654827548285482954830548315483254833548345483554836548375483854839548405484154842548435484454845548465484754848548495485054851548525485354854548555485654857548585485954860548615486254863548645486554866548675486854869548705487154872548735487454875548765487754878548795488054881548825488354884548855488654887548885488954890548915489254893548945489554896548975489854899549005490154902549035490454905549065490754908549095491054911549125491354914549155491654917549185491954920549215492254923549245492554926549275492854929549305493154932549335493454935549365493754938549395494054941549425494354944549455494654947549485494954950549515495254953549545495554956549575495854959549605496154962549635496454965549665496754968549695497054971549725497354974549755497654977549785497954980549815498254983549845498554986549875498854989549905499154992549935499454995549965499754998549995500055001550025500355004550055500655007550085500955010550115501255013550145501555016550175501855019550205502155022550235502455025550265502755028550295503055031550325503355034550355503655037550385503955040550415504255043550445504555046550475504855049550505505155052550535505455055550565505755058550595506055061550625506355064550655506655067550685506955070550715507255073550745507555076550775507855079550805508155082550835508455085550865508755088550895509055091550925509355094550955509655097550985509955100551015510255103551045510555106551075510855109551105511155112551135511455115551165511755118551195512055121551225512355124551255512655127551285512955130551315513255133551345513555136551375513855139551405514155142551435514455145551465514755148551495515055151551525515355154551555515655157551585515955160551615516255163551645516555166551675516855169551705517155172551735517455175551765517755178551795518055181551825518355184551855518655187551885518955190551915519255193551945519555196551975519855199552005520155202552035520455205552065520755208552095521055211552125521355214552155521655217552185521955220552215522255223552245522555226552275522855229552305523155232552335523455235552365523755238552395524055241552425524355244552455524655247552485524955250552515525255253552545525555256552575525855259552605526155262552635526455265552665526755268552695527055271552725527355274552755527655277552785527955280552815528255283552845528555286552875528855289552905529155292552935529455295552965529755298552995530055301553025530355304553055530655307553085530955310553115531255313553145531555316553175531855319553205532155322553235532455325553265532755328553295533055331553325533355334553355533655337553385533955340553415534255343553445534555346553475534855349553505535155352553535535455355553565535755358553595536055361553625536355364553655536655367553685536955370553715537255373553745537555376553775537855379553805538155382553835538455385553865538755388553895539055391553925539355394553955539655397553985539955400554015540255403554045540555406554075540855409554105541155412554135541455415554165541755418554195542055421554225542355424554255542655427554285542955430554315543255433554345543555436554375543855439554405544155442554435544455445554465544755448554495545055451554525545355454554555545655457554585545955460554615546255463554645546555466554675546855469554705547155472554735547455475554765547755478554795548055481554825548355484554855548655487554885548955490554915549255493554945549555496554975549855499555005550155502555035550455505555065550755508555095551055511555125551355514555155551655517555185551955520555215552255523555245552555526555275552855529555305553155532555335553455535555365553755538555395554055541555425554355544555455554655547555485554955550555515555255553555545555555556555575555855559555605556155562555635556455565555665556755568555695557055571555725557355574555755557655577555785557955580555815558255583555845558555586555875558855589555905559155592555935559455595555965559755598555995560055601556025560355604556055560655607556085560955610556115561255613556145561555616556175561855619556205562155622556235562455625556265562755628556295563055631556325563355634556355563655637556385563955640556415564255643556445564555646556475564855649556505565155652556535565455655556565565755658556595566055661556625566355664556655566655667556685566955670556715567255673556745567555676556775567855679556805568155682556835568455685556865568755688556895569055691556925569355694556955569655697556985569955700557015570255703557045570555706557075570855709557105571155712557135571455715557165571755718557195572055721557225572355724557255572655727557285572955730557315573255733557345573555736557375573855739557405574155742557435574455745557465574755748557495575055751557525575355754557555575655757557585575955760557615576255763557645576555766557675576855769557705577155772557735577455775557765577755778557795578055781557825578355784557855578655787557885578955790557915579255793557945579555796557975579855799558005580155802558035580455805558065580755808558095581055811558125581355814558155581655817558185581955820558215582255823558245582555826558275582855829558305583155832558335583455835558365583755838558395584055841558425584355844558455584655847558485584955850558515585255853558545585555856558575585855859558605586155862558635586455865558665586755868558695587055871558725587355874558755587655877558785587955880558815588255883558845588555886558875588855889558905589155892558935589455895558965589755898558995590055901559025590355904559055590655907559085590955910559115591255913559145591555916559175591855919559205592155922559235592455925559265592755928559295593055931559325593355934559355593655937559385593955940559415594255943559445594555946559475594855949559505595155952559535595455955559565595755958559595596055961559625596355964559655596655967559685596955970559715597255973559745597555976559775597855979559805598155982559835598455985559865598755988559895599055991559925599355994559955599655997559985599956000560015600256003560045600556006560075600856009560105601156012560135601456015560165601756018560195602056021560225602356024560255602656027560285602956030560315603256033560345603556036560375603856039560405604156042560435604456045560465604756048560495605056051560525605356054560555605656057560585605956060560615606256063560645606556066560675606856069560705607156072560735607456075560765607756078560795608056081560825608356084560855608656087560885608956090560915609256093560945609556096560975609856099561005610156102561035610456105561065610756108561095611056111561125611356114561155611656117561185611956120561215612256123561245612556126561275612856129561305613156132561335613456135561365613756138561395614056141561425614356144561455614656147561485614956150561515615256153561545615556156561575615856159561605616156162561635616456165561665616756168561695617056171561725617356174561755617656177561785617956180561815618256183561845618556186561875618856189561905619156192561935619456195561965619756198561995620056201562025620356204562055620656207562085620956210562115621256213562145621556216562175621856219562205622156222562235622456225562265622756228562295623056231562325623356234562355623656237562385623956240562415624256243562445624556246562475624856249562505625156252562535625456255562565625756258562595626056261562625626356264562655626656267562685626956270562715627256273562745627556276562775627856279562805628156282562835628456285562865628756288562895629056291562925629356294562955629656297562985629956300563015630256303563045630556306563075630856309563105631156312563135631456315563165631756318563195632056321563225632356324563255632656327563285632956330563315633256333563345633556336563375633856339563405634156342563435634456345563465634756348563495635056351563525635356354563555635656357563585635956360563615636256363563645636556366563675636856369563705637156372563735637456375563765637756378563795638056381563825638356384563855638656387563885638956390563915639256393563945639556396563975639856399564005640156402564035640456405564065640756408564095641056411564125641356414564155641656417564185641956420564215642256423564245642556426564275642856429564305643156432564335643456435564365643756438564395644056441564425644356444564455644656447564485644956450564515645256453564545645556456564575645856459564605646156462564635646456465564665646756468564695647056471564725647356474564755647656477564785647956480564815648256483564845648556486564875648856489564905649156492564935649456495564965649756498564995650056501565025650356504565055650656507565085650956510565115651256513565145651556516565175651856519565205652156522565235652456525565265652756528565295653056531565325653356534565355653656537565385653956540565415654256543565445654556546565475654856549565505655156552565535655456555565565655756558565595656056561565625656356564565655656656567565685656956570565715657256573565745657556576565775657856579565805658156582565835658456585565865658756588565895659056591565925659356594565955659656597565985659956600566015660256603566045660556606566075660856609566105661156612566135661456615566165661756618566195662056621566225662356624566255662656627566285662956630566315663256633566345663556636566375663856639566405664156642566435664456645566465664756648566495665056651566525665356654566555665656657566585665956660566615666256663566645666556666566675666856669566705667156672566735667456675566765667756678566795668056681566825668356684566855668656687566885668956690566915669256693566945669556696566975669856699567005670156702567035670456705567065670756708567095671056711567125671356714567155671656717567185671956720567215672256723567245672556726567275672856729567305673156732567335673456735567365673756738567395674056741567425674356744567455674656747567485674956750567515675256753567545675556756567575675856759567605676156762567635676456765567665676756768567695677056771567725677356774567755677656777567785677956780567815678256783567845678556786567875678856789567905679156792567935679456795567965679756798567995680056801568025680356804568055680656807568085680956810568115681256813568145681556816568175681856819568205682156822568235682456825568265682756828568295683056831568325683356834568355683656837568385683956840568415684256843568445684556846568475684856849568505685156852568535685456855568565685756858568595686056861568625686356864568655686656867568685686956870568715687256873568745687556876568775687856879568805688156882568835688456885568865688756888568895689056891568925689356894568955689656897568985689956900569015690256903569045690556906569075690856909569105691156912569135691456915569165691756918569195692056921569225692356924569255692656927569285692956930569315693256933569345693556936569375693856939569405694156942569435694456945569465694756948569495695056951569525695356954569555695656957569585695956960569615696256963569645696556966569675696856969569705697156972569735697456975569765697756978569795698056981569825698356984 |
- /**
- * @license Highcharts Gantt JS v8.2.0 (2020-08-20)
- *
- * (c) 2017-2018 Lars Cabrera, Torstein Honsi, Jon Arild Nygard & Oystein Moseng
- *
- * License: www.highcharts.com/license
- */
- 'use strict';
- (function (root, factory) {
- if (typeof module === 'object' && module.exports) {
- factory['default'] = factory;
- module.exports = root.document ?
- factory(root) :
- factory;
- } else if (typeof define === 'function' && define.amd) {
- define('highcharts/highcharts-gantt', function () {
- return factory(root);
- });
- } else {
- if (root.Highcharts) {
- root.Highcharts.error(16, true);
- }
- root.Highcharts = factory(root);
- }
- }(typeof window !== 'undefined' ? window : this, function (win) {
- var _modules = {};
- function _registerModule(obj, path, args, fn) {
- if (!obj.hasOwnProperty(path)) {
- obj[path] = fn.apply(null, args);
- }
- }
- _registerModule(_modules, 'Core/Globals.js', [], function () {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /* globals Image, window */
- /**
- * Reference to the global SVGElement class as a workaround for a name conflict
- * in the Highcharts namespace.
- *
- * @global
- * @typedef {global.SVGElement} GlobalSVGElement
- *
- * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement
- */
- // glob is a temporary fix to allow our es-modules to work.
- var glob = ( // @todo UMD variable named `window`, and glob named `win`
- typeof win !== 'undefined' ?
- win :
- typeof window !== 'undefined' ?
- window :
- {}), doc = glob.document, SVG_NS = 'http://www.w3.org/2000/svg', userAgent = (glob.navigator && glob.navigator.userAgent) || '', svg = (doc &&
- doc.createElementNS &&
- !!doc.createElementNS(SVG_NS, 'svg').createSVGRect), isMS = /(edge|msie|trident)/i.test(userAgent) && !glob.opera, isFirefox = userAgent.indexOf('Firefox') !== -1, isChrome = userAgent.indexOf('Chrome') !== -1, hasBidiBug = (isFirefox &&
- parseInt(userAgent.split('Firefox/')[1], 10) < 4 // issue #38
- );
- var H = {
- product: 'Highcharts',
- version: '8.2.0',
- deg2rad: Math.PI * 2 / 360,
- doc: doc,
- hasBidiBug: hasBidiBug,
- hasTouch: !!glob.TouchEvent,
- isMS: isMS,
- isWebKit: userAgent.indexOf('AppleWebKit') !== -1,
- isFirefox: isFirefox,
- isChrome: isChrome,
- isSafari: !isChrome && userAgent.indexOf('Safari') !== -1,
- isTouchDevice: /(Mobile|Android|Windows Phone)/.test(userAgent),
- SVG_NS: SVG_NS,
- chartCount: 0,
- seriesTypes: {},
- symbolSizes: {},
- svg: svg,
- win: glob,
- marginNames: ['plotTop', 'marginRight', 'marginBottom', 'plotLeft'],
- noop: function () { },
- /**
- * Theme options that should get applied to the chart. In module mode it
- * might not be possible to change this property because of read-only
- * restrictions, instead use {@link Highcharts.setOptions}.
- *
- * @name Highcharts.theme
- * @type {Highcharts.Options}
- */
- /**
- * An array containing the current chart objects in the page. A chart's
- * position in the array is preserved throughout the page's lifetime. When
- * a chart is destroyed, the array item becomes `undefined`.
- *
- * @name Highcharts.charts
- * @type {Array<Highcharts.Chart|undefined>}
- */
- charts: [],
- /**
- * A hook for defining additional date format specifiers. New
- * specifiers are defined as key-value pairs by using the
- * specifier as key, and a function which takes the timestamp as
- * value. This function returns the formatted portion of the
- * date.
- *
- * @sample highcharts/global/dateformats/
- * Adding support for week number
- *
- * @name Highcharts.dateFormats
- * @type {Highcharts.Dictionary<Highcharts.TimeFormatCallbackFunction>}
- */
- dateFormats: {}
- };
- return H;
- });
- _registerModule(_modules, 'Core/Utilities.js', [_modules['Core/Globals.js']], function (H) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * An animation configuration. Animation configurations can also be defined as
- * booleans, where `false` turns off animation and `true` defaults to a duration
- * of 500ms and defer of 0ms.
- *
- * @interface Highcharts.AnimationOptionsObject
- */ /**
- * A callback function to exectute when the animation finishes.
- * @name Highcharts.AnimationOptionsObject#complete
- * @type {Function|undefined}
- */ /**
- * The animation defer in milliseconds.
- * @name Highcharts.AnimationOptionsObject#defer
- * @type {number|undefined}
- */ /**
- * The animation duration in milliseconds.
- * @name Highcharts.AnimationOptionsObject#duration
- * @type {number|undefined}
- */ /**
- * The name of an easing function as defined on the `Math` object.
- * @name Highcharts.AnimationOptionsObject#easing
- * @type {string|Function|undefined}
- */ /**
- * A callback function to execute on each step of each attribute or CSS property
- * that's being animated. The first argument contains information about the
- * animation and progress.
- * @name Highcharts.AnimationOptionsObject#step
- * @type {Function|undefined}
- */
- /**
- * Creates a frame for the animated SVG element.
- *
- * @callback Highcharts.AnimationStepCallbackFunction
- *
- * @param {Highcharts.SVGElement} this
- * The SVG element to animate.
- *
- * @return {void}
- */
- /**
- * Interface description for a class.
- *
- * @interface Highcharts.Class<T>
- * @extends Function
- */ /**
- * Class costructor.
- * @function Highcharts.Class<T>#new
- * @param {...Array<*>} args
- * Constructor arguments.
- * @return {T}
- * Class instance.
- */
- /**
- * A style object with camel case property names to define visual appearance of
- * a SVG element or HTML element. The properties can be whatever styles are
- * supported on the given SVG or HTML element.
- *
- * @example
- * {
- * fontFamily: 'monospace',
- * fontSize: '1.2em'
- * }
- *
- * @interface Highcharts.CSSObject
- */ /**
- * @name Highcharts.CSSObject#[key:string]
- * @type {boolean|number|string|undefined}
- */ /**
- * Background style for the element.
- * @name Highcharts.CSSObject#background
- * @type {string|undefined}
- */ /**
- * Background color of the element.
- * @name Highcharts.CSSObject#backgroundColor
- * @type {Highcharts.ColorString|undefined}
- */ /**
- * Border style for the element.
- * @name Highcharts.CSSObject#border
- * @type {string|undefined}
- */ /**
- * Radius of the element border.
- * @name Highcharts.CSSObject#borderRadius
- * @type {number|undefined}
- */ /**
- * Color used in the element. The 'contrast' option is a Highcharts custom
- * property that results in black or white, depending on the background of the
- * element.
- * @name Highcharts.CSSObject#color
- * @type {'contrast'|Highcharts.ColorString|undefined}
- */ /**
- * Style of the mouse cursor when resting over the element.
- * @name Highcharts.CSSObject#cursor
- * @type {Highcharts.CursorValue|undefined}
- */ /**
- * Font family of the element text. Multiple values have to be in decreasing
- * preference order and separated by comma.
- * @name Highcharts.CSSObject#fontFamily
- * @type {string|undefined}
- */ /**
- * Font size of the element text.
- * @name Highcharts.CSSObject#fontSize
- * @type {string|undefined}
- */ /**
- * Font weight of the element text.
- * @name Highcharts.CSSObject#fontWeight
- * @type {string|undefined}
- */ /**
- * Height of the element.
- * @name Highcharts.CSSObject#height
- * @type {number|undefined}
- */ /**
- * Width of the element border.
- * @name Highcharts.CSSObject#lineWidth
- * @type {number|undefined}
- */ /**
- * Opacity of the element.
- * @name Highcharts.CSSObject#opacity
- * @type {number|undefined}
- */ /**
- * Space around the element content.
- * @name Highcharts.CSSObject#padding
- * @type {string|undefined}
- */ /**
- * Behaviour of the element when the mouse cursor rests over it.
- * @name Highcharts.CSSObject#pointerEvents
- * @type {string|undefined}
- */ /**
- * Positioning of the element.
- * @name Highcharts.CSSObject#position
- * @type {string|undefined}
- */ /**
- * Alignment of the element text.
- * @name Highcharts.CSSObject#textAlign
- * @type {string|undefined}
- */ /**
- * Additional decoration of the element text.
- * @name Highcharts.CSSObject#textDecoration
- * @type {string|undefined}
- */ /**
- * Outline style of the element text.
- * @name Highcharts.CSSObject#textOutline
- * @type {string|undefined}
- */ /**
- * Line break style of the element text. Highcharts SVG elements support
- * `ellipsis` when a `width` is set.
- * @name Highcharts.CSSObject#textOverflow
- * @type {string|undefined}
- */ /**
- * Top spacing of the element relative to the parent element.
- * @name Highcharts.CSSObject#top
- * @type {string|undefined}
- */ /**
- * Animated transition of selected element properties.
- * @name Highcharts.CSSObject#transition
- * @type {string|undefined}
- */ /**
- * Line break style of the element text.
- * @name Highcharts.CSSObject#whiteSpace
- * @type {string|undefined}
- */ /**
- * Width of the element.
- * @name Highcharts.CSSObject#width
- * @type {number|undefined}
- */
- /**
- * All possible cursor styles.
- *
- * @typedef {'alias'|'all-scroll'|'auto'|'cell'|'col-resize'|'context-menu'|'copy'|'crosshair'|'default'|'e-resize'|'ew-resize'|'grab'|'grabbing'|'help'|'move'|'n-resize'|'ne-resize'|'nesw-resize'|'no-drop'|'none'|'not-allowed'|'ns-resize'|'nw-resize'|'nwse-resize'|'pointer'|'progress'|'row-resize'|'s-resize'|'se-resize'|'sw-resize'|'text'|'vertical-text'|'w-resize'|'wait'|'zoom-in'|'zoom-out'} Highcharts.CursorValue
- */
- /**
- * All possible dash styles.
- *
- * @typedef {'Dash'|'DashDot'|'Dot'|'LongDash'|'LongDashDot'|'LongDashDotDot'|'ShortDash'|'ShortDashDot'|'ShortDashDotDot'|'ShortDot'|'Solid'} Highcharts.DashStyleValue
- */
- /**
- * Generic dictionary in TypeScript notation.
- * Use the native `Record<string, any>` instead.
- *
- * @deprecated
- * @interface Highcharts.Dictionary<T>
- */ /**
- * @name Highcharts.Dictionary<T>#[key:string]
- * @type {T}
- */
- /**
- * The function callback to execute when the event is fired. The `this` context
- * contains the instance, that fired the event.
- *
- * @callback Highcharts.EventCallbackFunction<T>
- *
- * @param {T} this
- *
- * @param {Highcharts.Dictionary<*>|Event} [eventArguments]
- * Event arguments.
- *
- * @return {boolean|void}
- */
- /**
- * The event options for adding function callback.
- *
- * @interface Highcharts.EventOptionsObject
- */ /**
- * The order the event handler should be called. This opens for having one
- * handler be called before another, independent of in which order they were
- * added.
- * @name Highcharts.EventOptionsObject#order
- * @type {number}
- */
- /**
- * Formats data as a string. Usually the data is accessible throught the `this`
- * keyword.
- *
- * @callback Highcharts.FormatterCallbackFunction<T>
- *
- * @param {T} this
- * Context to format
- *
- * @return {string}
- * Formatted text
- */
- /**
- * An object of key-value pairs for HTML attributes.
- *
- * @typedef {Highcharts.Dictionary<boolean|number|string|Function>} Highcharts.HTMLAttributes
- */
- /**
- * An HTML DOM element. The type is a reference to the regular HTMLElement in
- * the global scope.
- *
- * @typedef {global.HTMLElement} Highcharts.HTMLDOMElement
- *
- * @see https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement
- */
- /**
- * The iterator callback.
- *
- * @callback Highcharts.ObjectEachCallbackFunction<T>
- *
- * @param {T} this
- * The context.
- *
- * @param {*} value
- * The property value.
- *
- * @param {string} key
- * The property key.
- *
- * @param {*} obj
- * The object that objectEach is being applied to.
- */
- /**
- * An object containing `left` and `top` properties for the position in the
- * page.
- *
- * @interface Highcharts.OffsetObject
- */ /**
- * Left distance to the page border.
- * @name Highcharts.OffsetObject#left
- * @type {number}
- */ /**
- * Top distance to the page border.
- * @name Highcharts.OffsetObject#top
- * @type {number}
- */
- /**
- * Describes a range.
- *
- * @interface Highcharts.RangeObject
- */ /**
- * Maximum number of the range.
- * @name Highcharts.RangeObject#max
- * @type {number}
- */ /**
- * Minimum number of the range.
- * @name Highcharts.RangeObject#min
- * @type {number}
- */
- /**
- * If a number is given, it defines the pixel length. If a percentage string is
- * given, like for example `'50%'`, the setting defines a length relative to a
- * base size, for example the size of a container.
- *
- * @typedef {number|string} Highcharts.RelativeSize
- */
- /**
- * Proceed function to call original (wrapped) function.
- *
- * @callback Highcharts.WrapProceedFunction
- *
- * @param {*} [arg1]
- * Optional argument. Without any arguments defaults to first argument of
- * the wrapping function.
- *
- * @param {*} [arg2]
- * Optional argument. Without any arguments defaults to second argument
- * of the wrapping function.
- *
- * @param {*} [arg3]
- * Optional argument. Without any arguments defaults to third argument of
- * the wrapping function.
- *
- * @return {*}
- * Return value of the original function.
- */
- /**
- * The Highcharts object is the placeholder for all other members, and various
- * utility functions. The most important member of the namespace would be the
- * chart constructor.
- *
- * @example
- * var chart = Highcharts.chart('container', { ... });
- *
- * @namespace Highcharts
- */
- H.timers = [];
- var charts = H.charts,
- doc = H.doc,
- win = H.win;
- /**
- * Provide error messages for debugging, with links to online explanation. This
- * function can be overridden to provide custom error handling.
- *
- * @sample highcharts/chart/highcharts-error/
- * Custom error handler
- *
- * @function Highcharts.error
- *
- * @param {number|string} code
- * The error code. See
- * [errors.xml](https://github.com/highcharts/highcharts/blob/master/errors/errors.xml)
- * for available codes. If it is a string, the error message is printed
- * directly in the console.
- *
- * @param {boolean} [stop=false]
- * Whether to throw an error or just log a warning in the console.
- *
- * @param {Highcharts.Chart} [chart]
- * Reference to the chart that causes the error. Used in 'debugger'
- * module to display errors directly on the chart.
- * Important note: This argument is undefined for errors that lack
- * access to the Chart instance.
- *
- * @param {Highcharts.Dictionary<string>} [params]
- * Additional parameters for the generated message.
- *
- * @return {void}
- */
- function error(code, stop, chart, params) {
- var severity = stop ? 'Highcharts error' : 'Highcharts warning';
- if (code === 32) {
- code = severity + ": Deprecated member";
- }
- var isCode = isNumber(code),
- message = isCode ?
- severity + " #" + code + ": www.highcharts.com/errors/" + code + "/" :
- code.toString(),
- defaultHandler = function () {
- if (stop) {
- throw new Error(message);
- }
- // else ...
- if (win.console &&
- error.messages.indexOf(message) === -1 // prevent console flooting
- ) {
- console.log(message); // eslint-disable-line no-console
- }
- };
- if (typeof params !== 'undefined') {
- var additionalMessages_1 = '';
- if (isCode) {
- message += '?';
- }
- objectEach(params, function (value, key) {
- additionalMessages_1 += "\n - " + key + ": " + value;
- if (isCode) {
- message += encodeURI(key) + '=' + encodeURI(value);
- }
- });
- message += additionalMessages_1;
- }
- if (chart) {
- fireEvent(chart, 'displayError', { code: code, message: message, params: params }, defaultHandler);
- }
- else {
- defaultHandler();
- }
- error.messages.push(message);
- }
- (function (error) {
- error.messages = [];
- })(error || (error = {}));
- H.error = error;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * An animator object used internally. One instance applies to one property
- * (attribute or style prop) on one element. Animation is always initiated
- * through {@link SVGElement#animate}.
- *
- * @example
- * var rect = renderer.rect(0, 0, 10, 10).add();
- * rect.animate({ width: 100 });
- *
- * @private
- * @class
- * @name Highcharts.Fx
- */
- var Fx = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- /**
- *
- * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} elem
- * The element to animate.
- *
- * @param {Partial<Highcharts.AnimationOptionsObject>} options
- * Animation options.
- *
- * @param {string} prop
- * The single attribute or CSS property to animate.
- */
- function Fx(elem, options, prop) {
- this.options = options;
- this.elem = elem;
- this.prop = prop;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Set the current step of a path definition on SVGElement.
- *
- * @function Highcharts.Fx#dSetter
- *
- * @return {void}
- */
- Fx.prototype.dSetter = function () {
- var paths = this.paths,
- start = paths && paths[0],
- end = paths && paths[1],
- path = [],
- now = this.now || 0;
- // Land on the final path without adjustment points appended in the ends
- if (now === 1 || !start || !end) {
- path = this.toD || [];
- }
- else if (start.length === end.length && now < 1) {
- for (var i = 0; i < end.length; i++) {
- // Tween between the start segment and the end segment. Start
- // with a copy of the end segment and tween the appropriate
- // numerics
- var startSeg = start[i];
- var endSeg = end[i];
- var tweenSeg = [];
- for (var j = 0; j < endSeg.length; j++) {
- var startItem = startSeg[j];
- var endItem = endSeg[j];
- // Tween numbers
- if (typeof startItem === 'number' &&
- typeof endItem === 'number' &&
- // Arc boolean flags
- !(endSeg[0] === 'A' && (j === 4 || j === 5))) {
- tweenSeg[j] = startItem + now * (endItem - startItem);
- // Strings, take directly from the end segment
- }
- else {
- tweenSeg[j] = endItem;
- }
- }
- path.push(tweenSeg);
- }
- // If animation is finished or length not matching, land on right value
- }
- else {
- path = end;
- }
- this.elem.attr('d', path, void 0, true);
- };
- /**
- * Update the element with the current animation step.
- *
- * @function Highcharts.Fx#update
- *
- * @return {void}
- */
- Fx.prototype.update = function () {
- var elem = this.elem,
- prop = this.prop, // if destroyed, it is null
- now = this.now,
- step = this.options.step;
- // Animation setter defined from outside
- if (this[prop + 'Setter']) {
- this[prop + 'Setter']();
- // Other animations on SVGElement
- }
- else if (elem.attr) {
- if (elem.element) {
- elem.attr(prop, now, null, true);
- }
- // HTML styles, raw HTML content like container size
- }
- else {
- elem.style[prop] = now + this.unit;
- }
- if (step) {
- step.call(elem, now, this);
- }
- };
- /**
- * Run an animation.
- *
- * @function Highcharts.Fx#run
- *
- * @param {number} from
- * The current value, value to start from.
- *
- * @param {number} to
- * The end value, value to land on.
- *
- * @param {string} unit
- * The property unit, for example `px`.
- *
- * @return {void}
- */
- Fx.prototype.run = function (from, to, unit) {
- var self = this,
- options = self.options,
- timer = function (gotoEnd) {
- return timer.stopped ? false : self.step(gotoEnd);
- }, requestAnimationFrame = win.requestAnimationFrame ||
- function (step) {
- setTimeout(step, 13);
- }, step = function () {
- for (var i = 0; i < H.timers.length; i++) {
- if (!H.timers[i]()) {
- H.timers.splice(i--, 1);
- }
- }
- if (H.timers.length) {
- requestAnimationFrame(step);
- }
- };
- if (from === to && !this.elem['forceAnimate:' + this.prop]) {
- delete options.curAnim[this.prop];
- if (options.complete && Object.keys(options.curAnim).length === 0) {
- options.complete.call(this.elem);
- }
- }
- else { // #7166
- this.startTime = +new Date();
- this.start = from;
- this.end = to;
- this.unit = unit;
- this.now = this.start;
- this.pos = 0;
- timer.elem = this.elem;
- timer.prop = this.prop;
- if (timer() && H.timers.push(timer) === 1) {
- requestAnimationFrame(step);
- }
- }
- };
- /**
- * Run a single step in the animation.
- *
- * @function Highcharts.Fx#step
- *
- * @param {boolean} [gotoEnd]
- * Whether to go to the endpoint of the animation after abort.
- *
- * @return {boolean}
- * Returns `true` if animation continues.
- */
- Fx.prototype.step = function (gotoEnd) {
- var t = +new Date(),
- ret,
- done,
- options = this.options,
- elem = this.elem,
- complete = options.complete,
- duration = options.duration,
- curAnim = options.curAnim;
- if (elem.attr && !elem.element) { // #2616, element is destroyed
- ret = false;
- }
- else if (gotoEnd || t >= duration + this.startTime) {
- this.now = this.end;
- this.pos = 1;
- this.update();
- curAnim[this.prop] = true;
- done = true;
- objectEach(curAnim, function (val) {
- if (val !== true) {
- done = false;
- }
- });
- if (done && complete) {
- complete.call(elem);
- }
- ret = false;
- }
- else {
- this.pos = options.easing((t - this.startTime) / duration);
- this.now = this.start + ((this.end - this.start) * this.pos);
- this.update();
- ret = true;
- }
- return ret;
- };
- /**
- * Prepare start and end values so that the path can be animated one to one.
- *
- * @function Highcharts.Fx#initPath
- *
- * @param {Highcharts.SVGElement} elem
- * The SVGElement item.
- *
- * @param {Highcharts.SVGPathArray|undefined} fromD
- * Starting path definition.
- *
- * @param {Highcharts.SVGPathArray} toD
- * Ending path definition.
- *
- * @return {Array<Highcharts.SVGPathArray,Highcharts.SVGPathArray>}
- * An array containing start and end paths in array form so that
- * they can be animated in parallel.
- */
- Fx.prototype.initPath = function (elem, fromD, toD) {
- var shift,
- startX = elem.startX,
- endX = elem.endX,
- fullLength,
- i,
- start = fromD && fromD.slice(), // copy
- end = toD.slice(), // copy
- isArea = elem.isArea,
- positionFactor = isArea ? 2 : 1,
- reverse;
- if (!start) {
- return [end, end];
- }
- /**
- * If shifting points, prepend a dummy point to the end path.
- * @private
- * @param {Highcharts.SVGPathArray} arr - array
- * @param {Highcharts.SVGPathArray} other - array
- * @return {void}
- */
- function prepend(arr, other) {
- while (arr.length < fullLength) {
- // Move to, line to or curve to?
- var moveSegment = arr[0],
- otherSegment = other[fullLength - arr.length];
- if (otherSegment && moveSegment[0] === 'M') {
- if (otherSegment[0] === 'C') {
- arr[0] = [
- 'C',
- moveSegment[1],
- moveSegment[2],
- moveSegment[1],
- moveSegment[2],
- moveSegment[1],
- moveSegment[2]
- ];
- }
- else {
- arr[0] = ['L', moveSegment[1], moveSegment[2]];
- }
- }
- // Prepend a copy of the first point
- arr.unshift(moveSegment);
- // For areas, the bottom path goes back again to the left, so we
- // need to append a copy of the last point.
- if (isArea) {
- arr.push(arr[arr.length - 1]);
- }
- }
- }
- /**
- * Copy and append last point until the length matches the end length.
- * @private
- * @param {Highcharts.SVGPathArray} arr - array
- * @param {Highcharts.SVGPathArray} other - array
- * @return {void}
- */
- function append(arr, other) {
- while (arr.length < fullLength) {
- // Pull out the slice that is going to be appended or inserted.
- // In a line graph, the positionFactor is 1, and the last point
- // is sliced out. In an area graph, the positionFactor is 2,
- // causing the middle two points to be sliced out, since an area
- // path starts at left, follows the upper path then turns and
- // follows the bottom back.
- var segmentToAdd = arr[arr.length / positionFactor - 1].slice();
- // Disable the first control point of curve segments
- if (segmentToAdd[0] === 'C') {
- segmentToAdd[1] = segmentToAdd[5];
- segmentToAdd[2] = segmentToAdd[6];
- }
- if (!isArea) {
- arr.push(segmentToAdd);
- }
- else {
- var lowerSegmentToAdd = arr[arr.length / positionFactor].slice();
- arr.splice(arr.length / 2, 0, segmentToAdd, lowerSegmentToAdd);
- }
- }
- }
- // For sideways animation, find out how much we need to shift to get the
- // start path Xs to match the end path Xs.
- if (startX && endX) {
- for (i = 0; i < startX.length; i++) {
- // Moving left, new points coming in on right
- if (startX[i] === endX[0]) {
- shift = i;
- break;
- // Moving right
- }
- else if (startX[0] ===
- endX[endX.length - startX.length + i]) {
- shift = i;
- reverse = true;
- break;
- // Fixed from the right side, "scaling" left
- }
- else if (startX[startX.length - 1] ===
- endX[endX.length - startX.length + i]) {
- shift = startX.length - i;
- break;
- }
- }
- if (typeof shift === 'undefined') {
- start = [];
- }
- }
- if (start.length && isNumber(shift)) {
- // The common target length for the start and end array, where both
- // arrays are padded in opposite ends
- fullLength = end.length + shift * positionFactor;
- if (!reverse) {
- prepend(end, start);
- append(start, end);
- }
- else {
- prepend(start, end);
- append(end, start);
- }
- }
- return [start, end];
- };
- /**
- * Handle animation of the color attributes directly.
- *
- * @function Highcharts.Fx#fillSetter
- *
- * @return {void}
- */
- Fx.prototype.fillSetter = function () {
- Fx.prototype.strokeSetter.apply(this, arguments);
- };
- /**
- * Handle animation of the color attributes directly.
- *
- * @function Highcharts.Fx#strokeSetter
- *
- * @return {void}
- */
- Fx.prototype.strokeSetter = function () {
- this.elem.attr(this.prop, H.color(this.start).tweenTo(H.color(this.end), this.pos), null, true);
- };
- return Fx;
- }());
- H.Fx = Fx;
- /* eslint-disable valid-jsdoc */
- /**
- * Utility function to deep merge two or more objects and return a third object.
- * If the first argument is true, the contents of the second object is copied
- * into the first object. The merge function can also be used with a single
- * object argument to create a deep copy of an object.
- *
- * @function Highcharts.merge<T>
- *
- * @param {boolean} extend
- * Whether to extend the left-side object (a) or return a whole new
- * object.
- *
- * @param {T|undefined} a
- * The first object to extend. When only this is given, the function
- * returns a deep copy.
- *
- * @param {...Array<object|undefined>} [n]
- * An object to merge into the previous one.
- *
- * @return {T}
- * The merged object. If the first argument is true, the return is the
- * same as the second argument.
- */ /**
- * Utility function to deep merge two or more objects and return a third object.
- * The merge function can also be used with a single object argument to create a
- * deep copy of an object.
- *
- * @function Highcharts.merge<T>
- *
- * @param {T|undefined} a
- * The first object to extend. When only this is given, the function
- * returns a deep copy.
- *
- * @param {...Array<object|undefined>} [n]
- * An object to merge into the previous one.
- *
- * @return {T}
- * The merged object. If the first argument is true, the return is the
- * same as the second argument.
- */
- function merge() {
- /* eslint-enable valid-jsdoc */
- var i,
- args = arguments,
- len,
- ret = {},
- doCopy = function (copy,
- original) {
- // An object is replacing a primitive
- if (typeof copy !== 'object') {
- copy = {};
- }
- objectEach(original, function (value, key) {
- // Copy the contents of objects, but not arrays or DOM nodes
- if (isObject(value, true) &&
- !isClass(value) &&
- !isDOMElement(value)) {
- copy[key] = doCopy(copy[key] || {}, value);
- // Primitives and arrays are copied over directly
- }
- else {
- copy[key] = original[key];
- }
- });
- return copy;
- };
- // If first argument is true, copy into the existing object. Used in
- // setOptions.
- if (args[0] === true) {
- ret = args[1];
- args = Array.prototype.slice.call(args, 2);
- }
- // For each argument, extend the return
- len = args.length;
- for (i = 0; i < len; i++) {
- ret = doCopy(ret, args[i]);
- }
- return ret;
- }
- H.merge = merge;
- /**
- * Constrain a value to within a lower and upper threshold.
- *
- * @private
- * @param {number} value The initial value
- * @param {number} min The lower threshold
- * @param {number} max The upper threshold
- * @return {number} Returns a number value within min and max.
- */
- function clamp(value, min, max) {
- return value > min ? value < max ? value : max : min;
- }
- /**
- * Shortcut for parseInt
- *
- * @private
- * @function Highcharts.pInt
- *
- * @param {*} s
- * any
- *
- * @param {number} [mag]
- * Magnitude
- *
- * @return {number}
- * number
- */
- var pInt = H.pInt = function pInt(s,
- mag) {
- return parseInt(s,
- mag || 10);
- };
- /**
- * Utility function to check for string type.
- *
- * @function Highcharts.isString
- *
- * @param {*} s
- * The item to check.
- *
- * @return {boolean}
- * True if the argument is a string.
- */
- var isString = H.isString = function isString(s) {
- return typeof s === 'string';
- };
- /**
- * Utility function to check if an item is an array.
- *
- * @function Highcharts.isArray
- *
- * @param {*} obj
- * The item to check.
- *
- * @return {boolean}
- * True if the argument is an array.
- */
- var isArray = H.isArray = function isArray(obj) {
- var str = Object.prototype.toString.call(obj);
- return str === '[object Array]' || str === '[object Array Iterator]';
- };
- /**
- * Utility function to check if an item is of type object.
- *
- * @function Highcharts.isObject
- *
- * @param {*} obj
- * The item to check.
- *
- * @param {boolean} [strict=false]
- * Also checks that the object is not an array.
- *
- * @return {boolean}
- * True if the argument is an object.
- */
- function isObject(obj, strict) {
- return (!!obj &&
- typeof obj === 'object' &&
- (!strict || !isArray(obj))); // eslint-disable-line @typescript-eslint/no-explicit-any
- }
- H.isObject = isObject;
- /**
- * Utility function to check if an Object is a HTML Element.
- *
- * @function Highcharts.isDOMElement
- *
- * @param {*} obj
- * The item to check.
- *
- * @return {boolean}
- * True if the argument is a HTML Element.
- */
- var isDOMElement = H.isDOMElement = function isDOMElement(obj) {
- return isObject(obj) && typeof obj.nodeType === 'number';
- };
- /**
- * Utility function to check if an Object is a class.
- *
- * @function Highcharts.isClass
- *
- * @param {object|undefined} obj
- * The item to check.
- *
- * @return {boolean}
- * True if the argument is a class.
- */
- var isClass = H.isClass = function isClass(obj) {
- var c = obj && obj.constructor;
- return !!(isObject(obj, true) &&
- !isDOMElement(obj) &&
- (c && c.name && c.name !== 'Object'));
- };
- /**
- * Utility function to check if an item is a number and it is finite (not NaN,
- * Infinity or -Infinity).
- *
- * @function Highcharts.isNumber
- *
- * @param {*} n
- * The item to check.
- *
- * @return {boolean}
- * True if the item is a finite number
- */
- var isNumber = H.isNumber = function isNumber(n) {
- return typeof n === 'number' && !isNaN(n) && n < Infinity && n > -Infinity;
- };
- /**
- * Remove the last occurence of an item from an array.
- *
- * @function Highcharts.erase
- *
- * @param {Array<*>} arr
- * The array.
- *
- * @param {*} item
- * The item to remove.
- *
- * @return {void}
- */
- var erase = H.erase = function erase(arr,
- item) {
- var i = arr.length;
- while (i--) {
- if (arr[i] === item) {
- arr.splice(i, 1);
- break;
- }
- }
- };
- /**
- * Check if an object is null or undefined.
- *
- * @function Highcharts.defined
- *
- * @param {*} obj
- * The object to check.
- *
- * @return {boolean}
- * False if the object is null or undefined, otherwise true.
- */
- var defined = H.defined = function defined(obj) {
- return typeof obj !== 'undefined' && obj !== null;
- };
- /**
- * Set or get an attribute or an object of attributes. To use as a setter, pass
- * a key and a value, or let the second argument be a collection of keys and
- * values. To use as a getter, pass only a string as the second argument.
- *
- * @function Highcharts.attr
- *
- * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} elem
- * The DOM element to receive the attribute(s).
- *
- * @param {string|Highcharts.HTMLAttributes|Highcharts.SVGAttributes} [prop]
- * The property or an object of key-value pairs.
- *
- * @param {number|string} [value]
- * The value if a single property is set.
- *
- * @return {string|null|undefined}
- * When used as a getter, return the value.
- */
- function attr(elem, prop, value) {
- var ret;
- // if the prop is a string
- if (isString(prop)) {
- // set the value
- if (defined(value)) {
- elem.setAttribute(prop, value);
- // get the value
- }
- else if (elem && elem.getAttribute) {
- ret = elem.getAttribute(prop);
- // IE7 and below cannot get class through getAttribute (#7850)
- if (!ret && prop === 'class') {
- ret = elem.getAttribute(prop + 'Name');
- }
- }
- // else if prop is defined, it is a hash of key/value pairs
- }
- else {
- objectEach(prop, function (val, key) {
- elem.setAttribute(key, val);
- });
- }
- return ret;
- }
- H.attr = attr;
- /**
- * Check if an element is an array, and if not, make it into an array.
- *
- * @function Highcharts.splat
- *
- * @param {*} obj
- * The object to splat.
- *
- * @return {Array}
- * The produced or original array.
- */
- var splat = H.splat = function splat(obj) {
- return isArray(obj) ? obj : [obj];
- };
- /**
- * Set a timeout if the delay is given, otherwise perform the function
- * synchronously.
- *
- * @function Highcharts.syncTimeout
- *
- * @param {Function} fn
- * The function callback.
- *
- * @param {number} delay
- * Delay in milliseconds.
- *
- * @param {*} [context]
- * An optional context to send to the function callback.
- *
- * @return {number}
- * An identifier for the timeout that can later be cleared with
- * Highcharts.clearTimeout. Returns -1 if there is no timeout.
- */
- var syncTimeout = H.syncTimeout = function syncTimeout(fn,
- delay,
- context) {
- if (delay > 0) {
- return setTimeout(fn,
- delay,
- context);
- }
- fn.call(0, context);
- return -1;
- };
- /**
- * Internal clear timeout. The function checks that the `id` was not removed
- * (e.g. by `chart.destroy()`). For the details see
- * [issue #7901](https://github.com/highcharts/highcharts/issues/7901).
- *
- * @function Highcharts.clearTimeout
- *
- * @param {number} id
- * Id of a timeout.
- *
- * @return {void}
- */
- var internalClearTimeout = H.clearTimeout = function (id) {
- if (defined(id)) {
- clearTimeout(id);
- }
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Utility function to extend an object with the members of another.
- *
- * @function Highcharts.extend<T>
- *
- * @param {T|undefined} a
- * The object to be extended.
- *
- * @param {object} b
- * The object to add to the first one.
- *
- * @return {T}
- * Object a, the original object.
- */
- var extend = H.extend = function extend(a,
- b) {
- /* eslint-enable valid-jsdoc */
- var n;
- if (!a) {
- a = {};
- }
- for (n in b) { // eslint-disable-line guard-for-in
- a[n] = b[n];
- }
- return a;
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Return the first value that is not null or undefined.
- *
- * @function Highcharts.pick<T>
- *
- * @param {...Array<T|null|undefined>} items
- * Variable number of arguments to inspect.
- *
- * @return {T}
- * The value of the first argument that is not null or undefined.
- */
- function pick() {
- var args = arguments;
- var length = args.length;
- for (var i = 0; i < length; i++) {
- var arg = args[i];
- if (typeof arg !== 'undefined' && arg !== null) {
- return arg;
- }
- }
- }
- H.pick = pick;
- /**
- * Set CSS on a given element.
- *
- * @function Highcharts.css
- *
- * @param {Highcharts.HTMLDOMElement|Highcharts.SVGDOMElement} el
- * An HTML DOM element.
- *
- * @param {Highcharts.CSSObject} styles
- * Style object with camel case property names.
- *
- * @return {void}
- */
- var css = H.css = function css(el,
- styles) {
- if (H.isMS && !H.svg) { // #2686
- if (styles && typeof styles.opacity !== 'undefined') {
- styles.filter =
- 'alpha(opacity=' + (styles.opacity * 100) + ')';
- }
- }
- extend(el.style, styles);
- };
- /**
- * Utility function to create an HTML element with attributes and styles.
- *
- * @function Highcharts.createElement
- *
- * @param {string} tag
- * The HTML tag.
- *
- * @param {Highcharts.HTMLAttributes} [attribs]
- * Attributes as an object of key-value pairs.
- *
- * @param {Highcharts.CSSObject} [styles]
- * Styles as an object of key-value pairs.
- *
- * @param {Highcharts.HTMLDOMElement} [parent]
- * The parent HTML object.
- *
- * @param {boolean} [nopad=false]
- * If true, remove all padding, border and margin.
- *
- * @return {Highcharts.HTMLDOMElement}
- * The created DOM element.
- */
- var createElement = H.createElement = function createElement(tag,
- attribs,
- styles,
- parent,
- nopad) {
- var el = doc.createElement(tag);
- if (attribs) {
- extend(el, attribs);
- }
- if (nopad) {
- css(el, { padding: '0', border: 'none', margin: '0' });
- }
- if (styles) {
- css(el, styles);
- }
- if (parent) {
- parent.appendChild(el);
- }
- return el;
- };
- // eslint-disable-next-line valid-jsdoc
- /**
- * Extend a prototyped class by new members.
- *
- * @function Highcharts.extendClass<T>
- *
- * @param {Highcharts.Class<T>} parent
- * The parent prototype to inherit.
- *
- * @param {Highcharts.Dictionary<*>} members
- * A collection of prototype members to add or override compared to the
- * parent prototype.
- *
- * @return {Highcharts.Class<T>}
- * A new prototype.
- */
- var extendClass = H.extendClass = function extendClass(parent,
- members) {
- var obj = (function () { });
- obj.prototype = new parent(); // eslint-disable-line new-cap
- extend(obj.prototype, members);
- return obj;
- };
- /**
- * Left-pad a string to a given length by adding a character repetetively.
- *
- * @function Highcharts.pad
- *
- * @param {number} number
- * The input string or number.
- *
- * @param {number} [length]
- * The desired string length.
- *
- * @param {string} [padder=0]
- * The character to pad with.
- *
- * @return {string}
- * The padded string.
- */
- var pad = H.pad = function pad(number, length, padder) {
- return new Array((length || 2) +
- 1 -
- String(number)
- .replace('-', '')
- .length).join(padder || '0') + number;
- };
- /**
- * Return a length based on either the integer value, or a percentage of a base.
- *
- * @function Highcharts.relativeLength
- *
- * @param {Highcharts.RelativeSize} value
- * A percentage string or a number.
- *
- * @param {number} base
- * The full length that represents 100%.
- *
- * @param {number} [offset=0]
- * A pixel offset to apply for percentage values. Used internally in
- * axis positioning.
- *
- * @return {number}
- * The computed length.
- */
- var relativeLength = H.relativeLength = function relativeLength(value,
- base,
- offset) {
- return (/%$/).test(value) ?
- (base * parseFloat(value) / 100) + (offset || 0) :
- parseFloat(value);
- };
- /**
- * Wrap a method with extended functionality, preserving the original function.
- *
- * @function Highcharts.wrap
- *
- * @param {*} obj
- * The context object that the method belongs to. In real cases, this is
- * often a prototype.
- *
- * @param {string} method
- * The name of the method to extend.
- *
- * @param {Highcharts.WrapProceedFunction} func
- * A wrapper function callback. This function is called with the same
- * arguments as the original function, except that the original function
- * is unshifted and passed as the first argument.
- */
- var wrap = H.wrap = function wrap(obj,
- method,
- func) {
- var proceed = obj[method];
- obj[method] = function () {
- var args = Array.prototype.slice.call(arguments),
- outerArgs = arguments,
- ctx = this,
- ret;
- ctx.proceed = function () {
- proceed.apply(ctx, arguments.length ? arguments : outerArgs);
- };
- args.unshift(proceed);
- ret = func.apply(this, args);
- ctx.proceed = null;
- return ret;
- };
- };
- /**
- * Format a string according to a subset of the rules of Python's String.format
- * method.
- *
- * @example
- * var s = Highcharts.format(
- * 'The {color} fox was {len:.2f} feet long',
- * { color: 'red', len: Math.PI }
- * );
- * // => The red fox was 3.14 feet long
- *
- * @function Highcharts.format
- *
- * @param {string} str
- * The string to format.
- *
- * @param {Record<string, *>} ctx
- * The context, a collection of key-value pairs where each key is
- * replaced by its value.
- *
- * @param {Highcharts.Chart} [chart]
- * A `Chart` instance used to get numberFormatter and time.
- *
- * @return {string}
- * The formatted string.
- */
- var format = H.format = function (str,
- ctx,
- chart) {
- var splitter = '{',
- isInside = false,
- segment,
- valueAndFormat,
- ret = [],
- val,
- index;
- var floatRegex = /f$/;
- var decRegex = /\.([0-9])/;
- var lang = H.defaultOptions.lang;
- var time = chart && chart.time || H.time;
- var numberFormatter = chart && chart.numberFormatter || numberFormat;
- while (str) {
- index = str.indexOf(splitter);
- if (index === -1) {
- break;
- }
- segment = str.slice(0, index);
- if (isInside) { // we're on the closing bracket looking back
- valueAndFormat = segment.split(':');
- val = getNestedProperty(valueAndFormat.shift() || '', ctx);
- // Format the replacement
- if (valueAndFormat.length && typeof val === 'number') {
- segment = valueAndFormat.join(':');
- if (floatRegex.test(segment)) { // float
- var decimals = parseInt((segment.match(decRegex) || ['', '-1'])[1], 10);
- if (val !== null) {
- val = numberFormatter(val, decimals, lang.decimalPoint, segment.indexOf(',') > -1 ? lang.thousandsSep : '');
- }
- }
- else {
- val = time.dateFormat(segment, val);
- }
- }
- // Push the result and advance the cursor
- ret.push(val);
- }
- else {
- ret.push(segment);
- }
- str = str.slice(index + 1); // the rest
- isInside = !isInside; // toggle
- splitter = isInside ? '}' : '{'; // now look for next matching bracket
- }
- ret.push(str);
- return ret.join('');
- };
- /**
- * Get the magnitude of a number.
- *
- * @function Highcharts.getMagnitude
- *
- * @param {number} num
- * The number.
- *
- * @return {number}
- * The magnitude, where 1-9 are magnitude 1, 10-99 magnitude 2 etc.
- */
- var getMagnitude = H.getMagnitude = function (num) {
- return Math.pow(10,
- Math.floor(Math.log(num) / Math.LN10));
- };
- /**
- * Take an interval and normalize it to multiples of round numbers.
- *
- * @deprecated
- * @function Highcharts.normalizeTickInterval
- *
- * @param {number} interval
- * The raw, un-rounded interval.
- *
- * @param {Array<*>} [multiples]
- * Allowed multiples.
- *
- * @param {number} [magnitude]
- * The magnitude of the number.
- *
- * @param {boolean} [allowDecimals]
- * Whether to allow decimals.
- *
- * @param {boolean} [hasTickAmount]
- * If it has tickAmount, avoid landing on tick intervals lower than
- * original.
- *
- * @return {number}
- * The normalized interval.
- *
- * @todo
- * Move this function to the Axis prototype. It is here only for historical
- * reasons.
- */
- var normalizeTickInterval = H.normalizeTickInterval = function (interval,
- multiples,
- magnitude,
- allowDecimals,
- hasTickAmount) {
- var normalized,
- i,
- retInterval = interval;
- // round to a tenfold of 1, 2, 2.5 or 5
- magnitude = pick(magnitude, 1);
- normalized = interval / magnitude;
- // multiples for a linear scale
- if (!multiples) {
- multiples = hasTickAmount ?
- // Finer grained ticks when the tick amount is hard set, including
- // when alignTicks is true on multiple axes (#4580).
- [1, 1.2, 1.5, 2, 2.5, 3, 4, 5, 6, 8, 10] :
- // Else, let ticks fall on rounder numbers
- [1, 2, 2.5, 5, 10];
- // the allowDecimals option
- if (allowDecimals === false) {
- if (magnitude === 1) {
- multiples = multiples.filter(function (num) {
- return num % 1 === 0;
- });
- }
- else if (magnitude <= 0.1) {
- multiples = [1 / magnitude];
- }
- }
- }
- // normalize the interval to the nearest multiple
- for (i = 0; i < multiples.length; i++) {
- retInterval = multiples[i];
- // only allow tick amounts smaller than natural
- if ((hasTickAmount &&
- retInterval * magnitude >= interval) ||
- (!hasTickAmount &&
- (normalized <=
- (multiples[i] +
- (multiples[i + 1] || multiples[i])) / 2))) {
- break;
- }
- }
- // Multiply back to the correct magnitude. Correct floats to appropriate
- // precision (#6085).
- retInterval = correctFloat(retInterval * magnitude, -Math.round(Math.log(0.001) / Math.LN10));
- return retInterval;
- };
- /**
- * Sort an object array and keep the order of equal items. The ECMAScript
- * standard does not specify the behaviour when items are equal.
- *
- * @function Highcharts.stableSort
- *
- * @param {Array<*>} arr
- * The array to sort.
- *
- * @param {Function} sortFunction
- * The function to sort it with, like with regular Array.prototype.sort.
- *
- * @return {void}
- */
- var stableSort = H.stableSort = function stableSort(arr,
- sortFunction) {
- // @todo It seems like Chrome since v70 sorts in a stable way internally,
- // plus all other browsers do it, so over time we may be able to remove this
- // function
- var length = arr.length,
- sortValue,
- i;
- // Add index to each item
- for (i = 0; i < length; i++) {
- arr[i].safeI = i; // stable sort index
- }
- arr.sort(function (a, b) {
- sortValue = sortFunction(a, b);
- return sortValue === 0 ? a.safeI - b.safeI : sortValue;
- });
- // Remove index from items
- for (i = 0; i < length; i++) {
- delete arr[i].safeI; // stable sort index
- }
- };
- /**
- * Non-recursive method to find the lowest member of an array. `Math.min` raises
- * a maximum call stack size exceeded error in Chrome when trying to apply more
- * than 150.000 points. This method is slightly slower, but safe.
- *
- * @function Highcharts.arrayMin
- *
- * @param {Array<*>} data
- * An array of numbers.
- *
- * @return {number}
- * The lowest number.
- */
- var arrayMin = H.arrayMin = function arrayMin(data) {
- var i = data.length,
- min = data[0];
- while (i--) {
- if (data[i] < min) {
- min = data[i];
- }
- }
- return min;
- };
- /**
- * Non-recursive method to find the lowest member of an array. `Math.max` raises
- * a maximum call stack size exceeded error in Chrome when trying to apply more
- * than 150.000 points. This method is slightly slower, but safe.
- *
- * @function Highcharts.arrayMax
- *
- * @param {Array<*>} data
- * An array of numbers.
- *
- * @return {number}
- * The highest number.
- */
- var arrayMax = H.arrayMax = function arrayMax(data) {
- var i = data.length,
- max = data[0];
- while (i--) {
- if (data[i] > max) {
- max = data[i];
- }
- }
- return max;
- };
- /**
- * Utility method that destroys any SVGElement instances that are properties on
- * the given object. It loops all properties and invokes destroy if there is a
- * destroy method. The property is then delete.
- *
- * @function Highcharts.destroyObjectProperties
- *
- * @param {*} obj
- * The object to destroy properties on.
- *
- * @param {*} [except]
- * Exception, do not destroy this property, only delete it.
- *
- * @return {void}
- */
- var destroyObjectProperties = H.destroyObjectProperties =
- function destroyObjectProperties(obj,
- except) {
- objectEach(obj,
- function (val,
- n) {
- // If the object is non-null and destroy is defined
- if (val && val !== except && val.destroy) {
- // Invoke the destroy
- val.destroy();
- }
- // Delete the property from the object.
- delete obj[n];
- });
- };
- /**
- * Discard a HTML element by moving it to the bin and delete.
- *
- * @function Highcharts.discardElement
- *
- * @param {Highcharts.HTMLDOMElement} element
- * The HTML node to discard.
- *
- * @return {void}
- */
- var discardElement = H.discardElement = function discardElement(element) {
- var garbageBin = H.garbageBin;
- // create a garbage bin element, not part of the DOM
- if (!garbageBin) {
- garbageBin = createElement('div');
- }
- // move the node and empty bin
- if (element) {
- garbageBin.appendChild(element);
- }
- garbageBin.innerHTML = '';
- };
- /**
- * Fix JS round off float errors.
- *
- * @function Highcharts.correctFloat
- *
- * @param {number} num
- * A float number to fix.
- *
- * @param {number} [prec=14]
- * The precision.
- *
- * @return {number}
- * The corrected float number.
- */
- var correctFloat = H.correctFloat = function correctFloat(num,
- prec) {
- return parseFloat(num.toPrecision(prec || 14));
- };
- /**
- * Set the global animation to either a given value, or fall back to the given
- * chart's animation option.
- *
- * @function Highcharts.setAnimation
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>|undefined} animation
- * The animation object.
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @return {void}
- *
- * @todo
- * This function always relates to a chart, and sets a property on the renderer,
- * so it should be moved to the SVGRenderer.
- */
- var setAnimation = H.setAnimation = function setAnimation(animation,
- chart) {
- chart.renderer.globalAnimation = pick(animation,
- chart.options.chart.animation,
- true);
- };
- /**
- * Get the animation in object form, where a disabled animation is always
- * returned as `{ duration: 0 }`.
- *
- * @function Highcharts.animObject
- *
- * @param {boolean|Highcharts.AnimationOptionsObject} [animation=0]
- * An animation setting. Can be an object with duration, complete and
- * easing properties, or a boolean to enable or disable.
- *
- * @return {Highcharts.AnimationOptionsObject}
- * An object with at least a duration property.
- */
- var animObject = H.animObject = function animObject(animation) {
- return isObject(animation) ?
- H.merge({ duration: 500,
- defer: 0 },
- animation) :
- { duration: animation ? 500 : 0,
- defer: 0 };
- };
- /**
- * The time unit lookup
- *
- * @ignore
- */
- var timeUnits = H.timeUnits = {
- millisecond: 1,
- second: 1000,
- minute: 60000,
- hour: 3600000,
- day: 24 * 3600000,
- week: 7 * 24 * 3600000,
- month: 28 * 24 * 3600000,
- year: 364 * 24 * 3600000
- };
- /**
- * Format a number and return a string based on input settings.
- *
- * @sample highcharts/members/highcharts-numberformat/
- * Custom number format
- *
- * @function Highcharts.numberFormat
- *
- * @param {number} number
- * The input number to format.
- *
- * @param {number} decimals
- * The amount of decimals. A value of -1 preserves the amount in the
- * input number.
- *
- * @param {string} [decimalPoint]
- * The decimal point, defaults to the one given in the lang options, or
- * a dot.
- *
- * @param {string} [thousandsSep]
- * The thousands separator, defaults to the one given in the lang
- * options, or a space character.
- *
- * @return {string}
- * The formatted number.
- */
- var numberFormat = H.numberFormat = function numberFormat(number,
- decimals,
- decimalPoint,
- thousandsSep) {
- number = +number || 0;
- decimals = +decimals;
- var lang = H.defaultOptions.lang, origDec = (number.toString().split('.')[1] || '').split('e')[0].length, strinteger, thousands, ret, roundedNumber, exponent = number.toString().split('e'), fractionDigits;
- if (decimals === -1) {
- // Preserve decimals. Not huge numbers (#3793).
- decimals = Math.min(origDec, 20);
- }
- else if (!isNumber(decimals)) {
- decimals = 2;
- }
- else if (decimals && exponent[1] && exponent[1] < 0) {
- // Expose decimals from exponential notation (#7042)
- fractionDigits = decimals + +exponent[1];
- if (fractionDigits >= 0) {
- // remove too small part of the number while keeping the notation
- exponent[0] = (+exponent[0]).toExponential(fractionDigits)
- .split('e')[0];
- decimals = fractionDigits;
- }
- else {
- // fractionDigits < 0
- exponent[0] = exponent[0].split('.')[0] || 0;
- if (decimals < 20) {
- // use number instead of exponential notation (#7405)
- number = (exponent[0] * Math.pow(10, exponent[1]))
- .toFixed(decimals);
- }
- else {
- // or zero
- number = 0;
- }
- exponent[1] = 0;
- }
- }
- // Add another decimal to avoid rounding errors of float numbers. (#4573)
- // Then use toFixed to handle rounding.
- roundedNumber = (Math.abs(exponent[1] ? exponent[0] : number) +
- Math.pow(10, -Math.max(decimals, origDec) - 1)).toFixed(decimals);
- // A string containing the positive integer component of the number
- strinteger = String(pInt(roundedNumber));
- // Leftover after grouping into thousands. Can be 0, 1 or 2.
- thousands = strinteger.length > 3 ? strinteger.length % 3 : 0;
- // Language
- decimalPoint = pick(decimalPoint, lang.decimalPoint);
- thousandsSep = pick(thousandsSep, lang.thousandsSep);
- // Start building the return
- ret = number < 0 ? '-' : '';
- // Add the leftover after grouping into thousands. For example, in the
- // number 42 000 000, this line adds 42.
- ret += thousands ? strinteger.substr(0, thousands) + thousandsSep : '';
- // Add the remaining thousands groups, joined by the thousands separator
- ret += strinteger
- .substr(thousands)
- .replace(/(\d{3})(?=\d)/g, '$1' + thousandsSep);
- // Add the decimal point and the decimal component
- if (decimals) {
- // Get the decimal component
- ret += decimalPoint + roundedNumber.slice(-decimals);
- }
- if (exponent[1] && +ret !== 0) {
- ret += 'e' + exponent[1];
- }
- return ret;
- };
- /**
- * Easing definition
- *
- * @private
- * @function Math.easeInOutSine
- *
- * @param {number} pos
- * Current position, ranging from 0 to 1.
- *
- * @return {number}
- * Ease result
- */
- Math.easeInOutSine = function (pos) {
- return -0.5 * (Math.cos(Math.PI * pos) - 1);
- };
- /**
- * Returns the value of a property path on a given object.
- *
- * @private
- * @function getNestedProperty
- *
- * @param {string} path
- * Path to the property, for example `custom.myValue`.
- *
- * @param {unknown} obj
- * Instance containing the property on the specific path.
- *
- * @return {unknown}
- * The unknown property value.
- */
- function getNestedProperty(path, obj) {
- if (!path) {
- return obj;
- }
- var pathElements = path.split('.').reverse();
- var subProperty = obj;
- if (pathElements.length === 1) {
- return subProperty[path];
- }
- var pathElement = pathElements.pop();
- while (typeof pathElement !== 'undefined' &&
- typeof subProperty !== 'undefined' &&
- subProperty !== null) {
- subProperty = subProperty[pathElement];
- pathElement = pathElements.pop();
- }
- return subProperty;
- }
- /**
- * Get the computed CSS value for given element and property, only for numerical
- * properties. For width and height, the dimension of the inner box (excluding
- * padding) is returned. Used for fitting the chart within the container.
- *
- * @function Highcharts.getStyle
- *
- * @param {Highcharts.HTMLDOMElement} el
- * An HTML element.
- *
- * @param {string} prop
- * The property name.
- *
- * @param {boolean} [toInt=true]
- * Parse to integer.
- *
- * @return {number|string}
- * The numeric value.
- */
- var getStyle = H.getStyle = function (el,
- prop,
- toInt) {
- var style;
- // For width and height, return the actual inner pixel size (#4913)
- if (prop === 'width') {
- var offsetWidth = Math.min(el.offsetWidth,
- el.scrollWidth);
- // In flex boxes, we need to use getBoundingClientRect and floor it,
- // because scrollWidth doesn't support subpixel precision (#6427) ...
- var boundingClientRectWidth = el.getBoundingClientRect &&
- el.getBoundingClientRect().width;
- // ...unless if the containing div or its parents are transform-scaled
- // down, in which case the boundingClientRect can't be used as it is
- // also scaled down (#9871, #10498).
- if (boundingClientRectWidth < offsetWidth &&
- boundingClientRectWidth >= offsetWidth - 1) {
- offsetWidth = Math.floor(boundingClientRectWidth);
- }
- return Math.max(0, // #8377
- (offsetWidth -
- H.getStyle(el, 'padding-left') -
- H.getStyle(el, 'padding-right')));
- }
- if (prop === 'height') {
- return Math.max(0, // #8377
- Math.min(el.offsetHeight, el.scrollHeight) -
- H.getStyle(el, 'padding-top') -
- H.getStyle(el, 'padding-bottom'));
- }
- if (!win.getComputedStyle) {
- // SVG not supported, forgot to load oldie.js?
- error(27, true);
- }
- // Otherwise, get the computed style
- style = win.getComputedStyle(el, undefined); // eslint-disable-line no-undefined
- if (style) {
- style = style.getPropertyValue(prop);
- if (pick(toInt, prop !== 'opacity')) {
- style = pInt(style);
- }
- }
- return style;
- };
- /**
- * Get the defer as a number value from series animation options.
- *
- * @function Highcharts.getDeferredAnimation
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @return {number}
- * The numeric value.
- */
- var getDeferredAnimation = H.getDeferredAnimation = function (chart,
- animation,
- series) {
- var labelAnimation = animObject(animation);
- var s = series ? [series] : chart.series;
- var defer = 0;
- var duration = 0;
- s.forEach(function (series) {
- var seriesAnim = animObject(series.options.animation);
- defer = animation && defined(animation.defer) ?
- labelAnimation.defer :
- Math.max(defer, seriesAnim.duration + seriesAnim.defer);
- duration = Math.min(labelAnimation.duration, seriesAnim.duration);
- });
- // Disable defer for exporting
- if (chart.renderer.forExport) {
- defer = 0;
- }
- var anim = {
- defer: Math.max(0,
- defer - duration),
- duration: Math.min(defer,
- duration)
- };
- return anim;
- };
- /**
- * Search for an item in an array.
- *
- * @function Highcharts.inArray
- *
- * @deprecated
- *
- * @param {*} item
- * The item to search for.
- *
- * @param {Array<*>} arr
- * The array or node collection to search in.
- *
- * @param {number} [fromIndex=0]
- * The index to start searching from.
- *
- * @return {number}
- * The index within the array, or -1 if not found.
- */
- var inArray = H.inArray = function (item,
- arr,
- fromIndex) {
- error(32,
- false,
- void 0, { 'Highcharts.inArray': 'use Array.indexOf' });
- return arr.indexOf(item, fromIndex);
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Return the value of the first element in the array that satisfies the
- * provided testing function.
- *
- * @function Highcharts.find<T>
- *
- * @param {Array<T>} arr
- * The array to test.
- *
- * @param {Function} callback
- * The callback function. The function receives the item as the first
- * argument. Return `true` if this item satisfies the condition.
- *
- * @return {T|undefined}
- * The value of the element.
- */
- var find = H.find = Array.prototype.find ?
- /* eslint-enable valid-jsdoc */
- function (arr,
- callback) {
- return arr.find(callback);
- } :
- // Legacy implementation. PhantomJS, IE <= 11 etc. #7223.
- function (arr, callback) {
- var i,
- length = arr.length;
- for (i = 0; i < length; i++) {
- if (callback(arr[i], i)) { // eslint-disable-line callback-return
- return arr[i];
- }
- }
- };
- /**
- * Returns an array of a given object's own properties.
- *
- * @function Highcharts.keys
- * @deprecated
- *
- * @param {*} obj
- * The object of which the properties are to be returned.
- *
- * @return {Array<string>}
- * An array of strings that represents all the properties.
- */
- H.keys = function (obj) {
- error(32, false, void 0, { 'Highcharts.keys': 'use Object.keys' });
- return Object.keys(obj);
- };
- /**
- * Get the element's offset position, corrected for `overflow: auto`.
- *
- * @function Highcharts.offset
- *
- * @param {global.Element} el
- * The DOM element.
- *
- * @return {Highcharts.OffsetObject}
- * An object containing `left` and `top` properties for the position in
- * the page.
- */
- var offset = H.offset = function offset(el) {
- var docElem = doc.documentElement,
- box = (el.parentElement || el.parentNode) ?
- el.getBoundingClientRect() :
- { top: 0,
- left: 0 };
- return {
- top: box.top + (win.pageYOffset || docElem.scrollTop) -
- (docElem.clientTop || 0),
- left: box.left + (win.pageXOffset || docElem.scrollLeft) -
- (docElem.clientLeft || 0)
- };
- };
- /**
- * Stop running animation.
- *
- * @function Highcharts.stop
- *
- * @param {Highcharts.SVGElement} el
- * The SVGElement to stop animation on.
- *
- * @param {string} [prop]
- * The property to stop animating. If given, the stop method will stop a
- * single property from animating, while others continue.
- *
- * @return {void}
- *
- * @todo
- * A possible extension to this would be to stop a single property, when
- * we want to continue animating others. Then assign the prop to the timer
- * in the Fx.run method, and check for the prop here. This would be an
- * improvement in all cases where we stop the animation from .attr. Instead of
- * stopping everything, we can just stop the actual attributes we're setting.
- */
- var stop = H.stop = function (el,
- prop) {
- var i = H.timers.length;
- // Remove timers related to this element (#4519)
- while (i--) {
- if (H.timers[i].elem === el && (!prop || prop === H.timers[i].prop)) {
- H.timers[i].stopped = true; // #4667
- }
- }
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Iterate over object key pairs in an object.
- *
- * @function Highcharts.objectEach<T>
- *
- * @param {*} obj
- * The object to iterate over.
- *
- * @param {Highcharts.ObjectEachCallbackFunction<T>} fn
- * The iterator callback. It passes three arguments:
- * * value - The property value.
- * * key - The property key.
- * * obj - The object that objectEach is being applied to.
- *
- * @param {T} [ctx]
- * The context.
- *
- * @return {void}
- */
- var objectEach = H.objectEach = function objectEach(obj,
- fn,
- ctx) {
- /* eslint-enable valid-jsdoc */
- for (var key in obj) {
- if (Object.hasOwnProperty.call(obj,
- key)) {
- fn.call(ctx || obj[key],
- obj[key],
- key,
- obj);
- }
- }
- };
- /**
- * Iterate over an array.
- *
- * @deprecated
- * @function Highcharts.each
- *
- * @param {Array<*>} arr
- * The array to iterate over.
- *
- * @param {Function} fn
- * The iterator callback. It passes three arguments:
- * - `item`: The array item.
- * - `index`: The item's index in the array.
- * - `arr`: The array that each is being applied to.
- *
- * @param {*} [ctx]
- * The context.
- *
- * @return {void}
- */
- /**
- * Filter an array by a callback.
- *
- * @deprecated
- * @function Highcharts.grep
- *
- * @param {Array<*>} arr
- * The array to filter.
- *
- * @param {Function} callback
- * The callback function. The function receives the item as the first
- * argument. Return `true` if the item is to be preserved.
- *
- * @return {Array<*>}
- * A new, filtered array.
- */
- /**
- * Map an array by a callback.
- *
- * @deprecated
- * @function Highcharts.map
- *
- * @param {Array<*>} arr
- * The array to map.
- *
- * @param {Function} fn
- * The callback function. Return the new value for the new array.
- *
- * @return {Array<*>}
- * A new array item with modified items.
- */
- /**
- * Reduce an array to a single value.
- *
- * @deprecated
- * @function Highcharts.reduce
- *
- * @param {Array<*>} arr
- * The array to reduce.
- *
- * @param {Function} fn
- * The callback function. Return the reduced value. Receives 4
- * arguments: Accumulated/reduced value, current value, current array
- * index, and the array.
- *
- * @param {*} initialValue
- * The initial value of the accumulator.
- *
- * @return {*}
- * The reduced value.
- */
- /**
- * Test whether at least one element in the array passes the test implemented by
- * the provided function.
- *
- * @deprecated
- * @function Highcharts.some
- *
- * @param {Array<*>} arr
- * The array to test
- *
- * @param {Function} fn
- * The function to run on each item. Return truty to pass the test.
- * Receives arguments `currentValue`, `index` and `array`.
- *
- * @param {*} ctx
- * The context.
- *
- * @return {boolean}
- */
- objectEach({
- map: 'map',
- each: 'forEach',
- grep: 'filter',
- reduce: 'reduce',
- some: 'some'
- }, function (val, key) {
- H[key] = function (arr) {
- var _a;
- error(32, false, void 0, (_a = {}, _a["Highcharts." + key] = "use Array." + val, _a));
- return Array.prototype[val].apply(arr, [].slice.call(arguments, 1));
- };
- });
- /* eslint-disable valid-jsdoc */
- /**
- * Add an event listener.
- *
- * @function Highcharts.addEvent<T>
- *
- * @param {Highcharts.Class<T>|T} el
- * The element or object to add a listener to. It can be a
- * {@link HTMLDOMElement}, an {@link SVGElement} or any other object.
- *
- * @param {string} type
- * The event type.
- *
- * @param {Highcharts.EventCallbackFunction<T>|Function} fn
- * The function callback to execute when the event is fired.
- *
- * @param {Highcharts.EventOptionsObject} [options]
- * Options for adding the event.
- *
- * @return {Function}
- * A callback function to remove the added event.
- */
- var addEvent = H.addEvent = function (el,
- type,
- fn,
- options) {
- if (options === void 0) { options = {}; }
- /* eslint-enable valid-jsdoc */
- var events,
- addEventListener = (el.addEventListener || H.addEventListenerPolyfill);
- // If we're setting events directly on the constructor, use a separate
- // collection, `protoEvents` to distinguish it from the item events in
- // `hcEvents`.
- if (typeof el === 'function' && el.prototype) {
- events = el.prototype.protoEvents = el.prototype.protoEvents || {};
- }
- else {
- events = el.hcEvents = el.hcEvents || {};
- }
- // Allow click events added to points, otherwise they will be prevented by
- // the TouchPointer.pinch function after a pinch zoom operation (#7091).
- if (H.Point &&
- el instanceof H.Point &&
- el.series &&
- el.series.chart) {
- el.series.chart.runTrackerClick = true;
- }
- // Handle DOM events
- if (addEventListener) {
- addEventListener.call(el, type, fn, false);
- }
- if (!events[type]) {
- events[type] = [];
- }
- var eventObject = {
- fn: fn,
- order: typeof options.order === 'number' ? options.order : Infinity
- };
- events[type].push(eventObject);
- // Order the calls
- events[type].sort(function (a, b) {
- return a.order - b.order;
- });
- // Return a function that can be called to remove this event.
- return function () {
- removeEvent(el, type, fn);
- };
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Remove an event that was added with {@link Highcharts#addEvent}.
- *
- * @function Highcharts.removeEvent<T>
- *
- * @param {Highcharts.Class<T>|T} el
- * The element to remove events on.
- *
- * @param {string} [type]
- * The type of events to remove. If undefined, all events are removed
- * from the element.
- *
- * @param {Highcharts.EventCallbackFunction<T>} [fn]
- * The specific callback to remove. If undefined, all events that match
- * the element and optionally the type are removed.
- *
- * @return {void}
- */
- var removeEvent = H.removeEvent = function removeEvent(el,
- type,
- fn) {
- /* eslint-enable valid-jsdoc */
- var events;
- /**
- * @private
- * @param {string} type - event type
- * @param {Highcharts.EventCallbackFunction<T>} fn - callback
- * @return {void}
- */
- function removeOneEvent(type, fn) {
- var removeEventListener = (el.removeEventListener || H.removeEventListenerPolyfill);
- if (removeEventListener) {
- removeEventListener.call(el, type, fn, false);
- }
- }
- /**
- * @private
- * @param {any} eventCollection - collection
- * @return {void}
- */
- function removeAllEvents(eventCollection) {
- var types,
- len;
- if (!el.nodeName) {
- return; // break on non-DOM events
- }
- if (type) {
- types = {};
- types[type] = true;
- }
- else {
- types = eventCollection;
- }
- objectEach(types, function (_val, n) {
- if (eventCollection[n]) {
- len = eventCollection[n].length;
- while (len--) {
- removeOneEvent(n, eventCollection[n][len].fn);
- }
- }
- });
- }
- ['protoEvents', 'hcEvents'].forEach(function (coll, i) {
- var eventElem = i ? el : el.prototype;
- var eventCollection = eventElem && eventElem[coll];
- if (eventCollection) {
- if (type) {
- events = (eventCollection[type] || []);
- if (fn) {
- eventCollection[type] = events.filter(function (obj) {
- return fn !== obj.fn;
- });
- removeOneEvent(type, fn);
- }
- else {
- removeAllEvents(eventCollection);
- eventCollection[type] = [];
- }
- }
- else {
- removeAllEvents(eventCollection);
- eventElem[coll] = {};
- }
- }
- });
- };
- /* eslint-disable valid-jsdoc */
- /**
- * Fire an event that was registered with {@link Highcharts#addEvent}.
- *
- * @function Highcharts.fireEvent<T>
- *
- * @param {T} el
- * The object to fire the event on. It can be a {@link HTMLDOMElement},
- * an {@link SVGElement} or any other object.
- *
- * @param {string} type
- * The type of event.
- *
- * @param {Highcharts.Dictionary<*>|Event} [eventArguments]
- * Custom event arguments that are passed on as an argument to the event
- * handler.
- *
- * @param {Highcharts.EventCallbackFunction<T>|Function} [defaultFunction]
- * The default function to execute if the other listeners haven't
- * returned false.
- *
- * @return {void}
- */
- var fireEvent = H.fireEvent = function (el,
- type,
- eventArguments,
- defaultFunction) {
- /* eslint-enable valid-jsdoc */
- var e,
- i;
- eventArguments = eventArguments || {};
- if (doc.createEvent &&
- (el.dispatchEvent || el.fireEvent)) {
- e = doc.createEvent('Events');
- e.initEvent(type, true, true);
- extend(e, eventArguments);
- if (el.dispatchEvent) {
- el.dispatchEvent(e);
- }
- else {
- el.fireEvent(type, e);
- }
- }
- else {
- if (!eventArguments.target) {
- // We're running a custom event
- extend(eventArguments, {
- // Attach a simple preventDefault function to skip
- // default handler if called. The built-in
- // defaultPrevented property is not overwritable (#5112)
- preventDefault: function () {
- eventArguments.defaultPrevented = true;
- },
- // Setting target to native events fails with clicking
- // the zoom-out button in Chrome.
- target: el,
- // If the type is not set, we're running a custom event
- // (#2297). If it is set, we're running a browser event,
- // and setting it will cause en error in IE8 (#2465).
- type: type
- });
- }
- var fireInOrder = function (protoEvents,
- hcEvents) {
- if (protoEvents === void 0) { protoEvents = []; }
- if (hcEvents === void 0) { hcEvents = []; }
- var iA = 0;
- var iB = 0;
- var length = protoEvents.length + hcEvents.length;
- for (i = 0; i < length; i++) {
- var obj = (!protoEvents[iA] ?
- hcEvents[iB++] :
- !hcEvents[iB] ?
- protoEvents[iA++] :
- protoEvents[iA].order <= hcEvents[iB].order ?
- protoEvents[iA++] :
- hcEvents[iB++]);
- // If the event handler return false, prevent the default
- // handler from executing
- if (obj.fn.call(el, eventArguments) === false) {
- eventArguments.preventDefault();
- }
- }
- };
- fireInOrder(el.protoEvents && el.protoEvents[type], el.hcEvents && el.hcEvents[type]);
- }
- // Run the default if not prevented
- if (defaultFunction && !eventArguments.defaultPrevented) {
- defaultFunction.call(el, eventArguments);
- }
- };
- /**
- * The global animate method, which uses Fx to create individual animators.
- *
- * @function Highcharts.animate
- *
- * @param {Highcharts.HTMLDOMElement|Highcharts.SVGElement} el
- * The element to animate.
- *
- * @param {Highcharts.CSSObject|Highcharts.SVGAttributes} params
- * An object containing key-value pairs of the properties to animate.
- * Supports numeric as pixel-based CSS properties for HTML objects and
- * attributes for SVGElements.
- *
- * @param {Partial<Highcharts.AnimationOptionsObject>} [opt]
- * Animation options.
- *
- * @return {void}
- */
- var animate = H.animate = function (el,
- params,
- opt) {
- var start,
- unit = '',
- end,
- fx,
- args;
- if (!isObject(opt)) { // Number or undefined/null
- args = arguments;
- opt = {
- duration: args[2],
- easing: args[3],
- complete: args[4]
- };
- }
- if (!isNumber(opt.duration)) {
- opt.duration = 400;
- }
- opt.easing = typeof opt.easing === 'function' ?
- opt.easing :
- (Math[opt.easing] || Math.easeInOutSine);
- opt.curAnim = merge(params);
- objectEach(params, function (val, prop) {
- // Stop current running animation of this property
- stop(el, prop);
- fx = new Fx(el, opt, prop);
- end = null;
- if (prop === 'd' && isArray(params.d)) {
- fx.paths = fx.initPath(el, el.pathArray, params.d);
- fx.toD = params.d;
- start = 0;
- end = 1;
- }
- else if (el.attr) {
- start = el.attr(prop);
- }
- else {
- start = parseFloat(getStyle(el, prop)) || 0;
- if (prop !== 'opacity') {
- unit = 'px';
- }
- }
- if (!end) {
- end = val;
- }
- if (end && end.match && end.match('px')) {
- end = end.replace(/px/g, ''); // #4351
- }
- fx.run(start, end, unit);
- });
- };
- /**
- * Factory to create new series prototypes.
- *
- * @function Highcharts.seriesType
- *
- * @param {string} type
- * The series type name.
- *
- * @param {string} parent
- * The parent series type name. Use `line` to inherit from the basic
- * {@link Series} object.
- *
- * @param {Highcharts.SeriesOptionsType|Highcharts.Dictionary<*>} options
- * The additional default options that are merged with the parent's
- * options.
- *
- * @param {Highcharts.Dictionary<*>} [props]
- * The properties (functions and primitives) to set on the new
- * prototype.
- *
- * @param {Highcharts.Dictionary<*>} [pointProps]
- * Members for a series-specific extension of the {@link Point}
- * prototype if needed.
- *
- * @return {Highcharts.Series}
- * The newly created prototype as extended from {@link Series} or its
- * derivatives.
- */
- // docs: add to API + extending Highcharts
- var seriesType = H.seriesType = function (type,
- parent,
- options,
- props,
- pointProps) {
- var defaultOptions = getOptions(),
- seriesTypes = H.seriesTypes;
- // Merge the options
- defaultOptions.plotOptions[type] = merge(defaultOptions.plotOptions[parent], options);
- // Create the class
- seriesTypes[type] = extendClass(seriesTypes[parent] || function () { }, props);
- seriesTypes[type].prototype.type = type;
- // Create the point class if needed
- if (pointProps) {
- seriesTypes[type].prototype.pointClass =
- extendClass(H.Point, pointProps);
- }
- return seriesTypes[type];
- };
- var serialMode;
- /**
- * Get a unique key for using in internal element id's and pointers. The key is
- * composed of a random hash specific to this Highcharts instance, and a
- * counter.
- *
- * @example
- * var id = uniqueKey(); // => 'highcharts-x45f6hp-0'
- *
- * @function Highcharts.uniqueKey
- *
- * @return {string}
- * A unique key.
- */
- var uniqueKey = H.uniqueKey = (function () {
- var hash = Math.random().toString(36).substring(2, 9) + '-';
- var id = 0;
- return function () {
- return 'highcharts-' + (serialMode ? '' : hash) + id++;
- };
- }());
- /**
- * Activates a serial mode for element IDs provided by
- * {@link Highcharts.uniqueKey}. This mode can be used in automated tests, where
- * a simple comparison of two rendered SVG graphics is needed.
- *
- * **Note:** This is only for testing purposes and will break functionality in
- * webpages with multiple charts.
- *
- * @example
- * if (
- * process &&
- * process.env.NODE_ENV === 'development'
- * ) {
- * Highcharts.useSerialIds(true);
- * }
- *
- * @function Highcharts.useSerialIds
- *
- * @param {boolean} [mode]
- * Changes the state of serial mode.
- *
- * @return {boolean|undefined}
- * State of the serial mode.
- */
- var useSerialIds = H.useSerialIds = function (mode) {
- return (serialMode = pick(mode,
- serialMode));
- };
- var isFunction = H.isFunction = function (obj) {
- return typeof obj === 'function';
- };
- /**
- * Get the updated default options. Until 3.0.7, merely exposing defaultOptions
- * for outside modules wasn't enough because the setOptions method created a new
- * object.
- *
- * @function Highcharts.getOptions
- *
- * @return {Highcharts.Options}
- */
- var getOptions = H.getOptions = function () {
- return H.defaultOptions;
- };
- /**
- * Merge the default options with custom options and return the new options
- * structure. Commonly used for defining reusable templates.
- *
- * @sample highcharts/global/useutc-false Setting a global option
- * @sample highcharts/members/setoptions Applying a global theme
- *
- * @function Highcharts.setOptions
- *
- * @param {Highcharts.Options} options
- * The new custom chart options.
- *
- * @return {Highcharts.Options}
- * Updated options.
- */
- var setOptions = H.setOptions = function (options) {
- // Copy in the default options
- H.defaultOptions = merge(true,
- H.defaultOptions,
- options);
- // Update the time object
- if (options.time || options.global) {
- H.time.update(merge(H.defaultOptions.global, H.defaultOptions.time, options.global, options.time));
- }
- return H.defaultOptions;
- };
- // Register Highcharts as a plugin in jQuery
- if (win.jQuery) {
- /**
- * Highcharts-extended JQuery.
- *
- * @external JQuery
- */
- /**
- * Helper function to return the chart of the current JQuery selector
- * element.
- *
- * @function external:JQuery#highcharts
- *
- * @return {Highcharts.Chart}
- * The chart that is linked to the JQuery selector element.
- */ /**
- * Factory function to create a chart in the current JQuery selector
- * element.
- *
- * @function external:JQuery#highcharts
- *
- * @param {'Chart'|'Map'|'StockChart'|string} [className]
- * Name of the factory class in the Highcharts namespace.
- *
- * @param {Highcharts.Options} [options]
- * The chart options structure.
- *
- * @param {Highcharts.ChartCallbackFunction} [callback]
- * Function to run when the chart has loaded and and all external
- * images are loaded. Defining a
- * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load)
- * handler is equivalent.
- *
- * @return {JQuery}
- * The current JQuery selector.
- */
- win.jQuery.fn.highcharts = function () {
- var args = [].slice.call(arguments);
- if (this[0]) { // this[0] is the renderTo div
- // Create the chart
- if (args[0]) {
- new H[ // eslint-disable-line computed-property-spacing, no-new
- // Constructor defaults to Chart
- isString(args[0]) ? args.shift() : 'Chart'](this[0], args[0], args[1]);
- return this;
- }
- // When called without parameters or with the return argument,
- // return an existing chart
- return charts[attr(this[0], 'data-highcharts-chart')];
- }
- };
- }
- // TODO use named exports when supported.
- var utilitiesModule = {
- Fx: H.Fx,
- addEvent: addEvent,
- animate: animate,
- animObject: animObject,
- arrayMax: arrayMax,
- arrayMin: arrayMin,
- attr: attr,
- clamp: clamp,
- clearTimeout: internalClearTimeout,
- correctFloat: correctFloat,
- createElement: createElement,
- css: css,
- defined: defined,
- destroyObjectProperties: destroyObjectProperties,
- discardElement: discardElement,
- erase: erase,
- error: error,
- extend: extend,
- extendClass: extendClass,
- find: find,
- fireEvent: fireEvent,
- format: format,
- getDeferredAnimation: getDeferredAnimation,
- getMagnitude: getMagnitude,
- getNestedProperty: getNestedProperty,
- getOptions: getOptions,
- getStyle: getStyle,
- inArray: inArray,
- isArray: isArray,
- isClass: isClass,
- isDOMElement: isDOMElement,
- isFunction: isFunction,
- isNumber: isNumber,
- isObject: isObject,
- isString: isString,
- merge: merge,
- normalizeTickInterval: normalizeTickInterval,
- numberFormat: numberFormat,
- objectEach: objectEach,
- offset: offset,
- pad: pad,
- pick: pick,
- pInt: pInt,
- relativeLength: relativeLength,
- removeEvent: removeEvent,
- seriesType: seriesType,
- setAnimation: setAnimation,
- setOptions: setOptions,
- splat: splat,
- stableSort: stableSort,
- stop: stop,
- syncTimeout: syncTimeout,
- timeUnits: timeUnits,
- uniqueKey: uniqueKey,
- useSerialIds: useSerialIds,
- wrap: wrap
- };
- return utilitiesModule;
- });
- _registerModule(_modules, 'Core/Color.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * A valid color to be parsed and handled by Highcharts. Highcharts internally
- * supports hex colors like `#ffffff`, rgb colors like `rgb(255,255,255)` and
- * rgba colors like `rgba(255,255,255,1)`. Other colors may be supported by the
- * browsers and displayed correctly, but Highcharts is not able to process them
- * and apply concepts like opacity and brightening.
- *
- * @typedef {string} Highcharts.ColorString
- */
- /**
- * A valid color type than can be parsed and handled by Highcharts. It can be a
- * color string, a gradient object, or a pattern object.
- *
- * @typedef {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} Highcharts.ColorType
- */
- /**
- * Gradient options instead of a solid color.
- *
- * @example
- * // Linear gradient used as a color option
- * color: {
- * linearGradient: { x1: 0, x2: 0, y1: 0, y2: 1 },
- * stops: [
- * [0, '#003399'], // start
- * [0.5, '#ffffff'], // middle
- * [1, '#3366AA'] // end
- * ]
- * }
- *
- * @interface Highcharts.GradientColorObject
- */ /**
- * Holds an object that defines the start position and the end position relative
- * to the shape.
- * @name Highcharts.GradientColorObject#linearGradient
- * @type {Highcharts.LinearGradientColorObject|undefined}
- */ /**
- * Holds an object that defines the center position and the radius.
- * @name Highcharts.GradientColorObject#radialGradient
- * @type {Highcharts.RadialGradientColorObject|undefined}
- */ /**
- * The first item in each tuple is the position in the gradient, where 0 is the
- * start of the gradient and 1 is the end of the gradient. Multiple stops can be
- * applied. The second item is the color for each stop. This color can also be
- * given in the rgba format.
- * @name Highcharts.GradientColorObject#stops
- * @type {Array<Highcharts.GradientColorStopObject>}
- */
- /**
- * Color stop tuple.
- *
- * @see Highcharts.GradientColorObject
- *
- * @interface Highcharts.GradientColorStopObject
- */ /**
- * @name Highcharts.GradientColorStopObject#0
- * @type {number}
- */ /**
- * @name Highcharts.GradientColorStopObject#1
- * @type {Highcharts.ColorString}
- */ /**
- * @name Highcharts.GradientColorStopObject#color
- * @type {Highcharts.Color|undefined}
- */
- /**
- * Defines the start position and the end position for a gradient relative
- * to the shape. Start position (x1, y1) and end position (x2, y2) are relative
- * to the shape, where 0 means top/left and 1 is bottom/right.
- *
- * @interface Highcharts.LinearGradientColorObject
- */ /**
- * Start horizontal position of the gradient. Float ranges 0-1.
- * @name Highcharts.LinearGradientColorObject#x1
- * @type {number}
- */ /**
- * End horizontal position of the gradient. Float ranges 0-1.
- * @name Highcharts.LinearGradientColorObject#x2
- * @type {number}
- */ /**
- * Start vertical position of the gradient. Float ranges 0-1.
- * @name Highcharts.LinearGradientColorObject#y1
- * @type {number}
- */ /**
- * End vertical position of the gradient. Float ranges 0-1.
- * @name Highcharts.LinearGradientColorObject#y2
- * @type {number}
- */
- /**
- * Defines the center position and the radius for a gradient.
- *
- * @interface Highcharts.RadialGradientColorObject
- */ /**
- * Center horizontal position relative to the shape. Float ranges 0-1.
- * @name Highcharts.RadialGradientColorObject#cx
- * @type {number}
- */ /**
- * Center vertical position relative to the shape. Float ranges 0-1.
- * @name Highcharts.RadialGradientColorObject#cy
- * @type {number}
- */ /**
- * Radius relative to the shape. Float ranges 0-1.
- * @name Highcharts.RadialGradientColorObject#r
- * @type {number}
- */
- var isNumber = U.isNumber,
- merge = U.merge,
- pInt = U.pInt;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Handle color operations. Some object methods are chainable.
- *
- * @class
- * @name Highcharts.Color
- *
- * @param {Highcharts.ColorType} input
- * The input color in either rbga or hex format
- */
- var Color = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Color(input) {
- // Collection of parsers. This can be extended from the outside by pushing
- // parsers to Highcharts.Color.prototype.parsers.
- this.parsers = [{
- // RGBA color
- // eslint-disable-next-line max-len
- regex: /rgba\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]?(?:\.[0-9]+)?)\s*\)/,
- parse: function (result) {
- return [
- pInt(result[1]),
- pInt(result[2]),
- pInt(result[3]),
- parseFloat(result[4], 10)
- ];
- }
- }, {
- // RGB color
- regex: /rgb\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*\)/,
- parse: function (result) {
- return [pInt(result[1]), pInt(result[2]), pInt(result[3]), 1];
- }
- }];
- this.rgba = [];
- // Backwards compatibility, allow instanciation without new (#13053)
- if (!(this instanceof Color)) {
- return new Color(input);
- }
- this.init(input);
- }
- /* *
- *
- * Static Functions
- *
- * */
- /**
- * Creates a color instance out of a color string or object.
- *
- * @function Highcharts.Color.parse
- *
- * @param {Highcharts.ColorType} input
- * The input color in either rbga or hex format.
- *
- * @return {Highcharts.Color}
- * Color instance.
- */
- Color.parse = function (input) {
- return new Color(input);
- };
- /* *
- *
- * Functions
- *
- * */
- /**
- * Parse the input color to rgba array
- *
- * @private
- * @function Highcharts.Color#init
- *
- * @param {Highcharts.ColorType} input
- * The input color in either rbga or hex format
- *
- * @return {void}
- */
- Color.prototype.init = function (input) {
- var result,
- rgba,
- i,
- parser,
- len;
- this.input = input = Color.names[input && input.toLowerCase ?
- input.toLowerCase() :
- ''] || input;
- // Gradients
- if (input && input.stops) {
- this.stops = input.stops.map(function (stop) {
- return new Color(stop[1]);
- });
- // Solid colors
- }
- else {
- // Bitmasking as input[0] is not working for legacy IE.
- if (input &&
- input.charAt &&
- input.charAt() === '#') {
- len = input.length;
- input = parseInt(input.substr(1), 16);
- // Handle long-form, e.g. #AABBCC
- if (len === 7) {
- rgba = [
- (input & 0xFF0000) >> 16,
- (input & 0xFF00) >> 8,
- (input & 0xFF),
- 1
- ];
- // Handle short-form, e.g. #ABC
- // In short form, the value is assumed to be the same
- // for both nibbles for each component. e.g. #ABC = #AABBCC
- }
- else if (len === 4) {
- rgba = [
- (((input & 0xF00) >> 4) |
- (input & 0xF00) >> 8),
- (((input & 0xF0) >> 4) |
- (input & 0xF0)),
- ((input & 0xF) << 4) | (input & 0xF),
- 1
- ];
- }
- }
- // Otherwise, check regex parsers
- if (!rgba) {
- i = this.parsers.length;
- while (i-- && !rgba) {
- parser = this.parsers[i];
- result = parser.regex.exec(input);
- if (result) {
- rgba = parser.parse(result);
- }
- }
- }
- }
- this.rgba = rgba || [];
- };
- /**
- * Return the color or gradient stops in the specified format
- *
- * @function Highcharts.Color#get
- *
- * @param {string} [format]
- * Possible values are 'a', 'rgb', 'rgba' (default).
- *
- * @return {Highcharts.ColorType}
- * This color as a string or gradient stops.
- */
- Color.prototype.get = function (format) {
- var input = this.input,
- rgba = this.rgba,
- ret;
- if (typeof this.stops !== 'undefined') {
- ret = merge(input);
- ret.stops = [].concat(ret.stops);
- this.stops.forEach(function (stop, i) {
- ret.stops[i] = [
- ret.stops[i][0],
- stop.get(format)
- ];
- });
- // it's NaN if gradient colors on a column chart
- }
- else if (rgba && isNumber(rgba[0])) {
- if (format === 'rgb' || (!format && rgba[3] === 1)) {
- ret = 'rgb(' + rgba[0] + ',' + rgba[1] + ',' + rgba[2] + ')';
- }
- else if (format === 'a') {
- ret = rgba[3];
- }
- else {
- ret = 'rgba(' + rgba.join(',') + ')';
- }
- }
- else {
- ret = input;
- }
- return ret;
- };
- /**
- * Brighten the color instance.
- *
- * @function Highcharts.Color#brighten
- *
- * @param {number} alpha
- * The alpha value.
- *
- * @return {Highcharts.Color}
- * This color with modifications.
- */
- Color.prototype.brighten = function (alpha) {
- var i,
- rgba = this.rgba;
- if (this.stops) {
- this.stops.forEach(function (stop) {
- stop.brighten(alpha);
- });
- }
- else if (isNumber(alpha) && alpha !== 0) {
- for (i = 0; i < 3; i++) {
- rgba[i] += pInt(alpha * 255);
- if (rgba[i] < 0) {
- rgba[i] = 0;
- }
- if (rgba[i] > 255) {
- rgba[i] = 255;
- }
- }
- }
- return this;
- };
- /**
- * Set the color's opacity to a given alpha value.
- *
- * @function Highcharts.Color#setOpacity
- *
- * @param {number} alpha
- * Opacity between 0 and 1.
- *
- * @return {Highcharts.Color}
- * Color with modifications.
- */
- Color.prototype.setOpacity = function (alpha) {
- this.rgba[3] = alpha;
- return this;
- };
- /**
- * Return an intermediate color between two colors.
- *
- * @function Highcharts.Color#tweenTo
- *
- * @param {Highcharts.Color} to
- * The color object to tween to.
- *
- * @param {number} pos
- * The intermediate position, where 0 is the from color (current
- * color item), and 1 is the `to` color.
- *
- * @return {Highcharts.ColorString}
- * The intermediate color in rgba notation.
- */
- Color.prototype.tweenTo = function (to, pos) {
- // Check for has alpha, because rgba colors perform worse due to lack of
- // support in WebKit.
- var fromRgba = this.rgba,
- toRgba = to.rgba,
- hasAlpha,
- ret;
- // Unsupported color, return to-color (#3920, #7034)
- if (!toRgba.length || !fromRgba || !fromRgba.length) {
- ret = to.input || 'none';
- // Interpolate
- }
- else {
- hasAlpha = (toRgba[3] !== 1 || fromRgba[3] !== 1);
- ret = (hasAlpha ? 'rgba(' : 'rgb(') +
- Math.round(toRgba[0] + (fromRgba[0] - toRgba[0]) * (1 - pos)) +
- ',' +
- Math.round(toRgba[1] + (fromRgba[1] - toRgba[1]) * (1 - pos)) +
- ',' +
- Math.round(toRgba[2] + (fromRgba[2] - toRgba[2]) * (1 - pos)) +
- (hasAlpha ?
- (',' +
- (toRgba[3] + (fromRgba[3] - toRgba[3]) * (1 - pos))) :
- '') +
- ')';
- }
- return ret;
- };
- /* *
- *
- * Static Properties
- *
- * */
- // Collection of named colors. Can be extended from the outside by adding
- // colors to Highcharts.Color.names.
- Color.names = {
- white: '#ffffff',
- black: '#000000'
- };
- return Color;
- }());
- H.Color = Color;
- /**
- * Creates a color instance out of a color string.
- *
- * @function Highcharts.color
- *
- * @param {Highcharts.ColorType} input
- * The input color in either rbga or hex format
- *
- * @return {Highcharts.Color}
- * Color instance
- */
- H.color = Color.parse;
- return H.Color;
- });
- _registerModule(_modules, 'Core/Renderer/SVG/SVGElement.js', [_modules['Core/Color.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Color, H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var deg2rad = H.deg2rad,
- doc = H.doc,
- hasTouch = H.hasTouch,
- isFirefox = H.isFirefox,
- noop = H.noop,
- svg = H.svg,
- SVG_NS = H.SVG_NS,
- win = H.win;
- var animate = U.animate,
- animObject = U.animObject,
- attr = U.attr,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- erase = U.erase,
- extend = U.extend,
- fireEvent = U.fireEvent,
- isArray = U.isArray,
- isFunction = U.isFunction,
- isNumber = U.isNumber,
- isString = U.isString,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- pInt = U.pInt,
- stop = U.stop,
- syncTimeout = U.syncTimeout,
- uniqueKey = U.uniqueKey;
- /**
- * The horizontal alignment of an element.
- *
- * @typedef {"center"|"left"|"right"} Highcharts.AlignValue
- */
- /**
- * Options to align the element relative to the chart or another box.
- *
- * @interface Highcharts.AlignObject
- */ /**
- * Horizontal alignment. Can be one of `left`, `center` and `right`.
- *
- * @name Highcharts.AlignObject#align
- * @type {Highcharts.AlignValue|undefined}
- *
- * @default left
- */ /**
- * Vertical alignment. Can be one of `top`, `middle` and `bottom`.
- *
- * @name Highcharts.AlignObject#verticalAlign
- * @type {Highcharts.VerticalAlignValue|undefined}
- *
- * @default top
- */ /**
- * Horizontal pixel offset from alignment.
- *
- * @name Highcharts.AlignObject#x
- * @type {number|undefined}
- *
- * @default 0
- */ /**
- * Vertical pixel offset from alignment.
- *
- * @name Highcharts.AlignObject#y
- * @type {number|undefined}
- *
- * @default 0
- */ /**
- * Use the `transform` attribute with translateX and translateY custom
- * attributes to align this elements rather than `x` and `y` attributes.
- *
- * @name Highcharts.AlignObject#alignByTranslate
- * @type {boolean|undefined}
- *
- * @default false
- */
- /**
- * Bounding box of an element.
- *
- * @interface Highcharts.BBoxObject
- * @extends Highcharts.PositionObject
- */ /**
- * Height of the bounding box.
- *
- * @name Highcharts.BBoxObject#height
- * @type {number}
- */ /**
- * Width of the bounding box.
- *
- * @name Highcharts.BBoxObject#width
- * @type {number}
- */ /**
- * Horizontal position of the bounding box.
- *
- * @name Highcharts.BBoxObject#x
- * @type {number}
- */ /**
- * Vertical position of the bounding box.
- *
- * @name Highcharts.BBoxObject#y
- * @type {number}
- */
- /**
- * An object of key-value pairs for SVG attributes. Attributes in Highcharts
- * elements for the most parts correspond to SVG, but some are specific to
- * Highcharts, like `zIndex`, `rotation`, `rotationOriginX`,
- * `rotationOriginY`, `translateX`, `translateY`, `scaleX` and `scaleY`. SVG
- * attributes containing a hyphen are _not_ camel-cased, they should be
- * quoted to preserve the hyphen.
- *
- * @example
- * {
- * 'stroke': '#ff0000', // basic
- * 'stroke-width': 2, // hyphenated
- * 'rotation': 45 // custom
- * 'd': ['M', 10, 10, 'L', 30, 30, 'z'] // path definition, note format
- * }
- *
- * @interface Highcharts.SVGAttributes
- */ /**
- * @name Highcharts.SVGAttributes#[key:string]
- * @type {*}
- */ /**
- * @name Highcharts.SVGAttributes#d
- * @type {string|Highcharts.SVGPathArray|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#fill
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#inverted
- * @type {boolean|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#matrix
- * @type {Array<number>|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#rotation
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#rotationOriginX
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#rotationOriginY
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#scaleX
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#scaleY
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#stroke
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#style
- * @type {string|Highcharts.CSSObject|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#translateX
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#translateY
- * @type {number|undefined}
- */ /**
- * @name Highcharts.SVGAttributes#zIndex
- * @type {number|undefined}
- */
- /**
- * An SVG DOM element. The type is a reference to the regular SVGElement in the
- * global scope.
- *
- * @typedef {globals.GlobalSVGElement} Highcharts.SVGDOMElement
- *
- * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGElement
- */
- /**
- * The vertical alignment of an element.
- *
- * @typedef {"bottom"|"middle"|"top"} Highcharts.VerticalAlignValue
- */
- ''; // detach doclets above
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The SVGElement prototype is a JavaScript wrapper for SVG elements used in the
- * rendering layer of Highcharts. Combined with the
- * {@link Highcharts.SVGRenderer}
- * object, these prototypes allow freeform annotation in the charts or even in
- * HTML pages without instanciating a chart. The SVGElement can also wrap HTML
- * labels, when `text` or `label` elements are created with the `useHTML`
- * parameter.
- *
- * The SVGElement instances are created through factory functions on the
- * {@link Highcharts.SVGRenderer}
- * object, like
- * {@link Highcharts.SVGRenderer#rect|rect},
- * {@link Highcharts.SVGRenderer#path|path},
- * {@link Highcharts.SVGRenderer#text|text},
- * {@link Highcharts.SVGRenderer#label|label},
- * {@link Highcharts.SVGRenderer#g|g}
- * and more.
- *
- * @class
- * @name Highcharts.SVGElement
- */
- var SVGElement = /** @class */ (function () {
- function SVGElement() {
- /* *
- *
- * Properties
- *
- * */
- this.element = void 0;
- this.height = void 0;
- this.opacity = 1; // Default base for animation
- this.renderer = void 0;
- this.SVG_NS = SVG_NS;
- // Custom attributes used for symbols, these should be filtered out when
- // setting SVGElement attributes (#9375).
- this.symbolCustomAttribs = [
- 'x',
- 'y',
- 'width',
- 'height',
- 'r',
- 'start',
- 'end',
- 'innerR',
- 'anchorX',
- 'anchorY',
- 'rounded'
- ];
- this.width = void 0;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Get the current value of an attribute or pseudo attribute,
- * used mainly for animation. Called internally from
- * the {@link Highcharts.SVGRenderer#attr} function.
- *
- * @private
- * @function Highcharts.SVGElement#_defaultGetter
- *
- * @param {string} key
- * Property key.
- *
- * @return {number|string}
- * Property value.
- */
- SVGElement.prototype._defaultGetter = function (key) {
- var ret = pick(this[key + 'Value'], // align getter
- this[key],
- this.element ? this.element.getAttribute(key) : null, 0);
- if (/^[\-0-9\.]+$/.test(ret)) { // is numerical
- ret = parseFloat(ret);
- }
- return ret;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#_defaultSetter
- *
- * @param {string} value
- *
- * @param {string} key
- *
- * @param {Highcharts.SVGDOMElement} element
- *
- * @return {void}
- */
- SVGElement.prototype._defaultSetter = function (value, key, element) {
- element.setAttribute(key, value);
- };
- /**
- * Add the element to the DOM. All elements must be added this way.
- *
- * @sample highcharts/members/renderer-g
- * Elements added to a group
- *
- * @function Highcharts.SVGElement#add
- *
- * @param {Highcharts.SVGElement} [parent]
- * The parent item to add it to. If undefined, the element is added
- * to the {@link Highcharts.SVGRenderer.box}.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.add = function (parent) {
- var renderer = this.renderer,
- element = this.element,
- inserted;
- if (parent) {
- this.parentGroup = parent;
- }
- // Mark as inverted
- this.parentInverted = parent && parent.inverted;
- // Build formatted text
- if (typeof this.textStr !== 'undefined' &&
- this.element.nodeName === 'text' // Not for SVGLabel instances
- ) {
- renderer.buildText(this);
- }
- // Mark as added
- this.added = true;
- // If we're adding to renderer root, or other elements in the group
- // have a z index, we need to handle it
- if (!parent || parent.handleZ || this.zIndex) {
- inserted = this.zIndexSetter();
- }
- // If zIndex is not handled, append at the end
- if (!inserted) {
- (parent ?
- parent.element :
- renderer.box).appendChild(element);
- }
- // fire an event for internal hooks
- if (this.onAdd) {
- this.onAdd();
- }
- return this;
- };
- /**
- * Add a class name to an element.
- *
- * @function Highcharts.SVGElement#addClass
- *
- * @param {string} className
- * The new class name to add.
- *
- * @param {boolean} [replace=false]
- * When true, the existing class name(s) will be overwritten with the new
- * one. When false, the new one is added.
- *
- * @return {Highcharts.SVGElement}
- * Return the SVG element for chainability.
- */
- SVGElement.prototype.addClass = function (className, replace) {
- var currentClassName = replace ? '' : (this.attr('class') || '');
- // Trim the string and remove duplicates
- className = (className || '')
- .split(/ /g)
- .reduce(function (newClassName, name) {
- if (currentClassName.indexOf(name) === -1) {
- newClassName.push(name);
- }
- return newClassName;
- }, (currentClassName ?
- [currentClassName] :
- []))
- .join(' ');
- if (className !== currentClassName) {
- this.attr('class', className);
- }
- return this;
- };
- /**
- * This method is executed in the end of `attr()`, after setting all
- * attributes in the hash. In can be used to efficiently consolidate
- * multiple attributes in one SVG property -- e.g., translate, rotate and
- * scale are merged in one "transform" attribute in the SVG node.
- *
- * @private
- * @function Highcharts.SVGElement#afterSetters
- */
- SVGElement.prototype.afterSetters = function () {
- // Update transform. Do this outside the loop to prevent redundant
- // updating for batch setting of attributes.
- if (this.doTransform) {
- this.updateTransform();
- this.doTransform = false;
- }
- };
- /**
- * Align the element relative to the chart or another box.
- *
- * @function Highcharts.SVGElement#align
- *
- * @param {Highcharts.AlignObject} [alignOptions]
- * The alignment options. The function can be called without this
- * parameter in order to re-align an element after the box has been
- * updated.
- *
- * @param {boolean} [alignByTranslate]
- * Align element by translation.
- *
- * @param {string|Highcharts.BBoxObject} [box]
- * The box to align to, needs a width and height. When the box is a
- * string, it refers to an object in the Renderer. For example, when
- * box is `spacingBox`, it refers to `Renderer.spacingBox` which
- * holds `width`, `height`, `x` and `y` properties.
- *
- * @return {Highcharts.SVGElement} Returns the SVGElement for chaining.
- */
- SVGElement.prototype.align = function (alignOptions, alignByTranslate, box) {
- var align,
- vAlign,
- x,
- y,
- attribs = {},
- alignTo,
- renderer = this.renderer,
- alignedObjects = renderer.alignedObjects,
- alignFactor,
- vAlignFactor;
- // First call on instanciate
- if (alignOptions) {
- this.alignOptions = alignOptions;
- this.alignByTranslate = alignByTranslate;
- if (!box || isString(box)) {
- this.alignTo = alignTo = box || 'renderer';
- // prevent duplicates, like legendGroup after resize
- erase(alignedObjects, this);
- alignedObjects.push(this);
- box = void 0; // reassign it below
- }
- // When called on resize, no arguments are supplied
- }
- else {
- alignOptions = this.alignOptions;
- alignByTranslate = this.alignByTranslate;
- alignTo = this.alignTo;
- }
- box = pick(box, renderer[alignTo], renderer);
- // Assign variables
- align = alignOptions.align;
- vAlign = alignOptions.verticalAlign;
- // default: left align
- x = (box.x || 0) + (alignOptions.x || 0);
- // default: top align
- y = (box.y || 0) + (alignOptions.y || 0);
- // Align
- if (align === 'right') {
- alignFactor = 1;
- }
- else if (align === 'center') {
- alignFactor = 2;
- }
- if (alignFactor) {
- x += (box.width - (alignOptions.width || 0)) /
- alignFactor;
- }
- attribs[alignByTranslate ? 'translateX' : 'x'] = Math.round(x);
- // Vertical align
- if (vAlign === 'bottom') {
- vAlignFactor = 1;
- }
- else if (vAlign === 'middle') {
- vAlignFactor = 2;
- }
- if (vAlignFactor) {
- y += (box.height - (alignOptions.height || 0)) /
- vAlignFactor;
- }
- attribs[alignByTranslate ? 'translateY' : 'y'] = Math.round(y);
- // Animate only if already placed
- this[this.placed ? 'animate' : 'attr'](attribs);
- this.placed = true;
- this.alignAttr = attribs;
- return this;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#alignSetter
- * @param {"left"|"center"|"right"} value
- */
- SVGElement.prototype.alignSetter = function (value) {
- var convert = {
- left: 'start',
- center: 'middle',
- right: 'end'
- };
- if (convert[value]) {
- this.alignValue = value;
- this.element.setAttribute('text-anchor', convert[value]);
- }
- };
- /**
- * Animate to given attributes or CSS properties.
- *
- * @sample highcharts/members/element-on/
- * Setting some attributes by animation
- *
- * @function Highcharts.SVGElement#animate
- *
- * @param {Highcharts.SVGAttributes} params
- * SVG attributes or CSS to animate.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [options]
- * Animation options.
- *
- * @param {Function} [complete]
- * Function to perform at the end of animation.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.animate = function (params, options, complete) {
- var _this = this;
- var animOptions = animObject(pick(options,
- this.renderer.globalAnimation,
- true)),
- deferTime = animOptions.defer;
- // When the page is hidden save resources in the background by not
- // running animation at all (#9749).
- if (pick(doc.hidden, doc.msHidden, doc.webkitHidden, false)) {
- animOptions.duration = 0;
- }
- if (animOptions.duration !== 0) {
- // allows using a callback with the global animation without
- // overwriting it
- if (complete) {
- animOptions.complete = complete;
- }
- // If defer option is defined delay the animation #12901
- syncTimeout(function () {
- if (_this.element) {
- animate(_this, params, animOptions);
- }
- }, deferTime);
- }
- else {
- this.attr(params, void 0, complete);
- // Call the end step synchronously
- objectEach(params, function (val, prop) {
- if (animOptions.step) {
- animOptions.step.call(this, val, { prop: prop, pos: 1 });
- }
- }, this);
- }
- return this;
- };
- /**
- * Apply a text outline through a custom CSS property, by copying the text
- * element and apply stroke to the copy. Used internally. Contrast checks at
- * [example](https://jsfiddle.net/highcharts/43soe9m1/2/).
- *
- * @example
- * // Specific color
- * text.css({
- * textOutline: '1px black'
- * });
- * // Automatic contrast
- * text.css({
- * color: '#000000', // black text
- * textOutline: '1px contrast' // => white outline
- * });
- *
- * @private
- * @function Highcharts.SVGElement#applyTextOutline
- *
- * @param {string} textOutline
- * A custom CSS `text-outline` setting, defined by `width color`.
- */
- SVGElement.prototype.applyTextOutline = function (textOutline) {
- var elem = this.element,
- tspans,
- hasContrast = textOutline.indexOf('contrast') !== -1,
- styles = {},
- color,
- strokeWidth,
- firstRealChild;
- // When the text shadow is set to contrast, use dark stroke for light
- // text and vice versa.
- if (hasContrast) {
- styles.textOutline = textOutline = textOutline.replace(/contrast/g, this.renderer.getContrast(elem.style.fill));
- }
- // Extract the stroke width and color
- textOutline = textOutline.split(' ');
- color = textOutline[textOutline.length - 1];
- strokeWidth = textOutline[0];
- if (strokeWidth && strokeWidth !== 'none' && H.svg) {
- this.fakeTS = true; // Fake text shadow
- tspans = [].slice.call(elem.getElementsByTagName('tspan'));
- // In order to get the right y position of the clone,
- // copy over the y setter
- this.ySetter = this.xSetter;
- // Since the stroke is applied on center of the actual outline, we
- // need to double it to get the correct stroke-width outside the
- // glyphs.
- strokeWidth = strokeWidth.replace(/(^[\d\.]+)(.*?)$/g, function (match, digit, unit) {
- return (2 * digit) + unit;
- });
- // Remove shadows from previous runs.
- this.removeTextOutline(tspans);
- // Check if the element contains RTL characters.
- // Comparing against Hebrew and Arabic characters,
- // excluding Arabic digits. Source:
- // https://www.unicode.org/Public/UNIDATA/extracted/DerivedBidiClass.txt
- var isRTL_1 = elem.textContent ?
- /^[\u0591-\u065F\u066A-\u07FF\uFB1D-\uFDFD\uFE70-\uFEFC]/
- .test(elem.textContent) : false;
- // For each of the tspans, create a stroked copy behind it.
- firstRealChild = elem.firstChild;
- tspans.forEach(function (tspan, y) {
- var clone;
- // Let the first line start at the correct X position
- if (y === 0) {
- tspan.setAttribute('x', elem.getAttribute('x'));
- y = elem.getAttribute('y');
- tspan.setAttribute('y', y || 0);
- if (y === null) {
- elem.setAttribute('y', 0);
- }
- }
- // Create the clone and apply outline properties.
- // For RTL elements apply outline properties for orginal element
- // to prevent outline from overlapping the text.
- // For RTL in Firefox keep the orginal order (#10162).
- clone = tspan.cloneNode(true);
- attr((isRTL_1 && !isFirefox) ? tspan : clone, {
- 'class': 'highcharts-text-outline',
- fill: color,
- stroke: color,
- 'stroke-width': strokeWidth,
- 'stroke-linejoin': 'round'
- });
- elem.insertBefore(clone, firstRealChild);
- });
- // Create a whitespace between tspan and clone,
- // to fix the display of Arabic characters in Firefox.
- if (isRTL_1 && isFirefox && tspans[0]) {
- var whitespace = tspans[0].cloneNode(true);
- whitespace.textContent = ' ';
- elem.insertBefore(whitespace, firstRealChild);
- }
- }
- };
- /**
- * @function Highcharts.SVGElement#attr
- * @param {string} key
- * @return {number|string}
- */ /**
- * Apply native and custom attributes to the SVG elements.
- *
- * In order to set the rotation center for rotation, set x and y to 0 and
- * use `translateX` and `translateY` attributes to position the element
- * instead.
- *
- * Attributes frequently used in Highcharts are `fill`, `stroke`,
- * `stroke-width`.
- *
- * @sample highcharts/members/renderer-rect/
- * Setting some attributes
- *
- * @example
- * // Set multiple attributes
- * element.attr({
- * stroke: 'red',
- * fill: 'blue',
- * x: 10,
- * y: 10
- * });
- *
- * // Set a single attribute
- * element.attr('stroke', 'red');
- *
- * // Get an attribute
- * element.attr('stroke'); // => 'red'
- *
- * @function Highcharts.SVGElement#attr
- *
- * @param {string|Highcharts.SVGAttributes} [hash]
- * The native and custom SVG attributes.
- *
- * @param {number|string|Highcharts.SVGPathArray} [val]
- * If the type of the first argument is `string`, the second can be a
- * value, which will serve as a single attribute setter. If the first
- * argument is a string and the second is undefined, the function
- * serves as a getter and the current value of the property is
- * returned.
- *
- * @param {Function} [complete]
- * A callback function to execute after setting the attributes. This
- * makes the function compliant and interchangeable with the
- * {@link SVGElement#animate} function.
- *
- * @param {boolean} [continueAnimation=true]
- * Used internally when `.attr` is called as part of an animation
- * step. Otherwise, calling `.attr` for an attribute will stop
- * animation for that attribute.
- *
- * @return {Highcharts.SVGElement}
- * If used as a setter, it returns the current
- * {@link Highcharts.SVGElement} so the calls can be chained. If
- * used as a getter, the current value of the attribute is returned.
- */
- SVGElement.prototype.attr = function (hash, val, complete, continueAnimation) {
- var key,
- element = this.element,
- hasSetSymbolSize,
- ret = this,
- skipAttr,
- setter,
- symbolCustomAttribs = this.symbolCustomAttribs;
- // single key-value pair
- if (typeof hash === 'string' && typeof val !== 'undefined') {
- key = hash;
- hash = {};
- hash[key] = val;
- }
- // used as a getter: first argument is a string, second is undefined
- if (typeof hash === 'string') {
- ret = (this[hash + 'Getter'] ||
- this._defaultGetter).call(this, hash, element);
- // setter
- }
- else {
- objectEach(hash, function eachAttribute(val, key) {
- skipAttr = false;
- // Unless .attr is from the animator update, stop current
- // running animation of this property
- if (!continueAnimation) {
- stop(this, key);
- }
- // Special handling of symbol attributes
- if (this.symbolName &&
- symbolCustomAttribs.indexOf(key) !== -1) {
- if (!hasSetSymbolSize) {
- this.symbolAttr(hash);
- hasSetSymbolSize = true;
- }
- skipAttr = true;
- }
- if (this.rotation && (key === 'x' || key === 'y')) {
- this.doTransform = true;
- }
- if (!skipAttr) {
- setter = (this[key + 'Setter'] ||
- this._defaultSetter);
- setter.call(this, val, key, element);
- // Let the shadow follow the main element
- if (!this.styledMode &&
- this.shadows &&
- /^(width|height|visibility|x|y|d|transform|cx|cy|r)$/.test(key)) {
- this.updateShadows(key, val, setter);
- }
- }
- }, this);
- this.afterSetters();
- }
- // In accordance with animate, run a complete callback
- if (complete) {
- complete.call(this);
- }
- return ret;
- };
- /**
- * Apply a clipping rectangle to this element.
- *
- * @function Highcharts.SVGElement#clip
- *
- * @param {Highcharts.ClipRectElement} [clipRect]
- * The clipping rectangle. If skipped, the current clip is removed.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVG element to allow chaining.
- */
- SVGElement.prototype.clip = function (clipRect) {
- return this.attr('clip-path', clipRect ?
- 'url(' + this.renderer.url + '#' + clipRect.id + ')' :
- 'none');
- };
- /**
- * Calculate the coordinates needed for drawing a rectangle crisply and
- * return the calculated attributes.
- *
- * @function Highcharts.SVGElement#crisp
- *
- * @param {Highcharts.RectangleObject} rect
- * Rectangle to crisp.
- *
- * @param {number} [strokeWidth]
- * The stroke width to consider when computing crisp positioning. It can
- * also be set directly on the rect parameter.
- *
- * @return {Highcharts.RectangleObject}
- * The modified rectangle arguments.
- */
- SVGElement.prototype.crisp = function (rect, strokeWidth) {
- var wrapper = this,
- normalizer;
- strokeWidth = strokeWidth || rect.strokeWidth || 0;
- // Math.round because strokeWidth can sometimes have roundoff errors
- normalizer = Math.round(strokeWidth) % 2 / 2;
- // normalize for crisp edges
- rect.x = Math.floor(rect.x || wrapper.x || 0) + normalizer;
- rect.y = Math.floor(rect.y || wrapper.y || 0) + normalizer;
- rect.width = Math.floor((rect.width || wrapper.width || 0) - 2 * normalizer);
- rect.height = Math.floor((rect.height || wrapper.height || 0) - 2 * normalizer);
- if (defined(rect.strokeWidth)) {
- rect.strokeWidth = strokeWidth;
- }
- return rect;
- };
- /**
- * Build and apply an SVG gradient out of a common JavaScript configuration
- * object. This function is called from the attribute setters. An event
- * hook is added for supporting other complex color types.
- *
- * @private
- * @function Highcharts.SVGElement#complexColor
- *
- * @param {Highcharts.GradientColorObject|Highcharts.PatternObject} colorOptions
- * The gradient or pattern options structure.
- *
- * @param {string} prop
- * The property to apply, can either be `fill` or `stroke`.
- *
- * @param {Highcharts.SVGDOMElement} elem
- * SVG element to apply the gradient on.
- */
- SVGElement.prototype.complexColor = function (colorOptions, prop, elem) {
- var renderer = this.renderer,
- colorObject,
- gradName,
- gradAttr,
- radAttr,
- gradients,
- stops,
- stopColor,
- stopOpacity,
- radialReference,
- id,
- key = [],
- value;
- fireEvent(this.renderer, 'complexColor', {
- args: arguments
- }, function () {
- // Apply linear or radial gradients
- if (colorOptions.radialGradient) {
- gradName = 'radialGradient';
- }
- else if (colorOptions.linearGradient) {
- gradName = 'linearGradient';
- }
- if (gradName) {
- gradAttr = colorOptions[gradName];
- gradients = renderer.gradients;
- stops = colorOptions.stops;
- radialReference = elem.radialReference;
- // Keep < 2.2 kompatibility
- if (isArray(gradAttr)) {
- colorOptions[gradName] = gradAttr = {
- x1: gradAttr[0],
- y1: gradAttr[1],
- x2: gradAttr[2],
- y2: gradAttr[3],
- gradientUnits: 'userSpaceOnUse'
- };
- }
- // Correct the radial gradient for the radial reference system
- if (gradName === 'radialGradient' &&
- radialReference &&
- !defined(gradAttr.gradientUnits)) {
- // Save the radial attributes for updating
- radAttr = gradAttr;
- gradAttr = merge(gradAttr, renderer.getRadialAttr(radialReference, radAttr), { gradientUnits: 'userSpaceOnUse' });
- }
- // Build the unique key to detect whether we need to create a
- // new element (#1282)
- objectEach(gradAttr, function (val, n) {
- if (n !== 'id') {
- key.push(n, val);
- }
- });
- objectEach(stops, function (val) {
- key.push(val);
- });
- key = key.join(',');
- // Check if a gradient object with the same config object is
- // created within this renderer
- if (gradients[key]) {
- id = gradients[key].attr('id');
- }
- else {
- // Set the id and create the element
- gradAttr.id = id = uniqueKey();
- var gradientObject_1 = gradients[key] =
- renderer.createElement(gradName)
- .attr(gradAttr)
- .add(renderer.defs);
- gradientObject_1.radAttr = radAttr;
- // The gradient needs to keep a list of stops to be able to
- // destroy them
- gradientObject_1.stops = [];
- stops.forEach(function (stop) {
- var stopObject;
- if (stop[1].indexOf('rgba') === 0) {
- colorObject = Color.parse(stop[1]);
- stopColor = colorObject.get('rgb');
- stopOpacity = colorObject.get('a');
- }
- else {
- stopColor = stop[1];
- stopOpacity = 1;
- }
- stopObject = renderer.createElement('stop').attr({
- offset: stop[0],
- 'stop-color': stopColor,
- 'stop-opacity': stopOpacity
- }).add(gradientObject_1);
- // Add the stop element to the gradient
- gradientObject_1.stops.push(stopObject);
- });
- }
- // Set the reference to the gradient object
- value = 'url(' + renderer.url + '#' + id + ')';
- elem.setAttribute(prop, value);
- elem.gradient = key;
- // Allow the color to be concatenated into tooltips formatters
- // etc. (#2995)
- colorOptions.toString = function () {
- return value;
- };
- }
- });
- };
- /**
- * Set styles for the element. In addition to CSS styles supported by
- * native SVG and HTML elements, there are also some custom made for
- * Highcharts, like `width`, `ellipsis` and `textOverflow` for SVG text
- * elements.
- *
- * @sample highcharts/members/renderer-text-on-chart/
- * Styled text
- *
- * @function Highcharts.SVGElement#css
- *
- * @param {Highcharts.CSSObject} styles
- * The new CSS styles.
- *
- * @return {Highcharts.SVGElement}
- * Return the SVG element for chaining.
- */
- SVGElement.prototype.css = function (styles) {
- var oldStyles = this.styles, newStyles = {}, elem = this.element, textWidth, serializedCss = '', hyphenate, hasNew = !oldStyles,
- // These CSS properties are interpreted internally by the SVG
- // renderer, but are not supported by SVG and should not be added to
- // the DOM. In styled mode, no CSS should find its way to the DOM
- // whatsoever (#6173, #6474).
- svgPseudoProps = ['textOutline', 'textOverflow', 'width'];
- // convert legacy
- if (styles && styles.color) {
- styles.fill = styles.color;
- }
- // Filter out existing styles to increase performance (#2640)
- if (oldStyles) {
- objectEach(styles, function (style, n) {
- if (oldStyles && oldStyles[n] !== style) {
- newStyles[n] = style;
- hasNew = true;
- }
- });
- }
- if (hasNew) {
- // Merge the new styles with the old ones
- if (oldStyles) {
- styles = extend(oldStyles, newStyles);
- }
- // Get the text width from style
- if (styles) {
- // Previously set, unset it (#8234)
- if (styles.width === null || styles.width === 'auto') {
- delete this.textWidth;
- // Apply new
- }
- else if (elem.nodeName.toLowerCase() === 'text' &&
- styles.width) {
- textWidth = this.textWidth = pInt(styles.width);
- }
- }
- // store object
- this.styles = styles;
- if (textWidth && (!svg && this.renderer.forExport)) {
- delete styles.width;
- }
- // Serialize and set style attribute
- if (elem.namespaceURI === this.SVG_NS) { // #7633
- hyphenate = function (a, b) {
- return '-' + b.toLowerCase();
- };
- objectEach(styles, function (style, n) {
- if (svgPseudoProps.indexOf(n) === -1) {
- serializedCss +=
- n.replace(/([A-Z])/g, hyphenate) + ':' +
- style + ';';
- }
- });
- if (serializedCss) {
- attr(elem, 'style', serializedCss); // #1881
- }
- }
- else {
- css(elem, styles);
- }
- if (this.added) {
- // Rebuild text after added. Cache mechanisms in the buildText
- // will prevent building if there are no significant changes.
- if (this.element.nodeName === 'text') {
- this.renderer.buildText(this);
- }
- // Apply text outline after added
- if (styles && styles.textOutline) {
- this.applyTextOutline(styles.textOutline);
- }
- }
- }
- return this;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#dashstyleSetter
- * @param {string} value
- */
- SVGElement.prototype.dashstyleSetter = function (value) {
- var i,
- strokeWidth = this['stroke-width'];
- // If "inherit", like maps in IE, assume 1 (#4981). With HC5 and the new
- // strokeWidth function, we should be able to use that instead.
- if (strokeWidth === 'inherit') {
- strokeWidth = 1;
- }
- value = value && value.toLowerCase();
- if (value) {
- var v = value
- .replace('shortdashdotdot', '3,1,1,1,1,1,')
- .replace('shortdashdot', '3,1,1,1')
- .replace('shortdot', '1,1,')
- .replace('shortdash', '3,1,')
- .replace('longdash', '8,3,')
- .replace(/dot/g, '1,3,')
- .replace('dash', '4,3,')
- .replace(/,$/, '')
- .split(','); // ending comma
- i = v.length;
- while (i--) {
- v[i] = '' + (pInt(v[i]) * pick(strokeWidth, NaN));
- }
- value = v.join(',').replace(/NaN/g, 'none'); // #3226
- this.element.setAttribute('stroke-dasharray', value);
- }
- };
- /**
- * Destroy the element and element wrapper and clear up the DOM and event
- * hooks.
- *
- * @function Highcharts.SVGElement#destroy
- */
- SVGElement.prototype.destroy = function () {
- var wrapper = this,
- element = wrapper.element || {},
- renderer = wrapper.renderer,
- parentToClean = (renderer.isSVG &&
- element.nodeName === 'SPAN' &&
- wrapper.parentGroup ||
- void 0),
- grandParent,
- ownerSVGElement = element.ownerSVGElement,
- i;
- // remove events
- element.onclick = element.onmouseout = element.onmouseover =
- element.onmousemove = element.point = null;
- stop(wrapper); // stop running animations
- if (wrapper.clipPath && ownerSVGElement) {
- var clipPath_1 = wrapper.clipPath;
- // Look for existing references to this clipPath and remove them
- // before destroying the element (#6196).
- // The upper case version is for Edge
- [].forEach.call(ownerSVGElement.querySelectorAll('[clip-path],[CLIP-PATH]'), function (el) {
- var clipPathAttr = el.getAttribute('clip-path');
- if (clipPathAttr.indexOf(clipPath_1.element.id) > -1) {
- el.removeAttribute('clip-path');
- }
- });
- wrapper.clipPath = clipPath_1.destroy();
- }
- // Destroy stops in case this is a gradient object @todo old code?
- if (wrapper.stops) {
- for (i = 0; i < wrapper.stops.length; i++) {
- wrapper.stops[i].destroy();
- }
- wrapper.stops.length = 0;
- wrapper.stops = void 0;
- }
- // remove element
- wrapper.safeRemoveChild(element);
- if (!renderer.styledMode) {
- wrapper.destroyShadows();
- }
- // In case of useHTML, clean up empty containers emulating SVG groups
- // (#1960, #2393, #2697).
- while (parentToClean &&
- parentToClean.div &&
- parentToClean.div.childNodes.length === 0) {
- grandParent = parentToClean.parentGroup;
- wrapper.safeRemoveChild(parentToClean.div);
- delete parentToClean.div;
- parentToClean = grandParent;
- }
- // remove from alignObjects
- if (wrapper.alignTo) {
- erase(renderer.alignedObjects, wrapper);
- }
- objectEach(wrapper, function (val, key) {
- // Destroy child elements of a group
- if (wrapper[key] &&
- wrapper[key].parentGroup === wrapper &&
- wrapper[key].destroy) {
- wrapper[key].destroy();
- }
- // Delete all properties
- delete wrapper[key];
- });
- return;
- };
- /**
- * Destroy shadows on the element.
- *
- * @private
- * @function Highcharts.SVGElement#destroyShadows
- *
- * @return {void}
- */
- SVGElement.prototype.destroyShadows = function () {
- (this.shadows || []).forEach(function (shadow) {
- this.safeRemoveChild(shadow);
- }, this);
- this.shadows = void 0;
- };
- /**
- * @private
- */
- SVGElement.prototype.destroyTextPath = function (elem, path) {
- var textElement = elem.getElementsByTagName('text')[0];
- var tspans;
- if (textElement) {
- // Remove textPath attributes
- textElement.removeAttribute('dx');
- textElement.removeAttribute('dy');
- // Remove ID's:
- path.element.setAttribute('id', '');
- // Check if textElement includes textPath,
- if (this.textPathWrapper &&
- textElement.getElementsByTagName('textPath').length) {
- // Move nodes to <text>
- tspans = this.textPathWrapper.element.childNodes;
- // Now move all <tspan>'s to the <textPath> node
- while (tspans.length) {
- textElement.appendChild(tspans[0]);
- }
- // Remove <textPath> from the DOM
- textElement.removeChild(this.textPathWrapper.element);
- }
- }
- else if (elem.getAttribute('dx') || elem.getAttribute('dy')) {
- // Remove textPath attributes from elem
- // to get correct text-outline position
- elem.removeAttribute('dx');
- elem.removeAttribute('dy');
- }
- if (this.textPathWrapper) {
- // Set textPathWrapper to undefined and destroy it
- this.textPathWrapper = this.textPathWrapper.destroy();
- }
- };
- /**
- * @private
- * @function Highcharts.SVGElement#dSettter
- * @param {number|string|Highcharts.SVGPathArray} value
- * @param {string} key
- * @param {Highcharts.SVGDOMElement} element
- */
- SVGElement.prototype.dSetter = function (value, key, element) {
- if (isArray(value)) {
- // Backwards compatibility, convert one-dimensional array into an
- // array of segments
- if (typeof value[0] === 'string') {
- value = this.renderer.pathToSegments(value);
- }
- this.pathArray = value;
- value = value.reduce(function (acc, seg, i) {
- if (!seg || !seg.join) {
- return (seg || '').toString();
- }
- return (i ? acc + ' ' : '') + seg.join(' ');
- }, '');
- }
- if (/(NaN| {2}|^$)/.test(value)) {
- value = 'M 0 0';
- }
- // Check for cache before resetting. Resetting causes disturbance in the
- // DOM, causing flickering in some cases in Edge/IE (#6747). Also
- // possible performance gain.
- if (this[key] !== value) {
- element.setAttribute(key, value);
- this[key] = value;
- }
- };
- /**
- * Fade out an element by animating its opacity down to 0, and hide it on
- * complete. Used internally for the tooltip.
- *
- * @function Highcharts.SVGElement#fadeOut
- *
- * @param {number} [duration=150]
- * The fade duration in milliseconds.
- */
- SVGElement.prototype.fadeOut = function (duration) {
- var elemWrapper = this;
- elemWrapper.animate({
- opacity: 0
- }, {
- duration: pick(duration, 150),
- complete: function () {
- // #3088, assuming we're only using this for tooltips
- elemWrapper.attr({ y: -9999 }).hide();
- }
- });
- };
- /**
- * @private
- * @function Highcharts.SVGElement#fillSetter
- * @param {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject} value
- * @param {string} key
- * @param {Highcharts.SVGDOMElement} element
- */
- SVGElement.prototype.fillSetter = function (value, key, element) {
- if (typeof value === 'string') {
- element.setAttribute(key, value);
- }
- else if (value) {
- this.complexColor(value, key, element);
- }
- };
- /**
- * Get the bounding box (width, height, x and y) for the element. Generally
- * used to get rendered text size. Since this is called a lot in charts,
- * the results are cached based on text properties, in order to save DOM
- * traffic. The returned bounding box includes the rotation, so for example
- * a single text line of rotation 90 will report a greater height, and a
- * width corresponding to the line-height.
- *
- * @sample highcharts/members/renderer-on-chart/
- * Draw a rectangle based on a text's bounding box
- *
- * @function Highcharts.SVGElement#getBBox
- *
- * @param {boolean} [reload]
- * Skip the cache and get the updated DOM bouding box.
- *
- * @param {number} [rot]
- * Override the element's rotation. This is internally used on axis
- * labels with a value of 0 to find out what the bounding box would
- * be have been if it were not rotated.
- *
- * @return {Highcharts.BBoxObject}
- * The bounding box with `x`, `y`, `width` and `height` properties.
- */
- SVGElement.prototype.getBBox = function (reload, rot) {
- var wrapper = this,
- bBox, // = wrapper.bBox,
- renderer = wrapper.renderer,
- width,
- height,
- element = wrapper.element,
- styles = wrapper.styles,
- fontSize,
- textStr = wrapper.textStr,
- toggleTextShadowShim,
- cache = renderer.cache,
- cacheKeys = renderer.cacheKeys,
- isSVG = element.namespaceURI === wrapper.SVG_NS,
- cacheKey;
- var rotation = pick(rot,
- wrapper.rotation, 0);
- fontSize = renderer.styledMode ? (element &&
- SVGElement.prototype.getStyle.call(element, 'font-size')) : (styles && styles.fontSize);
- // Avoid undefined and null (#7316)
- if (defined(textStr)) {
- cacheKey = textStr.toString();
- // Since numbers are monospaced, and numerical labels appear a lot
- // in a chart, we assume that a label of n characters has the same
- // bounding box as others of the same length. Unless there is inner
- // HTML in the label. In that case, leave the numbers as is (#5899).
- if (cacheKey.indexOf('<') === -1) {
- cacheKey = cacheKey.replace(/[0-9]/g, '0');
- }
- // Properties that affect bounding box
- cacheKey += [
- '',
- rotation,
- fontSize,
- wrapper.textWidth,
- styles && styles.textOverflow,
- styles && styles.fontWeight // #12163
- ].join(',');
- }
- if (cacheKey && !reload) {
- bBox = cache[cacheKey];
- }
- // No cache found
- if (!bBox) {
- // SVG elements
- if (isSVG || renderer.forExport) {
- try { // Fails in Firefox if the container has display: none.
- // When the text shadow shim is used, we need to hide the
- // fake shadows to get the correct bounding box (#3872)
- toggleTextShadowShim = this.fakeTS && function (display) {
- [].forEach.call(element.querySelectorAll('.highcharts-text-outline'), function (tspan) {
- tspan.style.display = display;
- });
- };
- // Workaround for #3842, Firefox reporting wrong bounding
- // box for shadows
- if (isFunction(toggleTextShadowShim)) {
- toggleTextShadowShim('none');
- }
- bBox = element.getBBox ?
- // SVG: use extend because IE9 is not allowed to change
- // width and height in case of rotation (below)
- extend({}, element.getBBox()) : {
- // Legacy IE in export mode
- width: element.offsetWidth,
- height: element.offsetHeight
- };
- // #3842
- if (isFunction(toggleTextShadowShim)) {
- toggleTextShadowShim('');
- }
- }
- catch (e) {
- '';
- }
- // If the bBox is not set, the try-catch block above failed. The
- // other condition is for Opera that returns a width of
- // -Infinity on hidden elements.
- if (!bBox || bBox.width < 0) {
- bBox = { width: 0, height: 0 };
- }
- // VML Renderer or useHTML within SVG
- }
- else {
- bBox = wrapper.htmlGetBBox();
- }
- // True SVG elements as well as HTML elements in modern browsers
- // using the .useHTML option need to compensated for rotation
- if (renderer.isSVG) {
- width = bBox.width;
- height = bBox.height;
- // Workaround for wrong bounding box in IE, Edge and Chrome on
- // Windows. With Highcharts' default font, IE and Edge report
- // a box height of 16.899 and Chrome rounds it to 17. If this
- // stands uncorrected, it results in more padding added below
- // the text than above when adding a label border or background.
- // Also vertical positioning is affected.
- // https://jsfiddle.net/highcharts/em37nvuj/
- // (#1101, #1505, #1669, #2568, #6213).
- if (isSVG) {
- bBox.height = height = ({
- '11px,17': 14,
- '13px,20': 16
- }[styles &&
- styles.fontSize + ',' + Math.round(height)] ||
- height);
- }
- // Adjust for rotated text
- if (rotation) {
- var rad = rotation * deg2rad;
- bBox.width = Math.abs(height * Math.sin(rad)) +
- Math.abs(width * Math.cos(rad));
- bBox.height = Math.abs(height * Math.cos(rad)) +
- Math.abs(width * Math.sin(rad));
- }
- }
- // Cache it. When loading a chart in a hidden iframe in Firefox and
- // IE/Edge, the bounding box height is 0, so don't cache it (#5620).
- if (cacheKey && bBox.height > 0) {
- // Rotate (#4681)
- while (cacheKeys.length > 250) {
- delete cache[cacheKeys.shift()];
- }
- if (!cache[cacheKey]) {
- cacheKeys.push(cacheKey);
- }
- cache[cacheKey] = bBox;
- }
- }
- return bBox;
- };
- /**
- * Get the computed style. Only in styled mode.
- *
- * @example
- * chart.series[0].points[0].graphic.getStyle('stroke-width'); // => '1px'
- *
- * @function Highcharts.SVGElement#getStyle
- *
- * @param {string} prop
- * The property name to check for.
- *
- * @return {string}
- * The current computed value.
- */
- SVGElement.prototype.getStyle = function (prop) {
- return win
- .getComputedStyle(this.element || this, '')
- .getPropertyValue(prop);
- };
- /**
- * Check if an element has the given class name.
- *
- * @function Highcharts.SVGElement#hasClass
- *
- * @param {string} className
- * The class name to check for.
- *
- * @return {boolean}
- * Whether the class name is found.
- */
- SVGElement.prototype.hasClass = function (className) {
- return ('' + this.attr('class'))
- .split(' ')
- .indexOf(className) !== -1;
- };
- /**
- * Hide the element, similar to setting the `visibility` attribute to
- * `hidden`.
- *
- * @function Highcharts.SVGElement#hide
- *
- * @param {boolean} [hideByTranslation=false]
- * The flag to determine if element should be hidden by moving out
- * of the viewport. Used for example for dataLabels.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.hide = function (hideByTranslation) {
- if (hideByTranslation) {
- this.attr({ y: -9999 });
- }
- else {
- this.attr({ visibility: 'hidden' });
- }
- return this;
- };
- /**
- * @private
- */
- SVGElement.prototype.htmlGetBBox = function () {
- return { height: 0, width: 0, x: 0, y: 0 };
- };
- /**
- * Initialize the SVG element. This function only exists to make the
- * initialization process overridable. It should not be called directly.
- *
- * @function Highcharts.SVGElement#init
- *
- * @param {Highcharts.SVGRenderer} renderer
- * The SVGRenderer instance to initialize to.
- *
- * @param {string} nodeName
- * The SVG node name.
- */
- SVGElement.prototype.init = function (renderer, nodeName) {
- /**
- * The primary DOM node. Each `SVGElement` instance wraps a main DOM
- * node, but may also represent more nodes.
- *
- * @name Highcharts.SVGElement#element
- * @type {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement}
- */
- this.element = nodeName === 'span' ?
- createElement(nodeName) :
- doc.createElementNS(this.SVG_NS, nodeName);
- /**
- * The renderer that the SVGElement belongs to.
- *
- * @name Highcharts.SVGElement#renderer
- * @type {Highcharts.SVGRenderer}
- */
- this.renderer = renderer;
- fireEvent(this, 'afterInit');
- };
- /**
- * Invert a group, rotate and flip. This is used internally on inverted
- * charts, where the points and graphs are drawn as if not inverted, then
- * the series group elements are inverted.
- *
- * @function Highcharts.SVGElement#invert
- *
- * @param {boolean} inverted
- * Whether to invert or not. An inverted shape can be un-inverted by
- * setting it to false.
- *
- * @return {Highcharts.SVGElement}
- * Return the SVGElement for chaining.
- */
- SVGElement.prototype.invert = function (inverted) {
- var wrapper = this;
- wrapper.inverted = inverted;
- wrapper.updateTransform();
- return wrapper;
- };
- /**
- * Add an event listener. This is a simple setter that replaces all other
- * events of the same type, opposed to the {@link Highcharts#addEvent}
- * function.
- *
- * @sample highcharts/members/element-on/
- * A clickable rectangle
- *
- * @function Highcharts.SVGElement#on
- *
- * @param {string} eventType
- * The event type. If the type is `click`, Highcharts will internally
- * translate it to a `touchstart` event on touch devices, to prevent the
- * browser from waiting for a click event from firing.
- *
- * @param {Function} handler
- * The handler callback.
- *
- * @return {Highcharts.SVGElement}
- * The SVGElement for chaining.
- */
- SVGElement.prototype.on = function (eventType, handler) {
- var svgElement = this,
- element = svgElement.element,
- touchStartPos,
- touchEventFired;
- // touch
- if (hasTouch && eventType === 'click') {
- element.ontouchstart = function (e) {
- // save touch position for later calculation
- touchStartPos = {
- clientX: e.touches[0].clientX,
- clientY: e.touches[0].clientY
- };
- };
- // Instead of ontouchstart, event handlers should be called
- // on touchend - similar to how current mouseup events are called
- element.ontouchend = function (e) {
- // hasMoved is a boolean variable containing logic if page
- // was scrolled, so if touch position changed more than
- // ~4px (value borrowed from general touch handler)
- var hasMoved = touchStartPos.clientX ? Math.sqrt(Math.pow(touchStartPos.clientX - e.changedTouches[0].clientX, 2) +
- Math.pow(touchStartPos.clientY - e.changedTouches[0].clientY, 2)) >= 4 : false;
- if (!hasMoved) { // only call handlers if page was not scrolled
- handler.call(element, e);
- }
- touchEventFired = true;
- // prevent other events from being fired. #9682
- e.preventDefault();
- };
- element.onclick = function (e) {
- // Do not call onclick handler if touch event was fired already.
- if (!touchEventFired) {
- handler.call(element, e);
- }
- };
- }
- else {
- // simplest possible event model for internal use
- element['on' + eventType] = handler;
- }
- return this;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#opacitySetter
- * @param {string} value
- * @param {string} key
- * @param {Highcharts.SVGDOMElement} element
- */
- SVGElement.prototype.opacitySetter = function (value, key, element) {
- this[key] = value;
- element.setAttribute(key, value);
- };
- /**
- * Remove a class name from the element.
- *
- * @function Highcharts.SVGElement#removeClass
- *
- * @param {string|RegExp} className
- * The class name to remove.
- *
- * @return {Highcharts.SVGElement} Returns the SVG element for chainability.
- */
- SVGElement.prototype.removeClass = function (className) {
- return this.attr('class', ('' + this.attr('class'))
- .replace(isString(className) ?
- new RegExp("(^| )" + className + "( |$)") : // #12064, #13590
- className, ' ')
- .replace(/ +/g, ' ')
- .trim());
- };
- /**
- * @private
- * @param {Array<Highcharts.SVGDOMElement>} tspans
- * Text spans.
- */
- SVGElement.prototype.removeTextOutline = function (tspans) {
- // Iterate from the end to
- // support removing items inside the cycle (#6472).
- var i = tspans.length,
- tspan;
- while (i--) {
- tspan = tspans[i];
- if (tspan.getAttribute('class') === 'highcharts-text-outline') {
- // Remove then erase
- erase(tspans, this.element.removeChild(tspan));
- }
- }
- };
- /**
- * Removes an element from the DOM.
- *
- * @private
- * @function Highcharts.SVGElement#safeRemoveChild
- *
- * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element
- * The DOM node to remove.
- */
- SVGElement.prototype.safeRemoveChild = function (element) {
- var parentNode = element.parentNode;
- if (parentNode) {
- parentNode.removeChild(element);
- }
- };
- /**
- * Set the coordinates needed to draw a consistent radial gradient across
- * a shape regardless of positioning inside the chart. Used on pie slices
- * to make all the slices have the same radial reference point.
- *
- * @function Highcharts.SVGElement#setRadialReference
- *
- * @param {Array<number>} coordinates
- * The center reference. The format is `[centerX, centerY, diameter]` in
- * pixels.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.setRadialReference = function (coordinates) {
- var existingGradient = (this.element.gradient &&
- this.renderer.gradients[this.element.gradient]);
- this.element.radialReference = coordinates;
- // On redrawing objects with an existing gradient, the gradient needs
- // to be repositioned (#3801)
- if (existingGradient && existingGradient.radAttr) {
- existingGradient.animate(this.renderer.getRadialAttr(coordinates, existingGradient.radAttr));
- }
- return this;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#setTextPath
- * @param {Highcharts.SVGElement} path
- * Path to follow.
- * @param {Highcharts.DataLabelsTextPathOptionsObject} textPathOptions
- * Options.
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.setTextPath = function (path, textPathOptions) {
- var elem = this.element,
- attribsMap = {
- textAnchor: 'text-anchor'
- },
- attrs,
- adder = false,
- textPathElement,
- textPathId,
- textPathWrapper = this.textPathWrapper,
- tspans,
- firstTime = !textPathWrapper;
- // Defaults
- textPathOptions = merge(true, {
- enabled: true,
- attributes: {
- dy: -5,
- startOffset: '50%',
- textAnchor: 'middle'
- }
- }, textPathOptions);
- attrs = textPathOptions.attributes;
- if (path && textPathOptions && textPathOptions.enabled) {
- // In case of fixed width for a text, string is rebuilt
- // (e.g. ellipsis is applied), so we need to rebuild textPath too
- if (textPathWrapper &&
- textPathWrapper.element.parentNode === null) {
- // When buildText functionality was triggered again
- // and deletes textPathWrapper parentNode
- firstTime = true;
- textPathWrapper = textPathWrapper.destroy();
- }
- else if (textPathWrapper) {
- // Case after drillup when spans were added into
- // the DOM outside the textPathWrapper parentGroup
- this.removeTextOutline.call(textPathWrapper.parentGroup, [].slice.call(elem.getElementsByTagName('tspan')));
- }
- // label() has padding, text() doesn't
- if (this.options && this.options.padding) {
- attrs.dx = -this.options.padding;
- }
- if (!textPathWrapper) {
- // Create <textPath>, defer the DOM adder
- this.textPathWrapper = textPathWrapper =
- this.renderer.createElement('textPath');
- adder = true;
- }
- textPathElement = textPathWrapper.element;
- // Set ID for the path
- textPathId = path.element.getAttribute('id');
- if (!textPathId) {
- path.element.setAttribute('id', textPathId = uniqueKey());
- }
- // Change DOM structure, by placing <textPath> tag in <text>
- if (firstTime) {
- tspans = elem.getElementsByTagName('tspan');
- // Now move all <tspan>'s to the <textPath> node
- while (tspans.length) {
- // Remove "y" from tspans, as Firefox translates them
- tspans[0].setAttribute('y', 0);
- // Remove "x" from tspans
- if (isNumber(attrs.dx)) {
- tspans[0].setAttribute('x', -attrs.dx);
- }
- textPathElement.appendChild(tspans[0]);
- }
- }
- // Add <textPath> to the DOM
- if (adder &&
- textPathWrapper) {
- textPathWrapper.add({
- // label() is placed in a group, text() is standalone
- element: this.text ? this.text.element : elem
- });
- }
- // Set basic options:
- // Use `setAttributeNS` because Safari needs this..
- textPathElement.setAttributeNS('http://www.w3.org/1999/xlink', 'href', this.renderer.url + '#' + textPathId);
- // Presentation attributes:
- // dx/dy options must by set on <text> (parent),
- // the rest should be set on <textPath>
- if (defined(attrs.dy)) {
- textPathElement.parentNode
- .setAttribute('dy', attrs.dy);
- delete attrs.dy;
- }
- if (defined(attrs.dx)) {
- textPathElement.parentNode
- .setAttribute('dx', attrs.dx);
- delete attrs.dx;
- }
- // Additional attributes
- objectEach(attrs, function (val, key) {
- textPathElement.setAttribute(attribsMap[key] || key, val);
- });
- // Remove translation, text that follows path does not need that
- elem.removeAttribute('transform');
- // Remove shadows and text outlines
- this.removeTextOutline.call(textPathWrapper, [].slice.call(elem.getElementsByTagName('tspan')));
- // Remove background and border for label(), see #10545
- // Alternatively, we can disable setting background rects in
- // series.drawDataLabels()
- if (this.text && !this.renderer.styledMode) {
- this.attr({
- fill: 'none',
- 'stroke-width': 0
- });
- }
- // Disable some functions
- this.updateTransform = noop;
- this.applyTextOutline = noop;
- }
- else if (textPathWrapper) {
- // Reset to prototype
- delete this.updateTransform;
- delete this.applyTextOutline;
- // Restore DOM structure:
- this.destroyTextPath(elem, path);
- // Bring attributes back
- this.updateTransform();
- // Set textOutline back for text()
- if (this.options && this.options.rotation) {
- this.applyTextOutline(this.options.style.textOutline);
- }
- }
- return this;
- };
- /**
- * Add a shadow to the element. Must be called after the element is added to
- * the DOM. In styled mode, this method is not used, instead use `defs` and
- * filters.
- *
- * @example
- * renderer.rect(10, 100, 100, 100)
- * .attr({ fill: 'red' })
- * .shadow(true);
- *
- * @function Highcharts.SVGElement#shadow
- *
- * @param {boolean|Highcharts.ShadowOptionsObject} [shadowOptions]
- * The shadow options. If `true`, the default options are applied. If
- * `false`, the current shadow will be removed.
- *
- * @param {Highcharts.SVGElement} [group]
- * The SVG group element where the shadows will be applied. The
- * default is to add it to the same parent as the current element.
- * Internally, this is ised for pie slices, where all the shadows are
- * added to an element behind all the slices.
- *
- * @param {boolean} [cutOff]
- * Used internally for column shadows.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.shadow = function (shadowOptions, group, cutOff) {
- var shadows = [],
- i,
- shadow,
- element = this.element,
- strokeWidth,
- shadowElementOpacity,
- update = false,
- oldShadowOptions = this.oldShadowOptions,
- // compensate for inverted plot area
- transform;
- var defaultShadowOptions = {
- color: '#000000',
- offsetX: 1,
- offsetY: 1,
- opacity: 0.15,
- width: 3
- };
- var options;
- if (shadowOptions === true) {
- options = defaultShadowOptions;
- }
- else if (typeof shadowOptions === 'object') {
- options = extend(defaultShadowOptions, shadowOptions);
- }
- // Update shadow when options change (#12091).
- if (options) {
- // Go over each key to look for change
- if (options && oldShadowOptions) {
- objectEach(options, function (value, key) {
- if (value !== oldShadowOptions[key]) {
- update = true;
- }
- });
- }
- if (update) {
- this.destroyShadows();
- }
- this.oldShadowOptions = options;
- }
- if (!options) {
- this.destroyShadows();
- }
- else if (!this.shadows) {
- shadowElementOpacity = options.opacity / options.width;
- transform = this.parentInverted ?
- 'translate(-1,-1)' :
- "translate(" + options.offsetX + ", " + options.offsetY + ")";
- for (i = 1; i <= options.width; i++) {
- shadow = element.cloneNode(false);
- strokeWidth = (options.width * 2) + 1 - (2 * i);
- attr(shadow, {
- stroke: (shadowOptions.color ||
- '#000000'),
- 'stroke-opacity': shadowElementOpacity * i,
- 'stroke-width': strokeWidth,
- transform: transform,
- fill: 'none'
- });
- shadow.setAttribute('class', (shadow.getAttribute('class') || '') + ' highcharts-shadow');
- if (cutOff) {
- attr(shadow, 'height', Math.max(attr(shadow, 'height') - strokeWidth, 0));
- shadow.cutHeight = strokeWidth;
- }
- if (group) {
- group.element.appendChild(shadow);
- }
- else if (element.parentNode) {
- element.parentNode.insertBefore(shadow, element);
- }
- shadows.push(shadow);
- }
- this.shadows = shadows;
- }
- return this;
- };
- /**
- * Show the element after it has been hidden.
- *
- * @function Highcharts.SVGElement#show
- *
- * @param {boolean} [inherit=false]
- * Set the visibility attribute to `inherit` rather than `visible`.
- * The difference is that an element with `visibility="visible"`
- * will be visible even if the parent is hidden.
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.show = function (inherit) {
- return this.attr({ visibility: inherit ? 'inherit' : 'visible' });
- };
- /**
- * WebKit and Batik have problems with a stroke-width of zero, so in this
- * case we remove the stroke attribute altogether. #1270, #1369, #3065,
- * #3072.
- *
- * @private
- * @function Highcharts.SVGElement#strokeSetter
- * @param {number|string} value
- * @param {string} key
- * @param {Highcharts.SVGDOMElement} element
- */
- SVGElement.prototype.strokeSetter = function (value, key, element) {
- this[key] = value;
- // Only apply the stroke attribute if the stroke width is defined and
- // larger than 0
- if (this.stroke && this['stroke-width']) {
- // Use prototype as instance may be overridden
- SVGElement.prototype.fillSetter.call(this, this.stroke, 'stroke', element);
- element.setAttribute('stroke-width', this['stroke-width']);
- this.hasStroke = true;
- }
- else if (key === 'stroke-width' && value === 0 && this.hasStroke) {
- element.removeAttribute('stroke');
- this.hasStroke = false;
- }
- else if (this.renderer.styledMode && this['stroke-width']) {
- element.setAttribute('stroke-width', this['stroke-width']);
- this.hasStroke = true;
- }
- };
- /**
- * Get the computed stroke width in pixel values. This is used extensively
- * when drawing shapes to ensure the shapes are rendered crisp and
- * positioned correctly relative to each other. Using
- * `shape-rendering: crispEdges` leaves us less control over positioning,
- * for example when we want to stack columns next to each other, or position
- * things pixel-perfectly within the plot box.
- *
- * The common pattern when placing a shape is:
- * - Create the SVGElement and add it to the DOM. In styled mode, it will
- * now receive a stroke width from the style sheet. In classic mode we
- * will add the `stroke-width` attribute.
- * - Read the computed `elem.strokeWidth()`.
- * - Place it based on the stroke width.
- *
- * @function Highcharts.SVGElement#strokeWidth
- *
- * @return {number}
- * The stroke width in pixels. Even if the given stroke widtch (in CSS or by
- * attributes) is based on `em` or other units, the pixel size is returned.
- */
- SVGElement.prototype.strokeWidth = function () {
- // In non-styled mode, read the stroke width as set by .attr
- if (!this.renderer.styledMode) {
- return this['stroke-width'] || 0;
- }
- // In styled mode, read computed stroke width
- var val = this.getStyle('stroke-width'),
- ret = 0,
- dummy;
- // Read pixel values directly
- if (val.indexOf('px') === val.length - 2) {
- ret = pInt(val);
- // Other values like em, pt etc need to be measured
- }
- else if (val !== '') {
- dummy = doc.createElementNS(SVG_NS, 'rect');
- attr(dummy, {
- width: val,
- 'stroke-width': 0
- });
- this.element.parentNode.appendChild(dummy);
- ret = dummy.getBBox().width;
- dummy.parentNode.removeChild(dummy);
- }
- return ret;
- };
- /**
- * If one of the symbol size affecting parameters are changed,
- * check all the others only once for each call to an element's
- * .attr() method
- *
- * @private
- * @function Highcharts.SVGElement#symbolAttr
- *
- * @param {Highcharts.SVGAttributes} hash
- * The attributes to set.
- */
- SVGElement.prototype.symbolAttr = function (hash) {
- var wrapper = this;
- [
- 'x',
- 'y',
- 'r',
- 'start',
- 'end',
- 'width',
- 'height',
- 'innerR',
- 'anchorX',
- 'anchorY',
- 'clockwise'
- ].forEach(function (key) {
- wrapper[key] = pick(hash[key], wrapper[key]);
- });
- wrapper.attr({
- d: wrapper.renderer.symbols[wrapper.symbolName](wrapper.x, wrapper.y, wrapper.width, wrapper.height, wrapper)
- });
- };
- /**
- * @private
- * @function Highcharts.SVGElement#textSetter
- * @param {string} value
- */
- SVGElement.prototype.textSetter = function (value) {
- if (value !== this.textStr) {
- // Delete size caches when the text changes
- // delete this.bBox; // old code in series-label
- delete this.textPxLength;
- this.textStr = value;
- if (this.added) {
- this.renderer.buildText(this);
- }
- }
- };
- /**
- * @private
- * @function Highcharts.SVGElement#titleSetter
- * @param {string} value
- */
- SVGElement.prototype.titleSetter = function (value) {
- var titleNode = this.element.getElementsByTagName('title')[0];
- if (!titleNode) {
- titleNode = doc.createElementNS(this.SVG_NS, 'title');
- this.element.appendChild(titleNode);
- }
- // Remove text content if it exists
- if (titleNode.firstChild) {
- titleNode.removeChild(titleNode.firstChild);
- }
- titleNode.appendChild(doc.createTextNode(
- // #3276, #3895
- String(pick(value, ''))
- .replace(/<[^>]*>/g, '')
- .replace(/</g, '<')
- .replace(/>/g, '>')));
- };
- /**
- * Bring the element to the front. Alternatively, a new zIndex can be set.
- *
- * @sample highcharts/members/element-tofront/
- * Click an element to bring it to front
- *
- * @function Highcharts.SVGElement#toFront
- *
- * @return {Highcharts.SVGElement}
- * Returns the SVGElement for chaining.
- */
- SVGElement.prototype.toFront = function () {
- var element = this.element;
- element.parentNode.appendChild(element);
- return this;
- };
- /**
- * Move an object and its children by x and y values.
- *
- * @function Highcharts.SVGElement#translate
- *
- * @param {number} x
- * The x value.
- *
- * @param {number} y
- * The y value.
- *
- * @return {Highcharts.SVGElement}
- */
- SVGElement.prototype.translate = function (x, y) {
- return this.attr({
- translateX: x,
- translateY: y
- });
- };
- /**
- * Update the shadow elements with new attributes.
- *
- * @private
- * @function Highcharts.SVGElement#updateShadows
- *
- * @param {string} key
- * The attribute name.
- *
- * @param {number} value
- * The value of the attribute.
- *
- * @param {Function} setter
- * The setter function, inherited from the parent wrapper.
- */
- SVGElement.prototype.updateShadows = function (key, value, setter) {
- var shadows = this.shadows;
- if (shadows) {
- var i = shadows.length;
- while (i--) {
- setter.call(shadows[i], key === 'height' ?
- Math.max(value - (shadows[i].cutHeight || 0), 0) :
- key === 'd' ? this.d : value, key, shadows[i]);
- }
- }
- };
- /**
- * Update the transform attribute based on internal properties. Deals with
- * the custom `translateX`, `translateY`, `rotation`, `scaleX` and `scaleY`
- * attributes and updates the SVG `transform` attribute.
- *
- * @private
- * @function Highcharts.SVGElement#updateTransform
- */
- SVGElement.prototype.updateTransform = function () {
- var wrapper = this,
- translateX = wrapper.translateX || 0,
- translateY = wrapper.translateY || 0,
- scaleX = wrapper.scaleX,
- scaleY = wrapper.scaleY,
- inverted = wrapper.inverted,
- rotation = wrapper.rotation,
- matrix = wrapper.matrix,
- element = wrapper.element,
- transform;
- // Flipping affects translate as adjustment for flipping around the
- // group's axis
- if (inverted) {
- translateX += wrapper.width;
- translateY += wrapper.height;
- }
- // Apply translate. Nearly all transformed elements have translation,
- // so instead of checking for translate = 0, do it always (#1767,
- // #1846).
- transform = ['translate(' + translateX + ',' + translateY + ')'];
- // apply matrix
- if (defined(matrix)) {
- transform.push('matrix(' + matrix.join(',') + ')');
- }
- // apply rotation
- if (inverted) {
- transform.push('rotate(90) scale(-1,1)');
- }
- else if (rotation) { // text rotation
- transform.push('rotate(' + rotation + ' ' +
- pick(this.rotationOriginX, element.getAttribute('x'), 0) +
- ' ' +
- pick(this.rotationOriginY, element.getAttribute('y') || 0) + ')');
- }
- // apply scale
- if (defined(scaleX) || defined(scaleY)) {
- transform.push('scale(' + pick(scaleX, 1) + ' ' + pick(scaleY, 1) + ')');
- }
- if (transform.length) {
- element.setAttribute('transform', transform.join(' '));
- }
- };
- /**
- * @private
- * @function Highcharts.SVGElement#visibilitySetter
- *
- * @param {string} value
- *
- * @param {string} key
- *
- * @param {Highcharts.SVGDOMElement} element
- *
- * @return {void}
- */
- SVGElement.prototype.visibilitySetter = function (value, key, element) {
- // IE9-11 doesn't handle visibilty:inherit well, so we remove the
- // attribute instead (#2881, #3909)
- if (value === 'inherit') {
- element.removeAttribute(key);
- }
- else if (this[key] !== value) { // #6747
- element.setAttribute(key, value);
- }
- this[key] = value;
- };
- /**
- * @private
- * @function Highcharts.SVGElement#xGetter
- *
- * @param {string} key
- *
- * @return {number|string|null}
- */
- SVGElement.prototype.xGetter = function (key) {
- if (this.element.nodeName === 'circle') {
- if (key === 'x') {
- key = 'cx';
- }
- else if (key === 'y') {
- key = 'cy';
- }
- }
- return this._defaultGetter(key);
- };
- /**
- * @private
- * @function Highcharts.SVGElement#zIndexSetter
- * @param {number} [value]
- * @param {string} [key]
- * @return {boolean}
- */
- SVGElement.prototype.zIndexSetter = function (value, key) {
- var renderer = this.renderer,
- parentGroup = this.parentGroup,
- parentWrapper = parentGroup || renderer,
- parentNode = parentWrapper.element || renderer.box,
- childNodes,
- otherElement,
- otherZIndex,
- element = this.element,
- inserted = false,
- undefinedOtherZIndex,
- svgParent = parentNode === renderer.box,
- run = this.added,
- i;
- if (defined(value)) {
- // So we can read it for other elements in the group
- element.setAttribute('data-z-index', value);
- value = +value;
- if (this[key] === value) {
- // Only update when needed (#3865)
- run = false;
- }
- }
- else if (defined(this[key])) {
- element.removeAttribute('data-z-index');
- }
- this[key] = value;
- // Insert according to this and other elements' zIndex. Before .add() is
- // called, nothing is done. Then on add, or by later calls to
- // zIndexSetter, the node is placed on the right place in the DOM.
- if (run) {
- value = this.zIndex;
- if (value && parentGroup) {
- parentGroup.handleZ = true;
- }
- childNodes = parentNode.childNodes;
- for (i = childNodes.length - 1; i >= 0 && !inserted; i--) {
- otherElement = childNodes[i];
- otherZIndex = otherElement.getAttribute('data-z-index');
- undefinedOtherZIndex = !defined(otherZIndex);
- if (otherElement !== element) {
- if (
- // Negative zIndex versus no zIndex:
- // On all levels except the highest. If the parent is
- // <svg>, then we don't want to put items before <desc>
- // or <defs>
- value < 0 &&
- undefinedOtherZIndex &&
- !svgParent &&
- !i) {
- parentNode.insertBefore(element, childNodes[i]);
- inserted = true;
- }
- else if (
- // Insert after the first element with a lower zIndex
- pInt(otherZIndex) <= value ||
- // If negative zIndex, add this before first undefined
- // zIndex element
- (undefinedOtherZIndex &&
- (!defined(value) || value >= 0))) {
- parentNode.insertBefore(element, childNodes[i + 1] || null // null for oldIE export
- );
- inserted = true;
- }
- }
- }
- if (!inserted) {
- parentNode.insertBefore(element, childNodes[svgParent ? 3 : 0] || null // null for oldIE
- );
- inserted = true;
- }
- }
- return inserted;
- };
- return SVGElement;
- }());
- // Some shared setters and getters
- SVGElement.prototype['stroke-widthSetter'] = SVGElement.prototype.strokeSetter;
- SVGElement.prototype.yGetter = SVGElement.prototype.xGetter;
- SVGElement.prototype.matrixSetter =
- SVGElement.prototype.rotationOriginXSetter =
- SVGElement.prototype.rotationOriginYSetter =
- SVGElement.prototype.rotationSetter =
- SVGElement.prototype.scaleXSetter =
- SVGElement.prototype.scaleYSetter =
- SVGElement.prototype.translateXSetter =
- SVGElement.prototype.translateYSetter =
- SVGElement.prototype.verticalAlignSetter = function (value, key) {
- this[key] = value;
- this.doTransform = true;
- };
- H.SVGElement = SVGElement;
- return H.SVGElement;
- });
- _registerModule(_modules, 'Core/Renderer/SVG/SVGLabel.js', [_modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (SVGElement, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var __extends = (this && this.__extends) || (function () {
- var extendStatics = function (d,
- b) {
- extendStatics = Object.setPrototypeOf ||
- ({ __proto__: [] } instanceof Array && function (d,
- b) { d.__proto__ = b; }) ||
- function (d,
- b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
- return extendStatics(d, b);
- };
- return function (d, b) {
- extendStatics(d, b);
- function __() { this.constructor = d; }
- d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
- };
- })();
- var defined = U.defined,
- extend = U.extend,
- isNumber = U.isNumber,
- merge = U.merge,
- removeEvent = U.removeEvent;
- /**
- * SVG label to render text.
- * @private
- * @class
- * @name Highcharts.SVGLabel
- * @augments Highcharts.SVGElement
- */
- var SVGLabel = /** @class */ (function (_super) {
- __extends(SVGLabel, _super);
- /* *
- *
- * Constructors
- *
- * */
- function SVGLabel(renderer, str, x, y, shape, anchorX, anchorY, useHTML, baseline, className) {
- var _this = _super.call(this) || this;
- _this.init(renderer, 'g');
- _this.textStr = str;
- _this.x = x;
- _this.y = y;
- _this.anchorX = anchorX;
- _this.anchorY = anchorY;
- _this.baseline = baseline;
- _this.className = className;
- if (className !== 'button') {
- _this.addClass('highcharts-label');
- }
- if (className) {
- _this.addClass('highcharts-' + className);
- }
- _this.text = renderer.text('', 0, 0, useHTML)
- .attr({
- zIndex: 1
- });
- // Validate the shape argument
- var hasBGImage;
- if (typeof shape === 'string') {
- hasBGImage = /^url\((.*?)\)$/.test(shape);
- if (_this.renderer.symbols[shape] || hasBGImage) {
- _this.symbolKey = shape;
- }
- }
- _this.bBox = SVGLabel.emptyBBox;
- _this.padding = 3;
- _this.paddingLeft = 0;
- _this.baselineOffset = 0;
- _this.needsBox = renderer.styledMode || hasBGImage;
- _this.deferredAttr = {};
- _this.alignFactor = 0;
- return _this;
- }
- /* *
- *
- * Functions
- *
- * */
- SVGLabel.prototype.alignSetter = function (value) {
- var alignFactor = {
- left: 0,
- center: 0.5,
- right: 1
- }[value];
- if (alignFactor !== this.alignFactor) {
- this.alignFactor = alignFactor;
- // Bounding box exists, means we're dynamically changing
- if (this.bBox && isNumber(this.xSetting)) {
- this.attr({ x: this.xSetting }); // #5134
- }
- }
- };
- SVGLabel.prototype.anchorXSetter = function (value, key) {
- this.anchorX = value;
- this.boxAttr(key, Math.round(value) - this.getCrispAdjust() - this.xSetting);
- };
- SVGLabel.prototype.anchorYSetter = function (value, key) {
- this.anchorY = value;
- this.boxAttr(key, value - this.ySetting);
- };
- /*
- * Set a box attribute, or defer it if the box is not yet created
- */
- SVGLabel.prototype.boxAttr = function (key, value) {
- if (this.box) {
- this.box.attr(key, value);
- }
- else {
- this.deferredAttr[key] = value;
- }
- };
- /*
- * Pick up some properties and apply them to the text instead of the
- * wrapper.
- */
- SVGLabel.prototype.css = function (styles) {
- if (styles) {
- var textStyles = {},
- isWidth,
- isFontStyle;
- // Create a copy to avoid altering the original object
- // (#537)
- styles = merge(styles);
- SVGLabel.textProps.forEach(function (prop) {
- if (typeof styles[prop] !== 'undefined') {
- textStyles[prop] = styles[prop];
- delete styles[prop];
- }
- });
- this.text.css(textStyles);
- isWidth = 'width' in textStyles;
- isFontStyle = 'fontSize' in textStyles ||
- 'fontWeight' in textStyles;
- // Update existing text, box (#9400, #12163)
- if (isWidth || isFontStyle) {
- this.updateBoxSize();
- // Keep updated (#9400, #12163)
- if (isFontStyle) {
- this.updateTextPadding();
- }
- }
- }
- return SVGElement.prototype.css.call(this, styles);
- };
- /*
- * Destroy and release memory.
- */
- SVGLabel.prototype.destroy = function () {
- // Added by button implementation
- removeEvent(this.element, 'mouseenter');
- removeEvent(this.element, 'mouseleave');
- if (this.text) {
- this.text.destroy();
- }
- if (this.box) {
- this.box = this.box.destroy();
- }
- // Call base implementation to destroy the rest
- SVGElement.prototype.destroy.call(this);
- return void 0;
- };
- SVGLabel.prototype.fillSetter = function (value, key) {
- if (value) {
- this.needsBox = true;
- }
- // for animation getter (#6776)
- this.fill = value;
- this.boxAttr(key, value);
- };
- /*
- * Return the bounding box of the box, not the group.
- */
- SVGLabel.prototype.getBBox = function () {
- var bBox = this.bBox;
- var padding = this.padding;
- return {
- width: bBox.width + 2 * padding,
- height: bBox.height + 2 * padding,
- x: bBox.x - padding,
- y: bBox.y - padding
- };
- };
- SVGLabel.prototype.getCrispAdjust = function () {
- return this.renderer.styledMode && this.box ?
- this.box.strokeWidth() % 2 / 2 :
- (this['stroke-width'] ? parseInt(this['stroke-width'], 10) : 0) % 2 / 2;
- };
- SVGLabel.prototype.heightSetter = function (value) {
- this.heightSetting = value;
- };
- // Event handling. In case of useHTML, we need to make sure that events
- // are captured on the span as well, and that mouseenter/mouseleave
- // between the SVG group and the HTML span are not treated as real
- // enter/leave events. #13310.
- SVGLabel.prototype.on = function (eventType, handler) {
- var label = this;
- var text = label.text;
- var span = text && text.element.tagName === 'SPAN' ? text : void 0;
- var selectiveHandler;
- if (span) {
- selectiveHandler = function (e) {
- if ((eventType === 'mouseenter' ||
- eventType === 'mouseleave') &&
- e.relatedTarget instanceof Element &&
- (label.element.contains(e.relatedTarget) ||
- span.element.contains(e.relatedTarget))) {
- return;
- }
- handler.call(label.element, e);
- };
- span.on(eventType, selectiveHandler);
- }
- SVGElement.prototype.on.call(label, eventType, selectiveHandler || handler);
- return label;
- };
- /*
- * After the text element is added, get the desired size of the border
- * box and add it before the text in the DOM.
- */
- SVGLabel.prototype.onAdd = function () {
- var str = this.textStr;
- this.text.add(this);
- this.attr({
- // Alignment is available now (#3295, 0 not rendered if given
- // as a value)
- text: (defined(str) ? str : ''),
- x: this.x,
- y: this.y
- });
- if (this.box && defined(this.anchorX)) {
- this.attr({
- anchorX: this.anchorX,
- anchorY: this.anchorY
- });
- }
- };
- SVGLabel.prototype.paddingSetter = function (value) {
- if (defined(value) && value !== this.padding) {
- this.padding = value;
- this.updateTextPadding();
- }
- };
- SVGLabel.prototype.paddingLeftSetter = function (value) {
- if (defined(value) && value !== this.paddingLeft) {
- this.paddingLeft = value;
- this.updateTextPadding();
- }
- };
- SVGLabel.prototype.rSetter = function (value, key) {
- this.boxAttr(key, value);
- };
- SVGLabel.prototype.shadow = function (b) {
- if (b && !this.renderer.styledMode) {
- this.updateBoxSize();
- if (this.box) {
- this.box.shadow(b);
- }
- }
- return this;
- };
- SVGLabel.prototype.strokeSetter = function (value, key) {
- // for animation getter (#6776)
- this.stroke = value;
- this.boxAttr(key, value);
- };
- SVGLabel.prototype['stroke-widthSetter'] = function (value, key) {
- if (value) {
- this.needsBox = true;
- }
- this['stroke-width'] = value;
- this.boxAttr(key, value);
- };
- SVGLabel.prototype['text-alignSetter'] = function (value) {
- this.textAlign = value;
- };
- SVGLabel.prototype.textSetter = function (text) {
- if (typeof text !== 'undefined') {
- // Must use .attr to ensure transforms are done (#10009)
- this.text.attr({ text: text });
- }
- this.updateBoxSize();
- this.updateTextPadding();
- };
- /*
- * This function runs after the label is added to the DOM (when the bounding
- * box is available), and after the text of the label is updated to detect
- * the new bounding box and reflect it in the border box.
- */
- SVGLabel.prototype.updateBoxSize = function () {
- var style = this.text.element.style,
- crispAdjust,
- attribs = {};
- var padding = this.padding;
- var paddingLeft = this.paddingLeft;
- // #12165 error when width is null (auto)
- // #12163 when fontweight: bold, recalculate bBox withot cache
- // #3295 && 3514 box failure when string equals 0
- var bBox = ((!isNumber(this.widthSetting) || !isNumber(this.heightSetting) || this.textAlign) &&
- defined(this.text.textStr)) ?
- this.text.getBBox() : SVGLabel.emptyBBox;
- this.width = ((this.widthSetting || bBox.width || 0) +
- 2 * padding +
- paddingLeft);
- this.height = (this.heightSetting || bBox.height || 0) + 2 * padding;
- // Update the label-scoped y offset. Math.min because of inline
- // style (#9400)
- this.baselineOffset = padding + Math.min(this.renderer.fontMetrics(style && style.fontSize, this.text).b,
- // When the height is 0, there is no bBox, so go with the font
- // metrics. Highmaps CSS demos.
- bBox.height || Infinity);
- if (this.needsBox) {
- // Create the border box if it is not already present
- if (!this.box) {
- // Symbol definition exists (#5324)
- var box = this.box = this.symbolKey ?
- this.renderer.symbol(this.symbolKey) :
- this.renderer.rect();
- box.addClass(// Don't use label className for buttons
- (this.className === 'button' ? '' : 'highcharts-label-box') +
- (this.className ? ' highcharts-' + this.className + '-box' : ''));
- box.add(this);
- crispAdjust = this.getCrispAdjust();
- attribs.x = crispAdjust;
- attribs.y = (this.baseline ? -this.baselineOffset : 0) + crispAdjust;
- }
- // Apply the box attributes
- attribs.width = Math.round(this.width);
- attribs.height = Math.round(this.height);
- this.box.attr(extend(attribs, this.deferredAttr));
- this.deferredAttr = {};
- }
- this.bBox = bBox;
- };
- /*
- * This function runs after setting text or padding, but only if padding
- * is changed.
- */
- SVGLabel.prototype.updateTextPadding = function () {
- var text = this.text;
- // Determine y based on the baseline
- var textY = this.baseline ? 0 : this.baselineOffset;
- var textX = this.paddingLeft + this.padding;
- // compensate for alignment
- if (defined(this.widthSetting) &&
- this.bBox &&
- (this.textAlign === 'center' || this.textAlign === 'right')) {
- textX += { center: 0.5, right: 1 }[this.textAlign] *
- (this.widthSetting - this.bBox.width);
- }
- // update if anything changed
- if (textX !== text.x || textY !== text.y) {
- text.attr('x', textX);
- // #8159 - prevent misplaced data labels in treemap
- // (useHTML: true)
- if (text.hasBoxWidthChanged) {
- this.bBox = text.getBBox(true);
- this.updateBoxSize();
- }
- if (typeof textY !== 'undefined') {
- text.attr('y', textY);
- }
- }
- // record current values
- text.x = textX;
- text.y = textY;
- };
- SVGLabel.prototype.widthSetter = function (value) {
- // width:auto => null
- this.widthSetting = isNumber(value) ? value : void 0;
- };
- SVGLabel.prototype.xSetter = function (value) {
- this.x = value; // for animation getter
- if (this.alignFactor) {
- value -= this.alignFactor * ((this.widthSetting || this.bBox.width) +
- 2 * this.padding);
- // Force animation even when setting to the same value (#7898)
- this['forceAnimate:x'] = true;
- }
- this.xSetting = Math.round(value);
- this.attr('translateX', this.xSetting);
- };
- SVGLabel.prototype.ySetter = function (value) {
- this.ySetting = this.y = Math.round(value);
- this.attr('translateY', this.ySetting);
- };
- /* *
- *
- * Static Properties
- *
- * */
- SVGLabel.emptyBBox = { width: 0, height: 0, x: 0, y: 0 };
- /* *
- *
- * Properties
- *
- * */
- /**
- * For labels, these CSS properties are applied to the `text` node directly.
- *
- * @private
- * @name Highcharts.SVGLabel#textProps
- * @type {Array<string>}
- */
- SVGLabel.textProps = [
- 'color', 'cursor', 'direction', 'fontFamily', 'fontSize', 'fontStyle',
- 'fontWeight', 'lineHeight', 'textAlign', 'textDecoration',
- 'textOutline', 'textOverflow', 'width'
- ];
- return SVGLabel;
- }(SVGElement));
- return SVGLabel;
- });
- _registerModule(_modules, 'Core/Renderer/SVG/SVGRenderer.js', [_modules['Core/Color.js'], _modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Renderer/SVG/SVGLabel.js'], _modules['Core/Utilities.js']], function (Color, H, SVGElement, SVGLabel, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- attr = U.attr,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- extend = U.extend,
- isArray = U.isArray,
- isNumber = U.isNumber,
- isObject = U.isObject,
- isString = U.isString,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- pInt = U.pInt,
- splat = U.splat,
- uniqueKey = U.uniqueKey;
- /**
- * A clipping rectangle that can be applied to one or more {@link SVGElement}
- * instances. It is instanciated with the {@link SVGRenderer#clipRect} function
- * and applied with the {@link SVGElement#clip} function.
- *
- * @example
- * var circle = renderer.circle(100, 100, 100)
- * .attr({ fill: 'red' })
- * .add();
- * var clipRect = renderer.clipRect(100, 100, 100, 100);
- *
- * // Leave only the lower right quarter visible
- * circle.clip(clipRect);
- *
- * @typedef {Highcharts.SVGElement} Highcharts.ClipRectElement
- */
- /**
- * The font metrics.
- *
- * @interface Highcharts.FontMetricsObject
- */ /**
- * The baseline relative to the top of the box.
- *
- * @name Highcharts.FontMetricsObject#b
- * @type {number}
- */ /**
- * The font size.
- *
- * @name Highcharts.FontMetricsObject#f
- * @type {number}
- */ /**
- * The line height.
- *
- * @name Highcharts.FontMetricsObject#h
- * @type {number}
- */
- /**
- * An object containing `x` and `y` properties for the position of an element.
- *
- * @interface Highcharts.PositionObject
- */ /**
- * X position of the element.
- * @name Highcharts.PositionObject#x
- * @type {number}
- */ /**
- * Y position of the element.
- * @name Highcharts.PositionObject#y
- * @type {number}
- */
- /**
- * A rectangle.
- *
- * @interface Highcharts.RectangleObject
- */ /**
- * Height of the rectangle.
- * @name Highcharts.RectangleObject#height
- * @type {number}
- */ /**
- * Width of the rectangle.
- * @name Highcharts.RectangleObject#width
- * @type {number}
- */ /**
- * Horizontal position of the rectangle.
- * @name Highcharts.RectangleObject#x
- * @type {number}
- */ /**
- * Vertical position of the rectangle.
- * @name Highcharts.RectangleObject#y
- * @type {number}
- */
- /**
- * The shadow options.
- *
- * @interface Highcharts.ShadowOptionsObject
- */ /**
- * The shadow color.
- * @name Highcharts.ShadowOptionsObject#color
- * @type {Highcharts.ColorString|undefined}
- * @default #000000
- */ /**
- * The horizontal offset from the element.
- *
- * @name Highcharts.ShadowOptionsObject#offsetX
- * @type {number|undefined}
- * @default 1
- */ /**
- * The vertical offset from the element.
- * @name Highcharts.ShadowOptionsObject#offsetY
- * @type {number|undefined}
- * @default 1
- */ /**
- * The shadow opacity.
- *
- * @name Highcharts.ShadowOptionsObject#opacity
- * @type {number|undefined}
- * @default 0.15
- */ /**
- * The shadow width or distance from the element.
- * @name Highcharts.ShadowOptionsObject#width
- * @type {number|undefined}
- * @default 3
- */
- /**
- * @interface Highcharts.SizeObject
- */ /**
- * @name Highcharts.SizeObject#height
- * @type {number}
- */ /**
- * @name Highcharts.SizeObject#width
- * @type {number}
- */
- /**
- * Serialized form of an SVG definition, including children. Some key
- * property names are reserved: tagName, textContent, and children.
- *
- * @interface Highcharts.SVGDefinitionObject
- */ /**
- * @name Highcharts.SVGDefinitionObject#[key:string]
- * @type {boolean|number|string|Array<Highcharts.SVGDefinitionObject>|undefined}
- */ /**
- * @name Highcharts.SVGDefinitionObject#children
- * @type {Array<Highcharts.SVGDefinitionObject>|undefined}
- */ /**
- * @name Highcharts.SVGDefinitionObject#tagName
- * @type {string|undefined}
- */ /**
- * @name Highcharts.SVGDefinitionObject#textContent
- * @type {string|undefined}
- */
- /**
- * Array of path commands, that will go into the `d` attribute of an SVG
- * element.
- *
- * @typedef {Array<(Array<Highcharts.SVGPathCommand>|Array<Highcharts.SVGPathCommand,number>|Array<Highcharts.SVGPathCommand,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number>|Array<Highcharts.SVGPathCommand,number,number,number,number,number,number,number>)>} Highcharts.SVGPathArray
- */
- /**
- * Possible path commands in an SVG path array. Valid values are `A`, `C`, `H`,
- * `L`, `M`, `Q`, `S`, `T`, `V`, `Z`.
- *
- * @typedef {string} Highcharts.SVGPathCommand
- * @validvalue ["a","c","h","l","m","q","s","t","v","z","A","C","H","L","M","Q","S","T","V","Z"]
- */
- /**
- * An extendable collection of functions for defining symbol paths. Symbols are
- * used internally for point markers, button and label borders and backgrounds,
- * or custom shapes. Extendable by adding to {@link SVGRenderer#symbols}.
- *
- * @interface Highcharts.SymbolDictionary
- */ /**
- * @name Highcharts.SymbolDictionary#[key:string]
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#arc
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#callout
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#circle
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#diamond
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#square
- * @type {Function|undefined}
- */ /**
- * @name Highcharts.SymbolDictionary#triangle
- * @type {Function|undefined}
- */
- /**
- * Can be one of `arc`, `callout`, `circle`, `diamond`, `square`, `triangle`,
- * and `triangle-down`. Symbols are used internally for point markers, button
- * and label borders and backgrounds, or custom shapes. Extendable by adding to
- * {@link SVGRenderer#symbols}.
- *
- * @typedef {"arc"|"callout"|"circle"|"diamond"|"square"|"triangle"|"triangle-down"} Highcharts.SymbolKeyValue
- */
- /**
- * Additional options, depending on the actual symbol drawn.
- *
- * @interface Highcharts.SymbolOptionsObject
- */ /**
- * The anchor X position for the `callout` symbol. This is where the chevron
- * points to.
- *
- * @name Highcharts.SymbolOptionsObject#anchorX
- * @type {number|undefined}
- */ /**
- * The anchor Y position for the `callout` symbol. This is where the chevron
- * points to.
- *
- * @name Highcharts.SymbolOptionsObject#anchorY
- * @type {number|undefined}
- */ /**
- * The end angle of an `arc` symbol.
- *
- * @name Highcharts.SymbolOptionsObject#end
- * @type {number|undefined}
- */ /**
- * Whether to draw `arc` symbol open or closed.
- *
- * @name Highcharts.SymbolOptionsObject#open
- * @type {boolean|undefined}
- */ /**
- * The radius of an `arc` symbol, or the border radius for the `callout` symbol.
- *
- * @name Highcharts.SymbolOptionsObject#r
- * @type {number|undefined}
- */ /**
- * The start angle of an `arc` symbol.
- *
- * @name Highcharts.SymbolOptionsObject#start
- * @type {number|undefined}
- */
- /* eslint-disable no-invalid-this, valid-jsdoc */
- var charts = H.charts,
- deg2rad = H.deg2rad,
- doc = H.doc,
- isFirefox = H.isFirefox,
- isMS = H.isMS,
- isWebKit = H.isWebKit,
- noop = H.noop,
- svg = H.svg,
- SVG_NS = H.SVG_NS,
- symbolSizes = H.symbolSizes,
- win = H.win;
- /**
- * Allows direct access to the Highcharts rendering layer in order to draw
- * primitive shapes like circles, rectangles, paths or text directly on a chart,
- * or independent from any chart. The SVGRenderer represents a wrapper object
- * for SVG in modern browsers. Through the VMLRenderer, part of the `oldie.js`
- * module, it also brings vector graphics to IE <= 8.
- *
- * An existing chart's renderer can be accessed through {@link Chart.renderer}.
- * The renderer can also be used completely decoupled from a chart.
- *
- * @sample highcharts/members/renderer-on-chart
- * Annotating a chart programmatically.
- * @sample highcharts/members/renderer-basic
- * Independent SVG drawing.
- *
- * @example
- * // Use directly without a chart object.
- * var renderer = new Highcharts.Renderer(parentNode, 600, 400);
- *
- * @class
- * @name Highcharts.SVGRenderer
- *
- * @param {Highcharts.HTMLDOMElement} container
- * Where to put the SVG in the web page.
- *
- * @param {number} width
- * The width of the SVG.
- *
- * @param {number} height
- * The height of the SVG.
- *
- * @param {Highcharts.CSSObject} [style]
- * The box style, if not in styleMode
- *
- * @param {boolean} [forExport=false]
- * Whether the rendered content is intended for export.
- *
- * @param {boolean} [allowHTML=true]
- * Whether the renderer is allowed to include HTML text, which will be
- * projected on top of the SVG.
- *
- * @param {boolean} [styledMode=false]
- * Whether the renderer belongs to a chart that is in styled mode.
- * If it does, it will avoid setting presentational attributes in
- * some cases, but not when set explicitly through `.attr` and `.css`
- * etc.
- */
- var SVGRenderer = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function SVGRenderer(container, width, height, style, forExport, allowHTML, styledMode) {
- /* *
- *
- * Properties
- *
- * */
- this.alignedObjects = void 0;
- /**
- * The root `svg` node of the renderer.
- *
- * @name Highcharts.SVGRenderer#box
- * @type {Highcharts.SVGDOMElement}
- */
- this.box = void 0;
- /**
- * The wrapper for the root `svg` node of the renderer.
- *
- * @name Highcharts.SVGRenderer#boxWrapper
- * @type {Highcharts.SVGElement}
- */
- this.boxWrapper = void 0;
- this.cache = void 0;
- this.cacheKeys = void 0;
- this.chartIndex = void 0;
- /**
- * A pointer to the `defs` node of the root SVG.
- *
- * @name Highcharts.SVGRenderer#defs
- * @type {Highcharts.SVGElement}
- */
- this.defs = void 0;
- this.globalAnimation = void 0;
- this.gradients = void 0;
- this.height = void 0;
- this.imgCount = void 0;
- this.isSVG = void 0;
- this.style = void 0;
- /**
- * Page url used for internal references.
- *
- * @private
- * @name Highcharts.SVGRenderer#url
- * @type {string}
- */
- this.url = void 0;
- this.width = void 0;
- this.init(container, width, height, style, forExport, allowHTML, styledMode);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Initialize the SVGRenderer. Overridable initializer function that takes
- * the same parameters as the constructor.
- *
- * @function Highcharts.SVGRenderer#init
- *
- * @param {Highcharts.HTMLDOMElement} container
- * Where to put the SVG in the web page.
- *
- * @param {number} width
- * The width of the SVG.
- *
- * @param {number} height
- * The height of the SVG.
- *
- * @param {Highcharts.CSSObject} [style]
- * The box style, if not in styleMode
- *
- * @param {boolean} [forExport=false]
- * Whether the rendered content is intended for export.
- *
- * @param {boolean} [allowHTML=true]
- * Whether the renderer is allowed to include HTML text, which will be
- * projected on top of the SVG.
- *
- * @param {boolean} [styledMode=false]
- * Whether the renderer belongs to a chart that is in styled mode. If it
- * does, it will avoid setting presentational attributes in some cases, but
- * not when set explicitly through `.attr` and `.css` etc.
- */
- SVGRenderer.prototype.init = function (container, width, height, style, forExport, allowHTML, styledMode) {
- var renderer = this,
- boxWrapper,
- element,
- desc;
- boxWrapper = renderer.createElement('svg')
- .attr({
- version: '1.1',
- 'class': 'highcharts-root'
- });
- if (!styledMode) {
- boxWrapper.css(this.getStyle(style));
- }
- element = boxWrapper.element;
- container.appendChild(element);
- // Always use ltr on the container, otherwise text-anchor will be
- // flipped and text appear outside labels, buttons, tooltip etc (#3482)
- attr(container, 'dir', 'ltr');
- // For browsers other than IE, add the namespace attribute (#1978)
- if (container.innerHTML.indexOf('xmlns') === -1) {
- attr(element, 'xmlns', this.SVG_NS);
- }
- // object properties
- renderer.isSVG = true;
- this.box = element;
- this.boxWrapper = boxWrapper;
- renderer.alignedObjects = [];
- // #24, #672, #1070
- this.url = ((isFirefox || isWebKit) &&
- doc.getElementsByTagName('base').length) ?
- win.location.href
- .split('#')[0] // remove the hash
- .replace(/<[^>]*>/g, '') // wing cut HTML
- // escape parantheses and quotes
- .replace(/([\('\)])/g, '\\$1')
- // replace spaces (needed for Safari only)
- .replace(/ /g, '%20') :
- '';
- // Add description
- desc = this.createElement('desc').add();
- desc.element.appendChild(doc.createTextNode('Created with Highcharts 8.2.0'));
- renderer.defs = this.createElement('defs').add();
- renderer.allowHTML = allowHTML;
- renderer.forExport = forExport;
- renderer.styledMode = styledMode;
- renderer.gradients = {}; // Object where gradient SvgElements are stored
- renderer.cache = {}; // Cache for numerical bounding boxes
- renderer.cacheKeys = [];
- renderer.imgCount = 0;
- renderer.setSize(width, height, false);
- // Issue 110 workaround:
- // In Firefox, if a div is positioned by percentage, its pixel position
- // may land between pixels. The container itself doesn't display this,
- // but an SVG element inside this container will be drawn at subpixel
- // precision. In order to draw sharp lines, this must be compensated
- // for. This doesn't seem to work inside iframes though (like in
- // jsFiddle).
- var subPixelFix,
- rect;
- if (isFirefox && container.getBoundingClientRect) {
- subPixelFix = function () {
- css(container, { left: 0, top: 0 });
- rect = container.getBoundingClientRect();
- css(container, {
- left: (Math.ceil(rect.left) - rect.left) + 'px',
- top: (Math.ceil(rect.top) - rect.top) + 'px'
- });
- };
- // run the fix now
- subPixelFix();
- // run it on resize
- renderer.unSubPixelFix = addEvent(win, 'resize', subPixelFix);
- }
- };
- /**
- * General method for adding a definition to the SVG `defs` tag. Can be used
- * for gradients, fills, filters etc. Styled mode only. A hook for adding
- * general definitions to the SVG's defs tag. Definitions can be referenced
- * from the CSS by its `id`. Read more in
- * [gradients, shadows and patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns).
- * Styled mode only.
- *
- * @function Highcharts.SVGRenderer#definition
- *
- * @param {Highcharts.SVGDefinitionObject} def
- * A serialized form of an SVG definition, including children.
- *
- * @return {Highcharts.SVGElement}
- * The inserted node.
- */
- SVGRenderer.prototype.definition = function (def) {
- var ren = this;
- /**
- * @private
- * @param {Highcharts.SVGDefinitionObject} config - SVG definition
- * @param {Highcharts.SVGElement} [parent] - parent node
- */
- function recurse(config, parent) {
- var ret;
- splat(config).forEach(function (item) {
- var node = ren.createElement(item.tagName),
- attr = {};
- // Set attributes
- objectEach(item, function (val, key) {
- if (key !== 'tagName' &&
- key !== 'children' &&
- key !== 'textContent') {
- attr[key] = val;
- }
- });
- node.attr(attr);
- // Add to the tree
- node.add(parent || ren.defs);
- // Add text content
- if (item.textContent) {
- node.element.appendChild(doc.createTextNode(item.textContent));
- }
- // Recurse
- recurse(item.children || [], node);
- ret = node;
- });
- // Return last node added (on top level it's the only one)
- return ret;
- }
- return recurse(def);
- };
- /**
- * Get the global style setting for the renderer.
- *
- * @private
- * @function Highcharts.SVGRenderer#getStyle
- *
- * @param {Highcharts.CSSObject} style
- * Style settings.
- *
- * @return {Highcharts.CSSObject}
- * The style settings mixed with defaults.
- */
- SVGRenderer.prototype.getStyle = function (style) {
- this.style = extend({
- fontFamily: '"Lucida Grande", "Lucida Sans Unicode", ' +
- 'Arial, Helvetica, sans-serif',
- fontSize: '12px'
- }, style);
- return this.style;
- };
- /**
- * Apply the global style on the renderer, mixed with the default styles.
- *
- * @function Highcharts.SVGRenderer#setStyle
- *
- * @param {Highcharts.CSSObject} style
- * CSS to apply.
- */
- SVGRenderer.prototype.setStyle = function (style) {
- this.boxWrapper.css(this.getStyle(style));
- };
- /**
- * Detect whether the renderer is hidden. This happens when one of the
- * parent elements has `display: none`. Used internally to detect when we
- * needto render preliminarily in another div to get the text bounding boxes
- * right.
- *
- * @function Highcharts.SVGRenderer#isHidden
- *
- * @return {boolean}
- * True if it is hidden.
- */
- SVGRenderer.prototype.isHidden = function () {
- return !this.boxWrapper.getBBox().width;
- };
- /**
- * Destroys the renderer and its allocated members.
- *
- * @function Highcharts.SVGRenderer#destroy
- *
- * @return {null}
- */
- SVGRenderer.prototype.destroy = function () {
- var renderer = this,
- rendererDefs = renderer.defs;
- renderer.box = null;
- renderer.boxWrapper = renderer.boxWrapper.destroy();
- // Call destroy on all gradient elements
- destroyObjectProperties(renderer.gradients || {});
- renderer.gradients = null;
- // Defs are null in VMLRenderer
- // Otherwise, destroy them here.
- if (rendererDefs) {
- renderer.defs = rendererDefs.destroy();
- }
- // Remove sub pixel fix handler (#982)
- if (renderer.unSubPixelFix) {
- renderer.unSubPixelFix();
- }
- renderer.alignedObjects = null;
- return null;
- };
- /**
- * Create a wrapper for an SVG element. Serves as a factory for
- * {@link SVGElement}, but this function is itself mostly called from
- * primitive factories like {@link SVGRenderer#path}, {@link
- * SVGRenderer#rect} or {@link SVGRenderer#text}.
- *
- * @function Highcharts.SVGRenderer#createElement
- *
- * @param {string} nodeName
- * The node name, for example `rect`, `g` etc.
- *
- * @return {Highcharts.SVGElement}
- * The generated SVGElement.
- */
- SVGRenderer.prototype.createElement = function (nodeName) {
- var wrapper = new this.Element();
- wrapper.init(this, nodeName);
- return wrapper;
- };
- /**
- * Get converted radial gradient attributes according to the radial
- * reference. Used internally from the {@link SVGElement#colorGradient}
- * function.
- *
- * @private
- * @function Highcharts.SVGRenderer#getRadialAttr
- */
- SVGRenderer.prototype.getRadialAttr = function (radialReference, gradAttr) {
- return {
- cx: (radialReference[0] - radialReference[2] / 2) +
- gradAttr.cx * radialReference[2],
- cy: (radialReference[1] - radialReference[2] / 2) +
- gradAttr.cy * radialReference[2],
- r: gradAttr.r * radialReference[2]
- };
- };
- /**
- * Truncate the text node contents to a given length. Used when the css
- * width is set. If the `textOverflow` is `ellipsis`, the text is truncated
- * character by character to the given length. If not, the text is
- * word-wrapped line by line.
- *
- * @private
- * @function Highcharts.SVGRenderer#truncate
- *
- * @return {boolean}
- * True if tspan is too long.
- */
- SVGRenderer.prototype.truncate = function (wrapper, tspan, text, words, startAt, width, getString) {
- var renderer = this,
- rotation = wrapper.rotation,
- str,
- // Word wrap can not be truncated to shorter than one word, ellipsis
- // text can be completely blank.
- minIndex = words ? 1 : 0,
- maxIndex = (text || words).length,
- currentIndex = maxIndex,
- // Cache the lengths to avoid checking the same twice
- lengths = [],
- updateTSpan = function (s) {
- if (tspan.firstChild) {
- tspan.removeChild(tspan.firstChild);
- }
- if (s) {
- tspan.appendChild(doc.createTextNode(s));
- }
- }, getSubStringLength = function (charEnd, concatenatedEnd) {
- // charEnd is useed when finding the character-by-character
- // break for ellipsis, concatenatedEnd is used for word-by-word
- // break for word wrapping.
- var end = concatenatedEnd || charEnd;
- if (typeof lengths[end] === 'undefined') {
- // Modern browsers
- if (tspan.getSubStringLength) {
- // Fails with DOM exception on unit-tests/legend/members
- // of unknown reason. Desired width is 0, text content
- // is "5" and end is 1.
- try {
- lengths[end] = startAt +
- tspan.getSubStringLength(0, words ? end + 1 : end);
- }
- catch (e) {
- '';
- }
- // Legacy
- }
- else if (renderer.getSpanWidth) { // #9058 jsdom
- updateTSpan(getString(text || words, charEnd));
- lengths[end] = startAt +
- renderer.getSpanWidth(wrapper, tspan);
- }
- }
- return lengths[end];
- }, actualWidth, truncated;
- wrapper.rotation = 0; // discard rotation when computing box
- actualWidth = getSubStringLength(tspan.textContent.length);
- truncated = startAt + actualWidth > width;
- if (truncated) {
- // Do a binary search for the index where to truncate the text
- while (minIndex <= maxIndex) {
- currentIndex = Math.ceil((minIndex + maxIndex) / 2);
- // When checking words for word-wrap, we need to build the
- // string and measure the subStringLength at the concatenated
- // word length.
- if (words) {
- str = getString(words, currentIndex);
- }
- actualWidth = getSubStringLength(currentIndex, str && str.length - 1);
- if (minIndex === maxIndex) {
- // Complete
- minIndex = maxIndex + 1;
- }
- else if (actualWidth > width) {
- // Too large. Set max index to current.
- maxIndex = currentIndex - 1;
- }
- else {
- // Within width. Set min index to current.
- minIndex = currentIndex;
- }
- }
- // If max index was 0 it means the shortest possible text was also
- // too large. For ellipsis that means only the ellipsis, while for
- // word wrap it means the whole first word.
- if (maxIndex === 0) {
- // Remove ellipsis
- updateTSpan('');
- // If the new text length is one less than the original, we don't
- // need the ellipsis
- }
- else if (!(text && maxIndex === text.length - 1)) {
- updateTSpan(str || getString(text || words, currentIndex));
- }
- }
- // When doing line wrapping, prepare for the next line by removing the
- // items from this line.
- if (words) {
- words.splice(0, currentIndex);
- }
- wrapper.actualWidth = actualWidth;
- wrapper.rotation = rotation; // Apply rotation again.
- return truncated;
- };
- /**
- * Parse a simple HTML string into SVG tspans. Called internally when text
- * is set on an SVGElement. The function supports a subset of HTML tags, CSS
- * text features like `width`, `text-overflow`, `white-space`, and also
- * attributes like `href` and `style`.
- *
- * @private
- * @function Highcharts.SVGRenderer#buildText
- *
- * @param {Highcharts.SVGElement} wrapper
- * The parent SVGElement.
- */
- SVGRenderer.prototype.buildText = function (wrapper) {
- var textNode = wrapper.element, renderer = this, forExport = renderer.forExport, textStr = pick(wrapper.textStr, '').toString(), hasMarkup = textStr.indexOf('<') !== -1, lines, childNodes = textNode.childNodes, truncated, parentX = attr(textNode, 'x'), textStyles = wrapper.styles, width = wrapper.textWidth, textLineHeight = textStyles && textStyles.lineHeight, textOutline = textStyles && textStyles.textOutline, ellipsis = textStyles && textStyles.textOverflow === 'ellipsis', noWrap = textStyles && textStyles.whiteSpace === 'nowrap', fontSize = textStyles && textStyles.fontSize, textCache, isSubsequentLine, i = childNodes.length, tempParent = width && !wrapper.added && this.box, getLineHeight = function (tspan) {
- var fontSizeStyle;
- if (!renderer.styledMode) {
- fontSizeStyle =
- /(px|em)$/.test(tspan && tspan.style.fontSize) ?
- tspan.style.fontSize :
- (fontSize || renderer.style.fontSize || 12);
- }
- return textLineHeight ?
- pInt(textLineHeight) :
- renderer.fontMetrics(fontSizeStyle,
- // Get the computed size from parent if not explicit
- (tspan.getAttribute('style') ? tspan : textNode)).h;
- }, unescapeEntities = function (inputStr, except) {
- objectEach(renderer.escapes, function (value, key) {
- if (!except || except.indexOf(value) === -1) {
- inputStr = inputStr.toString().replace(new RegExp(value, 'g'), key);
- }
- });
- return inputStr;
- }, parseAttribute = function (s, attr) {
- var start,
- delimiter;
- start = s.indexOf('<');
- s = s.substring(start, s.indexOf('>') - start);
- start = s.indexOf(attr + '=');
- if (start !== -1) {
- start = start + attr.length + 1;
- delimiter = s.charAt(start);
- if (delimiter === '"' || delimiter === "'") { // eslint-disable-line quotes
- s = s.substring(start + 1);
- return s.substring(0, s.indexOf(delimiter));
- }
- }
- };
- var regexMatchBreaks = /<br.*?>/g;
- // The buildText code is quite heavy, so if we're not changing something
- // that affects the text, skip it (#6113).
- textCache = [
- textStr,
- ellipsis,
- noWrap,
- textLineHeight,
- textOutline,
- fontSize,
- width
- ].join(',');
- if (textCache === wrapper.textCache) {
- return;
- }
- wrapper.textCache = textCache;
- // Remove old text
- while (i--) {
- textNode.removeChild(childNodes[i]);
- }
- // Skip tspans, add text directly to text node. The forceTSpan is a hook
- // used in text outline hack.
- if (!hasMarkup &&
- !textOutline &&
- !ellipsis &&
- !width &&
- (textStr.indexOf(' ') === -1 ||
- (noWrap && !regexMatchBreaks.test(textStr)))) {
- textNode.appendChild(doc.createTextNode(unescapeEntities(textStr)));
- // Complex strings, add more logic
- }
- else {
- if (tempParent) {
- // attach it to the DOM to read offset width
- tempParent.appendChild(textNode);
- }
- if (hasMarkup) {
- lines = renderer.styledMode ? (textStr
- .replace(/<(b|strong)>/g, '<span class="highcharts-strong">')
- .replace(/<(i|em)>/g, '<span class="highcharts-emphasized">')) : (textStr
- .replace(/<(b|strong)>/g, '<span style="font-weight:bold">')
- .replace(/<(i|em)>/g, '<span style="font-style:italic">'));
- lines = lines
- .replace(/<a/g, '<span')
- .replace(/<\/(b|strong|i|em|a)>/g, '</span>')
- .split(regexMatchBreaks);
- }
- else {
- lines = [textStr];
- }
- // Trim empty lines (#5261)
- lines = lines.filter(function (line) {
- return line !== '';
- });
- // build the lines
- lines.forEach(function (line, lineNo) {
- var spans,
- spanNo = 0,
- lineLength = 0;
- line = line
- // Trim to prevent useless/costly process on the spaces
- // (#5258)
- .replace(/^\s+|\s+$/g, '')
- .replace(/<span/g, '|||<span')
- .replace(/<\/span>/g, '</span>|||');
- spans = line.split('|||');
- spans.forEach(function buildTextSpans(span) {
- if (span !== '' || spans.length === 1) {
- var attributes = {},
- tspan = doc.createElementNS(renderer.SVG_NS, 'tspan'),
- a,
- classAttribute,
- styleAttribute, // #390
- hrefAttribute;
- classAttribute = parseAttribute(span, 'class');
- if (classAttribute) {
- attr(tspan, 'class', classAttribute);
- }
- styleAttribute = parseAttribute(span, 'style');
- if (styleAttribute) {
- styleAttribute = styleAttribute.replace(/(;| |^)color([ :])/, '$1fill$2');
- attr(tspan, 'style', styleAttribute);
- }
- // For anchors, wrap the tspan in an <a> tag and apply
- // the href attribute as is (#13559). Not for export
- // (#1529)
- hrefAttribute = parseAttribute(span, 'href');
- if (hrefAttribute && !forExport) {
- if (
- // Stop JavaScript links, vulnerable to XSS
- hrefAttribute.split(':')[0].toLowerCase()
- .indexOf('javascript') === -1) {
- a = doc.createElementNS(renderer.SVG_NS, 'a');
- attr(a, 'href', hrefAttribute);
- attr(tspan, 'class', 'highcharts-anchor');
- a.appendChild(tspan);
- if (!renderer.styledMode) {
- css(tspan, { cursor: 'pointer' });
- }
- }
- }
- // Strip away unsupported HTML tags (#7126)
- span = unescapeEntities(span.replace(/<[a-zA-Z\/](.|\n)*?>/g, '') || ' ');
- // Nested tags aren't supported, and cause crash in
- // Safari (#1596)
- if (span !== ' ') {
- // add the text node
- tspan.appendChild(doc.createTextNode(span));
- // First span in a line, align it to the left
- if (!spanNo) {
- if (lineNo && parentX !== null) {
- attributes.x = parentX;
- }
- }
- else {
- attributes.dx = 0; // #16
- }
- // add attributes
- attr(tspan, attributes);
- // Append it
- textNode.appendChild(a || tspan);
- // first span on subsequent line, add the line
- // height
- if (!spanNo && isSubsequentLine) {
- // allow getting the right offset height in
- // exporting in IE
- if (!svg && forExport) {
- css(tspan, { display: 'block' });
- }
- // Set the line height based on the font size of
- // either the text element or the tspan element
- attr(tspan, 'dy', getLineHeight(tspan));
- }
- // Check width and apply soft breaks or ellipsis
- if (width) {
- var words = span.replace(/([^\^])-/g, '$1- ').split(' '), // #1273
- hasWhiteSpace = !noWrap && (spans.length > 1 ||
- lineNo ||
- words.length > 1),
- wrapLineNo = 0,
- dy = getLineHeight(tspan);
- if (ellipsis) {
- truncated = renderer.truncate(wrapper, tspan, span, void 0, 0,
- // Target width
- Math.max(0,
- // Substract the font face to make
- // room for the ellipsis itself
- width - parseInt(fontSize || 12, 10)),
- // Build the text to test for
- function (text, currentIndex) {
- return text.substring(0, currentIndex) + '\u2026';
- });
- }
- else if (hasWhiteSpace) {
- while (words.length) {
- // For subsequent lines, create tspans
- // with the same style attributes as the
- // parent text node.
- if (words.length &&
- !noWrap &&
- wrapLineNo > 0) {
- tspan = doc.createElementNS(SVG_NS, 'tspan');
- attr(tspan, {
- dy: dy,
- x: parentX
- });
- if (styleAttribute) { // #390
- attr(tspan, 'style', styleAttribute);
- }
- // Start by appending the full
- // remaining text
- tspan.appendChild(doc.createTextNode(words.join(' ')
- .replace(/- /g, '-')));
- textNode.appendChild(tspan);
- }
- // For each line, truncate the remaining
- // words into the line length.
- renderer.truncate(wrapper, tspan, null, words, wrapLineNo === 0 ? lineLength : 0, width,
- // Build the text to test for
- function (text, currentIndex) {
- return words
- .slice(0, currentIndex)
- .join(' ')
- .replace(/- /g, '-');
- });
- lineLength = wrapper.actualWidth;
- wrapLineNo++;
- }
- }
- }
- spanNo++;
- }
- }
- });
- // To avoid beginning lines that doesn't add to the textNode
- // (#6144)
- isSubsequentLine = (isSubsequentLine ||
- textNode.childNodes.length);
- });
- if (ellipsis && truncated) {
- wrapper.attr('title', unescapeEntities(wrapper.textStr || '', ['<', '>']) // #7179
- );
- }
- if (tempParent) {
- tempParent.removeChild(textNode);
- }
- // Apply the text outline
- if (isString(textOutline) && wrapper.applyTextOutline) {
- wrapper.applyTextOutline(textOutline);
- }
- }
- };
- /**
- * Returns white for dark colors and black for bright colors.
- *
- * @function Highcharts.SVGRenderer#getContrast
- *
- * @param {Highcharts.ColorString} rgba
- * The color to get the contrast for.
- *
- * @return {Highcharts.ColorString}
- * The contrast color, either `#000000` or `#FFFFFF`.
- */
- SVGRenderer.prototype.getContrast = function (rgba) {
- rgba = Color.parse(rgba).rgba;
- // The threshold may be discussed. Here's a proposal for adding
- // different weight to the color channels (#6216)
- rgba[0] *= 1; // red
- rgba[1] *= 1.2; // green
- rgba[2] *= 0.5; // blue
- return rgba[0] + rgba[1] + rgba[2] >
- 1.8 * 255 ?
- '#000000' :
- '#FFFFFF';
- };
- /**
- * Create a button with preset states.
- *
- * @function Highcharts.SVGRenderer#button
- *
- * @param {string} text
- * The text or HTML to draw.
- *
- * @param {number} x
- * The x position of the button's left side.
- *
- * @param {number} y
- * The y position of the button's top side.
- *
- * @param {Highcharts.EventCallbackFunction<Highcharts.SVGElement>} callback
- * The function to execute on button click or touch.
- *
- * @param {Highcharts.SVGAttributes} [normalState]
- * SVG attributes for the normal state.
- *
- * @param {Highcharts.SVGAttributes} [hoverState]
- * SVG attributes for the hover state.
- *
- * @param {Highcharts.SVGAttributes} [pressedState]
- * SVG attributes for the pressed state.
- *
- * @param {Highcharts.SVGAttributes} [disabledState]
- * SVG attributes for the disabled state.
- *
- * @param {Highcharts.SymbolKeyValue} [shape=rect]
- * The shape type.
- *
- * @param {boolean} [useHTML=false]
- * Wether to use HTML to render the label.
- *
- * @return {Highcharts.SVGElement}
- * The button element.
- */
- SVGRenderer.prototype.button = function (text, x, y, callback, normalState, hoverState, pressedState, disabledState, shape, useHTML) {
- var label = this.label(text,
- x,
- y,
- shape,
- void 0,
- void 0,
- useHTML,
- void 0, 'button'),
- curState = 0,
- styledMode = this.styledMode,
- // Make a copy of normalState (#13798)
- // (reference to options.rangeSelector.buttonTheme)
- normalState = normalState ? merge(normalState) : normalState,
- userNormalStyle = normalState && normalState.style || {};
- // Remove stylable attributes
- if (normalState && normalState.style) {
- delete normalState.style;
- }
- // Default, non-stylable attributes
- label.attr(merge({ padding: 8, r: 2 }, normalState));
- if (!styledMode) {
- // Presentational
- var normalStyle,
- hoverStyle,
- pressedStyle,
- disabledStyle;
- // Normal state - prepare the attributes
- normalState = merge({
- fill: '#f7f7f7',
- stroke: '#cccccc',
- 'stroke-width': 1,
- style: {
- color: '#333333',
- cursor: 'pointer',
- fontWeight: 'normal'
- }
- }, {
- style: userNormalStyle
- }, normalState);
- normalStyle = normalState.style;
- delete normalState.style;
- // Hover state
- hoverState = merge(normalState, {
- fill: '#e6e6e6'
- }, hoverState);
- hoverStyle = hoverState.style;
- delete hoverState.style;
- // Pressed state
- pressedState = merge(normalState, {
- fill: '#e6ebf5',
- style: {
- color: '#000000',
- fontWeight: 'bold'
- }
- }, pressedState);
- pressedStyle = pressedState.style;
- delete pressedState.style;
- // Disabled state
- disabledState = merge(normalState, {
- style: {
- color: '#cccccc'
- }
- }, disabledState);
- disabledStyle = disabledState.style;
- delete disabledState.style;
- }
- // Add the events. IE9 and IE10 need mouseover and mouseout to funciton
- // (#667).
- addEvent(label.element, isMS ? 'mouseover' : 'mouseenter', function () {
- if (curState !== 3) {
- label.setState(1);
- }
- });
- addEvent(label.element, isMS ? 'mouseout' : 'mouseleave', function () {
- if (curState !== 3) {
- label.setState(curState);
- }
- });
- label.setState = function (state) {
- // Hover state is temporary, don't record it
- if (state !== 1) {
- label.state = curState = state;
- }
- // Update visuals
- label
- .removeClass(/highcharts-button-(normal|hover|pressed|disabled)/)
- .addClass('highcharts-button-' +
- ['normal', 'hover', 'pressed', 'disabled'][state || 0]);
- if (!styledMode) {
- label
- .attr([
- normalState,
- hoverState,
- pressedState,
- disabledState
- ][state || 0])
- .css([
- normalStyle,
- hoverStyle,
- pressedStyle,
- disabledStyle
- ][state || 0]);
- }
- };
- // Presentational attributes
- if (!styledMode) {
- label
- .attr(normalState)
- .css(extend({ cursor: 'default' }, normalStyle));
- }
- return label
- .on('click', function (e) {
- if (curState !== 3) {
- callback.call(label, e);
- }
- });
- };
- /**
- * Make a straight line crisper by not spilling out to neighbour pixels.
- *
- * @function Highcharts.SVGRenderer#crispLine
- *
- * @param {Highcharts.SVGPathArray} points
- * The original points on the format `[['M', 0, 0], ['L', 100, 0]]`.
- *
- * @param {number} width
- * The width of the line.
- *
- * @param {string} roundingFunction
- * The rounding function name on the `Math` object, can be one of
- * `round`, `floor` or `ceil`.
- *
- * @return {Highcharts.SVGPathArray}
- * The original points array, but modified to render crisply.
- */
- SVGRenderer.prototype.crispLine = function (points, width, roundingFunction) {
- if (roundingFunction === void 0) { roundingFunction = 'round'; }
- var start = points[0];
- var end = points[1];
- // Normalize to a crisp line
- if (start[1] === end[1]) {
- // Substract due to #1129. Now bottom and left axis gridlines behave
- // the same.
- start[1] = end[1] =
- Math[roundingFunction](start[1]) - (width % 2 / 2);
- }
- if (start[2] === end[2]) {
- start[2] = end[2] =
- Math[roundingFunction](start[2]) + (width % 2 / 2);
- }
- return points;
- };
- /**
- * Draw a path, wraps the SVG `path` element.
- *
- * @sample highcharts/members/renderer-path-on-chart/
- * Draw a path in a chart
- * @sample highcharts/members/renderer-path/
- * Draw a path independent from a chart
- *
- * @example
- * var path = renderer.path(['M', 10, 10, 'L', 30, 30, 'z'])
- * .attr({ stroke: '#ff00ff' })
- * .add();
- *
- * @function Highcharts.SVGRenderer#path
- *
- * @param {Highcharts.SVGPathArray} [path]
- * An SVG path definition in array form.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- *
- */ /**
- * Draw a path, wraps the SVG `path` element.
- *
- * @function Highcharts.SVGRenderer#path
- *
- * @param {Highcharts.SVGAttributes} [attribs]
- * The initial attributes.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.path = function (path) {
- var attribs = (this.styledMode ? {} : {
- fill: 'none'
- });
- if (isArray(path)) {
- attribs.d = path;
- }
- else if (isObject(path)) { // attributes
- extend(attribs, path);
- }
- return this.createElement('path').attr(attribs);
- };
- /**
- * Draw a circle, wraps the SVG `circle` element.
- *
- * @sample highcharts/members/renderer-circle/
- * Drawing a circle
- *
- * @function Highcharts.SVGRenderer#circle
- *
- * @param {number} [x]
- * The center x position.
- *
- * @param {number} [y]
- * The center y position.
- *
- * @param {number} [r]
- * The radius.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */ /**
- * Draw a circle, wraps the SVG `circle` element.
- *
- * @function Highcharts.SVGRenderer#circle
- *
- * @param {Highcharts.SVGAttributes} [attribs]
- * The initial attributes.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.circle = function (x, y, r) {
- var attribs = (isObject(x) ?
- x :
- typeof x === 'undefined' ? {} : { x: x, y: y, r: r }), wrapper = this.createElement('circle');
- // Setting x or y translates to cx and cy
- wrapper.xSetter = wrapper.ySetter = function (value, key, element) {
- element.setAttribute('c' + key, value);
- };
- return wrapper.attr(attribs);
- };
- /**
- * Draw and return an arc.
- *
- * @sample highcharts/members/renderer-arc/
- * Drawing an arc
- *
- * @function Highcharts.SVGRenderer#arc
- *
- * @param {number} [x=0]
- * Center X position.
- *
- * @param {number} [y=0]
- * Center Y position.
- *
- * @param {number} [r=0]
- * The outer radius' of the arc.
- *
- * @param {number} [innerR=0]
- * Inner radius like used in donut charts.
- *
- * @param {number} [start=0]
- * The starting angle of the arc in radians, where 0 is to the right and
- * `-Math.PI/2` is up.
- *
- * @param {number} [end=0]
- * The ending angle of the arc in radians, where 0 is to the right and
- * `-Math.PI/2` is up.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */ /**
- * Draw and return an arc. Overloaded function that takes arguments object.
- *
- * @function Highcharts.SVGRenderer#arc
- *
- * @param {Highcharts.SVGAttributes} attribs
- * Initial SVG attributes.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.arc = function (x, y, r, innerR, start, end) {
- var arc,
- options;
- if (isObject(x)) {
- options = x;
- y = options.y;
- r = options.r;
- innerR = options.innerR;
- start = options.start;
- end = options.end;
- x = options.x;
- }
- else {
- options = {
- innerR: innerR,
- start: start,
- end: end
- };
- }
- // Arcs are defined as symbols for the ability to set
- // attributes in attr and animate
- arc = this.symbol('arc', x, y, r, r, options);
- arc.r = r; // #959
- return arc;
- };
- /**
- * Draw and return a rectangle.
- *
- * @function Highcharts.SVGRenderer#rect
- *
- * @param {number} [x]
- * Left position.
- *
- * @param {number} [y]
- * Top position.
- *
- * @param {number} [width]
- * Width of the rectangle.
- *
- * @param {number} [height]
- * Height of the rectangle.
- *
- * @param {number} [r]
- * Border corner radius.
- *
- * @param {number} [strokeWidth]
- * A stroke width can be supplied to allow crisp drawing.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */ /**
- * Draw and return a rectangle.
- *
- * @sample highcharts/members/renderer-rect-on-chart/
- * Draw a rectangle in a chart
- * @sample highcharts/members/renderer-rect/
- * Draw a rectangle independent from a chart
- *
- * @function Highcharts.SVGRenderer#rect
- *
- * @param {Highcharts.SVGAttributes} [attributes]
- * General SVG attributes for the rectangle.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.rect = function (x, y, width, height, r, strokeWidth) {
- r = isObject(x) ? x.r : r;
- var wrapper = this.createElement('rect'),
- attribs = isObject(x) ?
- x :
- typeof x === 'undefined' ?
- {} :
- {
- x: x,
- y: y,
- width: Math.max(width, 0),
- height: Math.max(height, 0)
- };
- if (!this.styledMode) {
- if (typeof strokeWidth !== 'undefined') {
- attribs.strokeWidth = strokeWidth;
- attribs = wrapper.crisp(attribs);
- }
- attribs.fill = 'none';
- }
- if (r) {
- attribs.r = r;
- }
- wrapper.rSetter = function (value, key, element) {
- wrapper.r = value;
- attr(element, {
- rx: value,
- ry: value
- });
- };
- wrapper.rGetter = function () {
- return wrapper.r;
- };
- return wrapper.attr(attribs);
- };
- /**
- * Resize the {@link SVGRenderer#box} and re-align all aligned child
- * elements.
- *
- * @sample highcharts/members/renderer-g/
- * Show and hide grouped objects
- *
- * @function Highcharts.SVGRenderer#setSize
- *
- * @param {number} width
- * The new pixel width.
- *
- * @param {number} height
- * The new pixel height.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animate=true]
- * Whether and how to animate.
- */
- SVGRenderer.prototype.setSize = function (width, height, animate) {
- var renderer = this,
- alignedObjects = renderer.alignedObjects,
- i = alignedObjects.length;
- renderer.width = width;
- renderer.height = height;
- renderer.boxWrapper.animate({
- width: width,
- height: height
- }, {
- step: function () {
- this.attr({
- viewBox: '0 0 ' + this.attr('width') + ' ' +
- this.attr('height')
- });
- },
- duration: pick(animate, true) ? void 0 : 0
- });
- while (i--) {
- alignedObjects[i].align();
- }
- };
- /**
- * Create and return an svg group element. Child
- * {@link Highcharts.SVGElement} objects are added to the group by using the
- * group as the first parameter in {@link Highcharts.SVGElement#add|add()}.
- *
- * @function Highcharts.SVGRenderer#g
- *
- * @param {string} [name]
- * The group will be given a class name of `highcharts-{name}`. This
- * can be used for styling and scripting.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.g = function (name) {
- var elem = this.createElement('g');
- return name ?
- elem.attr({ 'class': 'highcharts-' + name }) :
- elem;
- };
- /**
- * Display an image.
- *
- * @sample highcharts/members/renderer-image-on-chart/
- * Add an image in a chart
- * @sample highcharts/members/renderer-image/
- * Add an image independent of a chart
- *
- * @function Highcharts.SVGRenderer#image
- *
- * @param {string} src
- * The image source.
- *
- * @param {number} [x]
- * The X position.
- *
- * @param {number} [y]
- * The Y position.
- *
- * @param {number} [width]
- * The image width. If omitted, it defaults to the image file width.
- *
- * @param {number} [height]
- * The image height. If omitted it defaults to the image file
- * height.
- *
- * @param {Function} [onload]
- * Event handler for image load.
- *
- * @return {Highcharts.SVGElement}
- * The generated wrapper element.
- */
- SVGRenderer.prototype.image = function (src, x, y, width, height, onload) {
- var attribs = { preserveAspectRatio: 'none' }, elemWrapper, dummy, setSVGImageSource = function (el, src) {
- // Set the href in the xlink namespace
- if (el.setAttributeNS) {
- el.setAttributeNS('http://www.w3.org/1999/xlink', 'href', src);
- }
- else {
- // could be exporting in IE
- // using href throws "not supported" in ie7 and under,
- // requries regex shim to fix later
- el.setAttribute('hc-svg-href', src);
- }
- }, onDummyLoad = function (e) {
- setSVGImageSource(elemWrapper.element, src);
- onload.call(elemWrapper, e);
- };
- // optional properties
- if (arguments.length > 1) {
- extend(attribs, {
- x: x,
- y: y,
- width: width,
- height: height
- });
- }
- elemWrapper = this.createElement('image').attr(attribs);
- // Add load event if supplied
- if (onload) {
- // We have to use a dummy HTML image since IE support for SVG image
- // load events is very buggy. First set a transparent src, wait for
- // dummy to load, and then add the real src to the SVG image.
- setSVGImageSource(elemWrapper.element, 'data:image/gif;base64,R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==' /* eslint-disable-line */);
- dummy = new win.Image();
- addEvent(dummy, 'load', onDummyLoad);
- dummy.src = src;
- if (dummy.complete) {
- onDummyLoad({});
- }
- }
- else {
- setSVGImageSource(elemWrapper.element, src);
- }
- return elemWrapper;
- };
- /**
- * Draw a symbol out of pre-defined shape paths from
- * {@link SVGRenderer#symbols}.
- * It is used in Highcharts for point makers, which cake a `symbol` option,
- * and label and button backgrounds like in the tooltip and stock flags.
- *
- * @function Highcharts.SVGRenderer#symbol
- *
- * @param {string} symbol
- * The symbol name.
- *
- * @param {number} [x]
- * The X coordinate for the top left position.
- *
- * @param {number} [y]
- * The Y coordinate for the top left position.
- *
- * @param {number} [width]
- * The pixel width.
- *
- * @param {number} [height]
- * The pixel height.
- *
- * @param {Highcharts.SymbolOptionsObject} [options]
- * Additional options, depending on the actual symbol drawn.
- *
- * @return {Highcharts.SVGElement}
- */
- SVGRenderer.prototype.symbol = function (symbol, x, y, width, height, options) {
- var ren = this,
- obj,
- imageRegex = /^url\((.*?)\)$/,
- isImage = imageRegex.test(symbol),
- sym = (!isImage && (this.symbols[symbol] ? symbol : 'circle')),
- // get the symbol definition function
- symbolFn = (sym && this.symbols[sym]),
- path,
- imageSrc,
- centerImage;
- if (symbolFn) {
- // Check if there's a path defined for this symbol
- if (typeof x === 'number') {
- path = symbolFn.call(this.symbols, Math.round(x || 0), Math.round(y || 0), width || 0, height || 0, options);
- }
- obj = this.path(path);
- if (!ren.styledMode) {
- obj.attr('fill', 'none');
- }
- // expando properties for use in animate and attr
- extend(obj, {
- symbolName: sym,
- x: x,
- y: y,
- width: width,
- height: height
- });
- if (options) {
- extend(obj, options);
- }
- // Image symbols
- }
- else if (isImage) {
- imageSrc = symbol.match(imageRegex)[1];
- // Create the image synchronously, add attribs async
- obj = this.image(imageSrc);
- // The image width is not always the same as the symbol width. The
- // image may be centered within the symbol, as is the case when
- // image shapes are used as label backgrounds, for example in flags.
- obj.imgwidth = pick(symbolSizes[imageSrc] && symbolSizes[imageSrc].width, options && options.width);
- obj.imgheight = pick(symbolSizes[imageSrc] && symbolSizes[imageSrc].height, options && options.height);
- /**
- * Set the size and position
- */
- centerImage = function () {
- obj.attr({
- width: obj.width,
- height: obj.height
- });
- };
- /**
- * Width and height setters that take both the image's physical size
- * and the label size into consideration, and translates the image
- * to center within the label.
- */
- ['width', 'height'].forEach(function (key) {
- obj[key + 'Setter'] = function (value, key) {
- var attribs = {}, imgSize = this['img' + key], trans = key === 'width' ? 'translateX' : 'translateY';
- this[key] = value;
- if (defined(imgSize)) {
- // Scale and center the image within its container.
- // The name `backgroundSize` is taken from the CSS spec,
- // but the value `within` is made up. Other possible
- // values in the spec, `cover` and `contain`, can be
- // implemented if needed.
- if (options &&
- options.backgroundSize === 'within' &&
- this.width &&
- this.height) {
- imgSize = Math.round(imgSize * Math.min(this.width / this.imgwidth, this.height / this.imgheight));
- }
- if (this.element) {
- this.element.setAttribute(key, imgSize);
- }
- if (!this.alignByTranslate) {
- attribs[trans] = ((this[key] || 0) - imgSize) / 2;
- this.attr(attribs);
- }
- }
- };
- });
- if (defined(x)) {
- obj.attr({
- x: x,
- y: y
- });
- }
- obj.isImg = true;
- if (defined(obj.imgwidth) && defined(obj.imgheight)) {
- centerImage();
- }
- else {
- // Initialize image to be 0 size so export will still function
- // if there's no cached sizes.
- obj.attr({ width: 0, height: 0 });
- // Create a dummy JavaScript image to get the width and height.
- createElement('img', {
- onload: function () {
- var chart = charts[ren.chartIndex];
- // Special case for SVGs on IE11, the width is not
- // accessible until the image is part of the DOM
- // (#2854).
- if (this.width === 0) {
- css(this, {
- position: 'absolute',
- top: '-999em'
- });
- doc.body.appendChild(this);
- }
- // Center the image
- symbolSizes[imageSrc] = {
- width: this.width,
- height: this.height
- };
- obj.imgwidth = this.width;
- obj.imgheight = this.height;
- if (obj.element) {
- centerImage();
- }
- // Clean up after #2854 workaround.
- if (this.parentNode) {
- this.parentNode.removeChild(this);
- }
- // Fire the load event when all external images are
- // loaded
- ren.imgCount--;
- if (!ren.imgCount && chart && !chart.hasLoaded) {
- chart.onload();
- }
- },
- src: imageSrc
- });
- this.imgCount++;
- }
- }
- return obj;
- };
- /**
- * Define a clipping rectangle. The clipping rectangle is later applied
- * to {@link SVGElement} objects through the {@link SVGElement#clip}
- * function.
- *
- * @example
- * var circle = renderer.circle(100, 100, 100)
- * .attr({ fill: 'red' })
- * .add();
- * var clipRect = renderer.clipRect(100, 100, 100, 100);
- *
- * // Leave only the lower right quarter visible
- * circle.clip(clipRect);
- *
- * @function Highcharts.SVGRenderer#clipRect
- *
- * @param {number} [x]
- *
- * @param {number} [y]
- *
- * @param {number} [width]
- *
- * @param {number} [height]
- *
- * @return {Highcharts.ClipRectElement}
- * A clipping rectangle.
- */
- SVGRenderer.prototype.clipRect = function (x, y, width, height) {
- var wrapper,
- // Add a hyphen at the end to avoid confusion in testing indexes
- // -1 and -10, -11 etc (#6550)
- id = uniqueKey() + '-', clipPath = this.createElement('clipPath').attr({
- id: id
- }).add(this.defs);
- wrapper = this.rect(x, y, width, height, 0).add(clipPath);
- wrapper.id = id;
- wrapper.clipPath = clipPath;
- wrapper.count = 0;
- return wrapper;
- };
- /**
- * Draw text. The text can contain a subset of HTML, like spans and anchors
- * and some basic text styling of these. For more advanced features like
- * border and background, use {@link Highcharts.SVGRenderer#label} instead.
- * To update the text after render, run `text.attr({ text: 'New text' })`.
- *
- * @sample highcharts/members/renderer-text-on-chart/
- * Annotate the chart freely
- * @sample highcharts/members/renderer-on-chart/
- * Annotate with a border and in response to the data
- * @sample highcharts/members/renderer-text/
- * Formatted text
- *
- * @function Highcharts.SVGRenderer#text
- *
- * @param {string} [str]
- * The text of (subset) HTML to draw.
- *
- * @param {number} [x]
- * The x position of the text's lower left corner.
- *
- * @param {number} [y]
- * The y position of the text's lower left corner.
- *
- * @param {boolean} [useHTML=false]
- * Use HTML to render the text.
- *
- * @return {Highcharts.SVGElement}
- * The text object.
- */
- SVGRenderer.prototype.text = function (str, x, y, useHTML) {
- // declare variables
- var renderer = this,
- wrapper,
- attribs = {};
- if (useHTML && (renderer.allowHTML || !renderer.forExport)) {
- return renderer.html(str, x, y);
- }
- attribs.x = Math.round(x || 0); // X always needed for line-wrap logic
- if (y) {
- attribs.y = Math.round(y);
- }
- if (defined(str)) {
- attribs.text = str;
- }
- wrapper = renderer.createElement('text')
- .attr(attribs);
- if (!useHTML) {
- wrapper.xSetter = function (value, key, element) {
- var tspans = element.getElementsByTagName('tspan'),
- tspan,
- parentVal = element.getAttribute(key),
- i;
- for (i = 0; i < tspans.length; i++) {
- tspan = tspans[i];
- // If the x values are equal, the tspan represents a
- // linebreak
- if (tspan.getAttribute(key) === parentVal) {
- tspan.setAttribute(key, value);
- }
- }
- element.setAttribute(key, value);
- };
- }
- return wrapper;
- };
- /**
- * Utility to return the baseline offset and total line height from the font
- * size.
- *
- * @function Highcharts.SVGRenderer#fontMetrics
- *
- * @param {number|string} [fontSize]
- * The current font size to inspect. If not given, the font size
- * will be found from the DOM element.
- *
- * @param {Highcharts.SVGElement|Highcharts.SVGDOMElement} [elem]
- * The element to inspect for a current font size.
- *
- * @return {Highcharts.FontMetricsObject}
- * The font metrics.
- */
- SVGRenderer.prototype.fontMetrics = function (fontSize, elem) {
- var lineHeight,
- baseline;
- if ((this.styledMode || !/px/.test(fontSize)) &&
- win.getComputedStyle // old IE doesn't support it
- ) {
- fontSize = elem && SVGElement.prototype.getStyle.call(elem, 'font-size');
- }
- else {
- fontSize = fontSize ||
- // When the elem is a DOM element (#5932)
- (elem && elem.style && elem.style.fontSize) ||
- // Fall back on the renderer style default
- (this.style && this.style.fontSize);
- }
- // Handle different units
- if (/px/.test(fontSize)) {
- fontSize = pInt(fontSize);
- }
- else {
- fontSize = 12;
- }
- // Empirical values found by comparing font size and bounding box
- // height. Applies to the default font family.
- // https://jsfiddle.net/highcharts/7xvn7/
- lineHeight = fontSize < 24 ? fontSize + 3 : Math.round(fontSize * 1.2);
- baseline = Math.round(lineHeight * 0.8);
- return {
- h: lineHeight,
- b: baseline,
- f: fontSize
- };
- };
- /**
- * Correct X and Y positioning of a label for rotation (#1764).
- *
- * @private
- * @function Highcharts.SVGRenderer#rotCorr
- *
- * @param {number} baseline
- *
- * @param {number} rotation
- *
- * @param {boolean} [alterY]
- *
- * @param {Highcharts.PositionObject}
- */
- SVGRenderer.prototype.rotCorr = function (baseline, rotation, alterY) {
- var y = baseline;
- if (rotation && alterY) {
- y = Math.max(y * Math.cos(rotation * deg2rad), 4);
- }
- return {
- x: (-baseline / 3) * Math.sin(rotation * deg2rad),
- y: y
- };
- };
- /**
- * Compatibility function to convert the legacy one-dimensional path array
- * into an array of segments.
- *
- * It is used in maps to parse the `path` option, and in SVGRenderer.dSetter
- * to support legacy paths from demos.
- *
- * @private
- * @function Highcharts.SVGRenderer#pathToSegments
- */
- SVGRenderer.prototype.pathToSegments = function (path) {
- var ret = [];
- var segment = [];
- var commandLength = {
- A: 8,
- C: 7,
- H: 2,
- L: 3,
- M: 3,
- Q: 5,
- S: 5,
- T: 3,
- V: 2
- };
- // Short, non-typesafe parsing of the one-dimensional array. It splits
- // the path on any string. This is not type checked against the tuple
- // types, but is shorter, and doesn't require specific checks for any
- // command type in SVG.
- for (var i = 0; i < path.length; i++) {
- // Command skipped, repeat previous or insert L/l for M/m
- if (isString(segment[0]) &&
- isNumber(path[i]) &&
- segment.length === commandLength[(segment[0].toUpperCase())]) {
- path.splice(i, 0, segment[0].replace('M', 'L').replace('m', 'l'));
- }
- // Split on string
- if (typeof path[i] === 'string') {
- if (segment.length) {
- ret.push(segment.slice(0));
- }
- segment.length = 0;
- }
- segment.push(path[i]);
- }
- ret.push(segment.slice(0));
- return ret;
- /*
- // Fully type-safe version where each tuple type is checked. The
- // downside is filesize and a lack of flexibility for unsupported
- // commands
- const ret: SVGPath = [],
- commands = {
- A: 7,
- C: 6,
- H: 1,
- L: 2,
- M: 2,
- Q: 4,
- S: 4,
- T: 2,
- V: 1,
- Z: 0
- };
- let i = 0,
- lastI = 0,
- lastCommand;
- while (i < path.length) {
- const item = path[i];
- let command;
- if (typeof item === 'string') {
- command = item;
- i += 1;
- } else {
- command = lastCommand || 'M';
- }
- // Upper case
- const commandUC = command.toUpperCase();
- if (commandUC in commands) {
- // No numeric parameters
- if (command === 'Z' || command === 'z') {
- ret.push([command]);
- // One numeric parameter
- } else {
- const val0 = path[i];
- if (typeof val0 === 'number') {
- // Horizontal line to
- if (command === 'H' || command === 'h') {
- ret.push([command, val0]);
- i += 1;
- // Vertical line to
- } else if (command === 'V' || command === 'v') {
- ret.push([command, val0]);
- i += 1;
- // Two numeric parameters
- } else {
- const val1 = path[i + 1];
- if (typeof val1 === 'number') {
- // lineTo
- if (command === 'L' || command === 'l') {
- ret.push([command, val0, val1]);
- i += 2;
- // moveTo
- } else if (command === 'M' || command === 'm') {
- ret.push([command, val0, val1]);
- i += 2;
- // Smooth quadratic bezier
- } else if (command === 'T' || command === 't') {
- ret.push([command, val0, val1]);
- i += 2;
- // Four numeric parameters
- } else {
- const val2 = path[i + 2],
- val3 = path[i + 3];
- if (
- typeof val2 === 'number' &&
- typeof val3 === 'number'
- ) {
- // Quadratic bezier to
- if (
- command === 'Q' ||
- command === 'q'
- ) {
- ret.push([
- command,
- val0,
- val1,
- val2,
- val3
- ]);
- i += 4;
- // Smooth cubic bezier to
- } else if (
- command === 'S' ||
- command === 's'
- ) {
- ret.push([
- command,
- val0,
- val1,
- val2,
- val3
- ]);
- i += 4;
- // Six numeric parameters
- } else {
- const val4 = path[i + 4],
- val5 = path[i + 5];
- if (
- typeof val4 === 'number' &&
- typeof val5 === 'number'
- ) {
- // Curve to
- if (
- command === 'C' ||
- command === 'c'
- ) {
- ret.push([
- command,
- val0,
- val1,
- val2,
- val3,
- val4,
- val5
- ]);
- i += 6;
- // Seven numeric parameters
- } else {
- const val6 = path[i + 6];
- // Arc to
- if (
- typeof val6 ===
- 'number' &&
- (
- command === 'A' ||
- command === 'a'
- )
- ) {
- ret.push([
- command,
- val0,
- val1,
- val2,
- val3,
- val4,
- val5,
- val6
- ]);
- i += 7;
- }
- }
- }
- }
- }
- }
- }
- }
- }
- }
- }
- // An unmarked command following a moveTo is a lineTo
- lastCommand = command === 'M' ? 'L' : command;
- if (i === lastI) {
- break;
- }
- lastI = i;
- }
- return ret;
- */
- };
- /**
- * Draw a label, which is an extended text element with support for border
- * and background. Highcharts creates a `g` element with a text and a `path`
- * or `rect` inside, to make it behave somewhat like a HTML div. Border and
- * background are set through `stroke`, `stroke-width` and `fill` attributes
- * using the {@link Highcharts.SVGElement#attr|attr} method. To update the
- * text after render, run `label.attr({ text: 'New text' })`.
- *
- * @sample highcharts/members/renderer-label-on-chart/
- * A label on the chart
- *
- * @function Highcharts.SVGRenderer#label
- *
- * @param {string} str
- * The initial text string or (subset) HTML to render.
- *
- * @param {number} x
- * The x position of the label's left side.
- *
- * @param {number} [y]
- * The y position of the label's top side or baseline, depending on
- * the `baseline` parameter.
- *
- * @param {string} [shape='rect']
- * The shape of the label's border/background, if any. Defaults to
- * `rect`. Other possible values are `callout` or other shapes
- * defined in {@link Highcharts.SVGRenderer#symbols}.
- *
- * @param {number} [anchorX]
- * In case the `shape` has a pointer, like a flag, this is the
- * coordinates it should be pinned to.
- *
- * @param {number} [anchorY]
- * In case the `shape` has a pointer, like a flag, this is the
- * coordinates it should be pinned to.
- *
- * @param {boolean} [useHTML=false]
- * Wether to use HTML to render the label.
- *
- * @param {boolean} [baseline=false]
- * Whether to position the label relative to the text baseline,
- * like {@link Highcharts.SVGRenderer#text|renderer.text}, or to the
- * upper border of the rectangle.
- *
- * @param {string} [className]
- * Class name for the group.
- *
- * @return {Highcharts.SVGElement}
- * The generated label.
- */
- SVGRenderer.prototype.label = function (str, x, y, shape, anchorX, anchorY, useHTML, baseline, className) {
- return new SVGLabel(this, str, x, y, shape, anchorX, anchorY, useHTML, baseline, className);
- };
- return SVGRenderer;
- }());
- /**
- * A pointer to the renderer's associated Element class. The VMLRenderer
- * will have a pointer to VMLElement here.
- *
- * @name Highcharts.SVGRenderer#Element
- * @type {Highcharts.SVGElement}
- */
- SVGRenderer.prototype.Element = SVGElement;
- /**
- * @private
- */
- SVGRenderer.prototype.SVG_NS = SVG_NS;
- /**
- * Dummy function for plugins, called every time the renderer is updated.
- * Prior to Highcharts 5, this was used for the canvg renderer.
- *
- * @deprecated
- * @function Highcharts.SVGRenderer#draw
- */
- SVGRenderer.prototype.draw = noop;
- /**
- * A collection of characters mapped to HTML entities. When `useHTML` on an
- * element is true, these entities will be rendered correctly by HTML. In
- * the SVG pseudo-HTML, they need to be unescaped back to simple characters,
- * so for example `<` will render as `<`.
- *
- * @example
- * // Add support for unescaping quotes
- * Highcharts.SVGRenderer.prototype.escapes['"'] = '"';
- *
- * @name Highcharts.SVGRenderer#escapes
- * @type {Highcharts.Dictionary<string>}
- */
- SVGRenderer.prototype.escapes = {
- '&': '&',
- '<': '<',
- '>': '>',
- "'": ''',
- '"': '"'
- };
- /**
- * An extendable collection of functions for defining symbol paths.
- *
- * @name Highcharts.SVGRenderer#symbols
- * @type {Highcharts.SymbolDictionary}
- */
- SVGRenderer.prototype.symbols = {
- circle: function (x, y, w, h) {
- // Return a full arc
- return this.arc(x + w / 2, y + h / 2, w / 2, h / 2, {
- start: Math.PI * 0.5,
- end: Math.PI * 2.5,
- open: false
- });
- },
- square: function (x, y, w, h) {
- return [
- ['M', x, y],
- ['L', x + w, y],
- ['L', x + w, y + h],
- ['L', x, y + h],
- ['Z']
- ];
- },
- triangle: function (x, y, w, h) {
- return [
- ['M', x + w / 2, y],
- ['L', x + w, y + h],
- ['L', x, y + h],
- ['Z']
- ];
- },
- 'triangle-down': function (x, y, w, h) {
- return [
- ['M', x, y],
- ['L', x + w, y],
- ['L', x + w / 2, y + h],
- ['Z']
- ];
- },
- diamond: function (x, y, w, h) {
- return [
- ['M', x + w / 2, y],
- ['L', x + w, y + h / 2],
- ['L', x + w / 2, y + h],
- ['L', x, y + h / 2],
- ['Z']
- ];
- },
- arc: function (x, y, w, h, options) {
- var arc = [];
- if (options) {
- var start = options.start || 0,
- end = options.end || 0,
- rx = options.r || w,
- ry = options.r || h || w,
- proximity = 0.001,
- fullCircle = Math.abs(end - start - 2 * Math.PI) <
- proximity,
- // Substract a small number to prevent cos and sin of start and
- // end from becoming equal on 360 arcs (related: #1561)
- end = end - proximity,
- innerRadius = options.innerR,
- open = pick(options.open,
- fullCircle),
- cosStart = Math.cos(start),
- sinStart = Math.sin(start),
- cosEnd = Math.cos(end),
- sinEnd = Math.sin(end),
- // Proximity takes care of rounding errors around PI (#6971)
- longArc = pick(options.longArc,
- end - start - Math.PI < proximity ? 0 : 1);
- arc.push([
- 'M',
- x + rx * cosStart,
- y + ry * sinStart
- ], [
- 'A',
- rx,
- ry,
- 0,
- longArc,
- pick(options.clockwise, 1),
- x + rx * cosEnd,
- y + ry * sinEnd
- ]);
- if (defined(innerRadius)) {
- arc.push(open ?
- [
- 'M',
- x + innerRadius * cosEnd,
- y + innerRadius * sinEnd
- ] : [
- 'L',
- x + innerRadius * cosEnd,
- y + innerRadius * sinEnd
- ], [
- 'A',
- innerRadius,
- innerRadius,
- 0,
- longArc,
- // Clockwise - opposite to the outer arc clockwise
- defined(options.clockwise) ? 1 - options.clockwise : 0,
- x + innerRadius * cosStart,
- y + innerRadius * sinStart
- ]);
- }
- if (!open) {
- arc.push(['Z']);
- }
- }
- return arc;
- },
- /**
- * Callout shape used for default tooltips, also used for rounded
- * rectangles in VML
- */
- callout: function (x, y, w, h, options) {
- var arrowLength = 6,
- halfDistance = 6,
- r = Math.min((options && options.r) || 0,
- w,
- h),
- safeDistance = r + halfDistance,
- anchorX = options && options.anchorX || 0,
- anchorY = options && options.anchorY || 0,
- path;
- path = [
- ['M', x + r, y],
- ['L', x + w - r, y],
- ['C', x + w, y, x + w, y, x + w, y + r],
- ['L', x + w, y + h - r],
- ['C', x + w, y + h, x + w, y + h, x + w - r, y + h],
- ['L', x + r, y + h],
- ['C', x, y + h, x, y + h, x, y + h - r],
- ['L', x, y + r],
- ['C', x, y, x, y, x + r, y] // top-left corner
- ];
- // Anchor on right side
- if (anchorX && anchorX > w) {
- // Chevron
- if (anchorY > y + safeDistance &&
- anchorY < y + h - safeDistance) {
- path.splice(3, 1, ['L', x + w, anchorY - halfDistance], ['L', x + w + arrowLength, anchorY], ['L', x + w, anchorY + halfDistance], ['L', x + w, y + h - r]);
- // Simple connector
- }
- else {
- path.splice(3, 1, ['L', x + w, h / 2], ['L', anchorX, anchorY], ['L', x + w, h / 2], ['L', x + w, y + h - r]);
- }
- // Anchor on left side
- }
- else if (anchorX && anchorX < 0) {
- // Chevron
- if (anchorY > y + safeDistance &&
- anchorY < y + h - safeDistance) {
- path.splice(7, 1, ['L', x, anchorY + halfDistance], ['L', x - arrowLength, anchorY], ['L', x, anchorY - halfDistance], ['L', x, y + r]);
- // Simple connector
- }
- else {
- path.splice(7, 1, ['L', x, h / 2], ['L', anchorX, anchorY], ['L', x, h / 2], ['L', x, y + r]);
- }
- }
- else if ( // replace bottom
- anchorY &&
- anchorY > h &&
- anchorX > x + safeDistance &&
- anchorX < x + w - safeDistance) {
- path.splice(5, 1, ['L', anchorX + halfDistance, y + h], ['L', anchorX, y + h + arrowLength], ['L', anchorX - halfDistance, y + h], ['L', x + r, y + h]);
- }
- else if ( // replace top
- anchorY &&
- anchorY < 0 &&
- anchorX > x + safeDistance &&
- anchorX < x + w - safeDistance) {
- path.splice(1, 1, ['L', anchorX - halfDistance, y], ['L', anchorX, y - arrowLength], ['L', anchorX + halfDistance, y], ['L', w - r, y]);
- }
- return path;
- }
- };
- H.SVGRenderer = SVGRenderer;
- H.Renderer = H.SVGRenderer;
- return H.Renderer;
- });
- _registerModule(_modules, 'Core/Renderer/HTML/HTML.js', [_modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Renderer/SVG/SVGRenderer.js'], _modules['Core/Utilities.js']], function (H, SVGElement, SVGRenderer, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var attr = U.attr,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- extend = U.extend,
- pick = U.pick,
- pInt = U.pInt;
- var isFirefox = H.isFirefox,
- isMS = H.isMS,
- isWebKit = H.isWebKit,
- win = H.win;
- /* eslint-disable valid-jsdoc */
- // Extend SvgElement for useHTML option.
- extend(SVGElement.prototype, /** @lends SVGElement.prototype */ {
- /**
- * Apply CSS to HTML elements. This is used in text within SVG rendering and
- * by the VML renderer
- *
- * @private
- * @function Highcharts.SVGElement#htmlCss
- *
- * @param {Highcharts.CSSObject} styles
- *
- * @return {Highcharts.SVGElement}
- */
- htmlCss: function (styles) {
- var wrapper = this,
- element = wrapper.element,
- // When setting or unsetting the width style, we need to update
- // transform (#8809)
- isSettingWidth = (element.tagName === 'SPAN' &&
- styles &&
- 'width' in styles),
- textWidth = pick(isSettingWidth && styles.width,
- void 0),
- doTransform;
- if (isSettingWidth) {
- delete styles.width;
- wrapper.textWidth = textWidth;
- doTransform = true;
- }
- if (styles && styles.textOverflow === 'ellipsis') {
- styles.whiteSpace = 'nowrap';
- styles.overflow = 'hidden';
- }
- wrapper.styles = extend(wrapper.styles, styles);
- css(wrapper.element, styles);
- // Now that all styles are applied, to the transform
- if (doTransform) {
- wrapper.htmlUpdateTransform();
- }
- return wrapper;
- },
- /**
- * VML and useHTML method for calculating the bounding box based on offsets.
- *
- * @private
- * @function Highcharts.SVGElement#htmlGetBBox
- *
- * @param {boolean} refresh
- * Whether to force a fresh value from the DOM or to use the cached
- * value.
- *
- * @return {Highcharts.BBoxObject}
- * A hash containing values for x, y, width and height.
- */
- htmlGetBBox: function () {
- var wrapper = this,
- element = wrapper.element;
- return {
- x: element.offsetLeft,
- y: element.offsetTop,
- width: element.offsetWidth,
- height: element.offsetHeight
- };
- },
- /**
- * VML override private method to update elements based on internal
- * properties based on SVG transform.
- *
- * @private
- * @function Highcharts.SVGElement#htmlUpdateTransform
- * @return {void}
- */
- htmlUpdateTransform: function () {
- // aligning non added elements is expensive
- if (!this.added) {
- this.alignOnAdd = true;
- return;
- }
- var wrapper = this,
- renderer = wrapper.renderer,
- elem = wrapper.element,
- translateX = wrapper.translateX || 0,
- translateY = wrapper.translateY || 0,
- x = wrapper.x || 0,
- y = wrapper.y || 0,
- align = wrapper.textAlign || 'left',
- alignCorrection = {
- left: 0,
- center: 0.5,
- right: 1
- }[align],
- styles = wrapper.styles,
- whiteSpace = styles && styles.whiteSpace;
- /**
- * @private
- * @return {number}
- */
- function getTextPxLength() {
- // Reset multiline/ellipsis in order to read width (#4928,
- // #5417)
- css(elem, {
- width: '',
- whiteSpace: whiteSpace || 'nowrap'
- });
- return elem.offsetWidth;
- }
- // apply translate
- css(elem, {
- marginLeft: translateX,
- marginTop: translateY
- });
- if (!renderer.styledMode && wrapper.shadows) { // used in labels/tooltip
- wrapper.shadows.forEach(function (shadow) {
- css(shadow, {
- marginLeft: translateX + 1,
- marginTop: translateY + 1
- });
- });
- }
- // apply inversion
- if (wrapper.inverted) { // wrapper is a group
- [].forEach.call(elem.childNodes, function (child) {
- renderer.invertChild(child, elem);
- });
- }
- if (elem.tagName === 'SPAN') {
- var rotation = wrapper.rotation, baseline, textWidth = wrapper.textWidth && pInt(wrapper.textWidth), currentTextTransform = [
- rotation,
- align,
- elem.innerHTML,
- wrapper.textWidth,
- wrapper.textAlign
- ].join(',');
- // Update textWidth. Use the memoized textPxLength if possible, to
- // avoid the getTextPxLength function using elem.offsetWidth.
- // Calling offsetWidth affects rendering time as it forces layout
- // (#7656).
- if (textWidth !== wrapper.oldTextWidth &&
- ((textWidth > wrapper.oldTextWidth) ||
- (wrapper.textPxLength || getTextPxLength()) > textWidth) && (
- // Only set the width if the text is able to word-wrap, or
- // text-overflow is ellipsis (#9537)
- /[ \-]/.test(elem.textContent || elem.innerText) ||
- elem.style.textOverflow === 'ellipsis')) { // #983, #1254
- css(elem, {
- width: textWidth + 'px',
- display: 'block',
- whiteSpace: whiteSpace || 'normal' // #3331
- });
- wrapper.oldTextWidth = textWidth;
- wrapper.hasBoxWidthChanged = true; // #8159
- }
- else {
- wrapper.hasBoxWidthChanged = false; // #8159
- }
- // Do the calculations and DOM access only if properties changed
- if (currentTextTransform !== wrapper.cTT) {
- baseline = renderer.fontMetrics(elem.style.fontSize, elem).b;
- // Renderer specific handling of span rotation, but only if we
- // have something to update.
- if (defined(rotation) &&
- ((rotation !== (wrapper.oldRotation || 0)) ||
- (align !== wrapper.oldAlign))) {
- wrapper.setSpanRotation(rotation, alignCorrection, baseline);
- }
- wrapper.getSpanCorrection(
- // Avoid elem.offsetWidth if we can, it affects rendering
- // time heavily (#7656)
- ((!defined(rotation) && wrapper.textPxLength) || // #7920
- elem.offsetWidth), baseline, alignCorrection, rotation, align);
- }
- // apply position with correction
- css(elem, {
- left: (x + (wrapper.xCorr || 0)) + 'px',
- top: (y + (wrapper.yCorr || 0)) + 'px'
- });
- // record current text transform
- wrapper.cTT = currentTextTransform;
- wrapper.oldRotation = rotation;
- wrapper.oldAlign = align;
- }
- },
- /**
- * Set the rotation of an individual HTML span.
- *
- * @private
- * @function Highcharts.SVGElement#setSpanRotation
- * @param {number} rotation
- * @param {number} alignCorrection
- * @param {number} baseline
- * @return {void}
- */
- setSpanRotation: function (rotation, alignCorrection, baseline) {
- var rotationStyle = {},
- cssTransformKey = this.renderer.getTransformKey();
- rotationStyle[cssTransformKey] = rotationStyle.transform =
- 'rotate(' + rotation + 'deg)';
- rotationStyle[cssTransformKey + (isFirefox ? 'Origin' : '-origin')] =
- rotationStyle.transformOrigin =
- (alignCorrection * 100) + '% ' + baseline + 'px';
- css(this.element, rotationStyle);
- },
- /**
- * Get the correction in X and Y positioning as the element is rotated.
- *
- * @private
- * @function Highcharts.SVGElement#getSpanCorrection
- * @param {number} width
- * @param {number} baseline
- * @param {number} alignCorrection
- * @return {void}
- */
- getSpanCorrection: function (width, baseline, alignCorrection) {
- this.xCorr = -width * alignCorrection;
- this.yCorr = -baseline;
- }
- });
- // Extend SvgRenderer for useHTML option.
- extend(SVGRenderer.prototype, /** @lends SVGRenderer.prototype */ {
- /**
- * @private
- * @function Highcharts.SVGRenderer#getTransformKey
- *
- * @return {string}
- */
- getTransformKey: function () {
- return isMS && !/Edge/.test(win.navigator.userAgent) ?
- '-ms-transform' :
- isWebKit ?
- '-webkit-transform' :
- isFirefox ?
- 'MozTransform' :
- win.opera ?
- '-o-transform' :
- '';
- },
- /**
- * Create HTML text node. This is used by the VML renderer as well as the
- * SVG renderer through the useHTML option.
- *
- * @private
- * @function Highcharts.SVGRenderer#html
- *
- * @param {string} str
- * The text of (subset) HTML to draw.
- *
- * @param {number} x
- * The x position of the text's lower left corner.
- *
- * @param {number} y
- * The y position of the text's lower left corner.
- *
- * @return {Highcharts.HTMLDOMElement}
- */
- html: function (str, x, y) {
- var wrapper = this.createElement('span'), element = wrapper.element, renderer = wrapper.renderer, isSVG = renderer.isSVG, addSetters = function (gWrapper, style) {
- // These properties are set as attributes on the SVG group, and
- // as identical CSS properties on the div. (#3542)
- ['opacity', 'visibility'].forEach(function (prop) {
- gWrapper[prop + 'Setter'] = function (value, key, elem) {
- var styleObject = gWrapper.div ?
- gWrapper.div.style :
- style;
- SVGElement.prototype[prop + 'Setter']
- .call(this, value, key, elem);
- if (styleObject) {
- styleObject[key] = value;
- }
- };
- });
- gWrapper.addedSetters = true;
- };
- // Text setter
- wrapper.textSetter = function (value) {
- if (value !== element.innerHTML) {
- delete this.bBox;
- delete this.oldTextWidth;
- }
- this.textStr = value;
- element.innerHTML = pick(value, '');
- wrapper.doTransform = true;
- };
- // Add setters for the element itself (#4938)
- if (isSVG) { // #4938, only for HTML within SVG
- addSetters(wrapper, wrapper.element.style);
- }
- // Various setters which rely on update transform
- wrapper.xSetter =
- wrapper.ySetter =
- wrapper.alignSetter =
- wrapper.rotationSetter =
- function (value, key) {
- if (key === 'align') {
- // Do not overwrite the SVGElement.align method. Same as VML.
- wrapper.alignValue = wrapper.textAlign = value;
- }
- else {
- wrapper[key] = value;
- }
- wrapper.doTransform = true;
- };
- // Runs at the end of .attr()
- wrapper.afterSetters = function () {
- // Update transform. Do this outside the loop to prevent redundant
- // updating for batch setting of attributes.
- if (this.doTransform) {
- this.htmlUpdateTransform();
- this.doTransform = false;
- }
- };
- // Set the default attributes
- wrapper
- .attr({
- text: str,
- x: Math.round(x),
- y: Math.round(y)
- })
- .css({
- position: 'absolute'
- });
- if (!renderer.styledMode) {
- wrapper.css({
- fontFamily: this.style.fontFamily,
- fontSize: this.style.fontSize
- });
- }
- // Keep the whiteSpace style outside the wrapper.styles collection
- element.style.whiteSpace = 'nowrap';
- // Use the HTML specific .css method
- wrapper.css = wrapper.htmlCss;
- // This is specific for HTML within SVG
- if (isSVG) {
- wrapper.add = function (svgGroupWrapper) {
- var htmlGroup,
- container = renderer.box.parentNode,
- parentGroup,
- parents = [];
- this.parentGroup = svgGroupWrapper;
- // Create a mock group to hold the HTML elements
- if (svgGroupWrapper) {
- htmlGroup = svgGroupWrapper.div;
- if (!htmlGroup) {
- // Read the parent chain into an array and read from top
- // down
- parentGroup = svgGroupWrapper;
- while (parentGroup) {
- parents.push(parentGroup);
- // Move up to the next parent group
- parentGroup = parentGroup.parentGroup;
- }
- // Ensure dynamically updating position when any parent
- // is translated
- parents.reverse().forEach(function (parentGroup) {
- var htmlGroupStyle,
- cls = attr(parentGroup.element, 'class');
- /**
- * Common translate setter for X and Y on the HTML
- * group. Reverted the fix for #6957 du to
- * positioning problems and offline export (#7254,
- * #7280, #7529)
- * @private
- * @param {*} value
- * @param {string} key
- * @return {void}
- */
- function translateSetter(value, key) {
- parentGroup[key] = value;
- if (key === 'translateX') {
- htmlGroupStyle.left = value + 'px';
- }
- else {
- htmlGroupStyle.top = value + 'px';
- }
- parentGroup.doTransform = true;
- }
- // Create a HTML div and append it to the parent div
- // to emulate the SVG group structure
- htmlGroup =
- parentGroup.div =
- parentGroup.div || createElement('div', cls ? { className: cls } : void 0, {
- position: 'absolute',
- left: (parentGroup.translateX || 0) + 'px',
- top: (parentGroup.translateY || 0) + 'px',
- display: parentGroup.display,
- opacity: parentGroup.opacity,
- pointerEvents: (parentGroup.styles &&
- parentGroup.styles.pointerEvents) // #5595
- // the top group is appended to container
- }, htmlGroup || container);
- // Shortcut
- htmlGroupStyle = htmlGroup.style;
- // Set listeners to update the HTML div's position
- // whenever the SVG group position is changed.
- extend(parentGroup, {
- // (#7287) Pass htmlGroup to use
- // the related group
- classSetter: (function (htmlGroup) {
- return function (value) {
- this.element.setAttribute('class', value);
- htmlGroup.className = value;
- };
- }(htmlGroup)),
- on: function () {
- if (parents[0].div) { // #6418
- wrapper.on.apply({ element: parents[0].div }, arguments);
- }
- return parentGroup;
- },
- translateXSetter: translateSetter,
- translateYSetter: translateSetter
- });
- if (!parentGroup.addedSetters) {
- addSetters(parentGroup);
- }
- });
- }
- }
- else {
- htmlGroup = container;
- }
- htmlGroup.appendChild(element);
- // Shared with VML:
- wrapper.added = true;
- if (wrapper.alignOnAdd) {
- wrapper.htmlUpdateTransform();
- }
- return wrapper;
- };
- }
- return wrapper;
- }
- });
- });
- _registerModule(_modules, 'Core/Axis/Tick.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Optional parameters for the tick.
- * @private
- * @interface Highcharts.TickParametersObject
- */ /**
- * Set category for the tick.
- * @name Highcharts.TickParametersObject#category
- * @type {string|undefined}
- */ /**
- * @name Highcharts.TickParametersObject#options
- * @type {Highcharts.Dictionary<any>|undefined}
- */ /**
- * Set tickmarkOffset for the tick.
- * @name Highcharts.TickParametersObject#tickmarkOffset
- * @type {number|undefined}
- */
- var clamp = U.clamp,
- correctFloat = U.correctFloat,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- extend = U.extend,
- fireEvent = U.fireEvent,
- isNumber = U.isNumber,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick;
- var deg2rad = H.deg2rad;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Tick class.
- *
- * @class
- * @name Highcharts.Tick
- *
- * @param {Highcharts.Axis} axis
- * The axis of the tick.
- *
- * @param {number} pos
- * The position of the tick on the axis in terms of axis values.
- *
- * @param {string} [type]
- * The type of tick, either 'minor' or an empty string
- *
- * @param {boolean} [noLabel=false]
- * Whether to disable the label or not. Defaults to false.
- *
- * @param {object} [parameters]
- * Optional parameters for the tick.
- */
- var Tick = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Tick(axis, pos, type, noLabel, parameters) {
- this.isNew = true;
- this.isNewLabel = true;
- /**
- * The related axis of the tick.
- * @name Highcharts.Tick#axis
- * @type {Highcharts.Axis}
- */
- this.axis = axis;
- /**
- * The logical position of the tick on the axis in terms of axis values.
- * @name Highcharts.Tick#pos
- * @type {number}
- */
- this.pos = pos;
- /**
- * The tick type, which can be `"minor"`, or an empty string.
- * @name Highcharts.Tick#type
- * @type {string}
- */
- this.type = type || '';
- this.parameters = parameters || {};
- /**
- * The mark offset of the tick on the axis. Usually `undefined`, numeric
- * for grid axes.
- * @name Highcharts.Tick#tickmarkOffset
- * @type {number|undefined}
- */
- this.tickmarkOffset = this.parameters.tickmarkOffset;
- this.options = this.parameters.options;
- fireEvent(this, 'init');
- if (!type && !noLabel) {
- this.addLabel();
- }
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Write the tick label.
- *
- * @private
- * @function Highcharts.Tick#addLabel
- * @return {void}
- */
- Tick.prototype.addLabel = function () {
- var tick = this,
- axis = tick.axis,
- options = axis.options,
- chart = axis.chart,
- categories = axis.categories,
- log = axis.logarithmic,
- names = axis.names,
- pos = tick.pos,
- labelOptions = pick(tick.options && tick.options.labels,
- options.labels),
- str,
- tickPositions = axis.tickPositions,
- isFirst = pos === tickPositions[0],
- isLast = pos === tickPositions[tickPositions.length - 1],
- value = this.parameters.category || (categories ?
- pick(categories[pos],
- names[pos],
- pos) :
- pos),
- label = tick.label,
- animateLabels = (!labelOptions.step || labelOptions.step === 1) &&
- axis.tickInterval === 1,
- tickPositionInfo = tickPositions.info,
- dateTimeLabelFormat,
- dateTimeLabelFormats,
- i,
- list;
- // Set the datetime label format. If a higher rank is set for this
- // position, use that. If not, use the general format.
- if (axis.dateTime && tickPositionInfo) {
- dateTimeLabelFormats = chart.time.resolveDTLFormat(options.dateTimeLabelFormats[(!options.grid &&
- tickPositionInfo.higherRanks[pos]) ||
- tickPositionInfo.unitName]);
- dateTimeLabelFormat = dateTimeLabelFormats.main;
- }
- // set properties for access in render method
- /**
- * True if the tick is the first one on the axis.
- * @name Highcharts.Tick#isFirst
- * @readonly
- * @type {boolean|undefined}
- */
- tick.isFirst = isFirst;
- /**
- * True if the tick is the last one on the axis.
- * @name Highcharts.Tick#isLast
- * @readonly
- * @type {boolean|undefined}
- */
- tick.isLast = isLast;
- // Get the string
- tick.formatCtx = {
- axis: axis,
- chart: chart,
- isFirst: isFirst,
- isLast: isLast,
- dateTimeLabelFormat: dateTimeLabelFormat,
- tickPositionInfo: tickPositionInfo,
- value: log ? correctFloat(log.lin2log(value)) : value,
- pos: pos
- };
- str = axis.labelFormatter.call(tick.formatCtx, this.formatCtx);
- // Set up conditional formatting based on the format list if existing.
- list = dateTimeLabelFormats && dateTimeLabelFormats.list;
- if (list) {
- tick.shortenLabel = function () {
- for (i = 0; i < list.length; i++) {
- label.attr({
- text: axis.labelFormatter.call(extend(tick.formatCtx, { dateTimeLabelFormat: list[i] }))
- });
- if (label.getBBox().width <
- axis.getSlotWidth(tick) - 2 *
- pick(labelOptions.padding, 5)) {
- return;
- }
- }
- label.attr({
- text: ''
- });
- };
- }
- // Call only after first render
- if (animateLabels && axis._addedPlotLB) {
- tick.moveLabel(str, labelOptions);
- }
- // First call
- if (!defined(label) && !tick.movedLabel) {
- /**
- * The rendered text label of the tick.
- * @name Highcharts.Tick#label
- * @type {Highcharts.SVGElement|undefined}
- */
- tick.label = label = tick.createLabel({ x: 0, y: 0 }, str, labelOptions);
- // Base value to detect change for new calls to getBBox
- tick.rotation = 0;
- // update
- }
- else if (label && label.textStr !== str && !animateLabels) {
- // When resetting text, also reset the width if dynamically set
- // (#8809)
- if (label.textWidth &&
- !(labelOptions.style && labelOptions.style.width) &&
- !label.styles.width) {
- label.css({ width: null });
- }
- label.attr({ text: str });
- label.textPxLength = label.getBBox().width;
- }
- };
- /**
- * Render and return the label of the tick.
- *
- * @private
- * @function Highcharts.Tick#createLabel
- * @param {Highcharts.PositionObject} xy
- * @param {string} str
- * @param {Highcharts.XAxisLabelsOptions} labelOptions
- * @return {Highcharts.SVGElement|undefined}
- */
- Tick.prototype.createLabel = function (xy, str, labelOptions) {
- var axis = this.axis,
- chart = axis.chart,
- label = defined(str) && labelOptions.enabled ?
- chart.renderer
- .text(str,
- xy.x,
- xy.y,
- labelOptions.useHTML)
- .add(axis.labelGroup) :
- null;
- // Un-rotated length
- if (label) {
- // Without position absolute, IE export sometimes is wrong
- if (!chart.styledMode) {
- label.css(merge(labelOptions.style));
- }
- label.textPxLength = label.getBBox().width;
- }
- return label;
- };
- /**
- * Destructor for the tick prototype
- *
- * @private
- * @function Highcharts.Tick#destroy
- * @return {void}
- */
- Tick.prototype.destroy = function () {
- destroyObjectProperties(this, this.axis);
- };
- /**
- * Gets the x and y positions for ticks in terms of pixels.
- *
- * @private
- * @function Highcharts.Tick#getPosition
- *
- * @param {boolean} horiz
- * Whether the tick is on an horizontal axis or not.
- *
- * @param {number} tickPos
- * Position of the tick.
- *
- * @param {number} tickmarkOffset
- * Tickmark offset for all ticks.
- *
- * @param {boolean} [old]
- * Whether the axis has changed or not.
- *
- * @return {Highcharts.PositionObject}
- * The tick position.
- *
- * @fires Highcharts.Tick#event:afterGetPosition
- */
- Tick.prototype.getPosition = function (horiz, tickPos, tickmarkOffset, old) {
- var axis = this.axis,
- chart = axis.chart,
- cHeight = (old && chart.oldChartHeight) || chart.chartHeight,
- pos;
- pos = {
- x: horiz ?
- correctFloat(axis.translate(tickPos + tickmarkOffset, null, null, old) +
- axis.transB) :
- (axis.left +
- axis.offset +
- (axis.opposite ?
- (((old && chart.oldChartWidth) ||
- chart.chartWidth) -
- axis.right -
- axis.left) :
- 0)),
- y: horiz ?
- (cHeight -
- axis.bottom +
- axis.offset -
- (axis.opposite ? axis.height : 0)) :
- correctFloat(cHeight -
- axis.translate(tickPos + tickmarkOffset, null, null, old) -
- axis.transB)
- };
- // Chrome workaround for #10516
- pos.y = clamp(pos.y, -1e5, 1e5);
- fireEvent(this, 'afterGetPosition', { pos: pos });
- return pos;
- };
- /**
- * Get the x, y position of the tick label
- *
- * @private
- * @return {Highcharts.PositionObject}
- */
- Tick.prototype.getLabelPosition = function (x, y, label, horiz, labelOptions, tickmarkOffset, index, step) {
- var axis = this.axis,
- transA = axis.transA,
- reversed = ( // #7911
- axis.isLinked && axis.linkedParent ?
- axis.linkedParent.reversed :
- axis.reversed),
- staggerLines = axis.staggerLines,
- rotCorr = axis.tickRotCorr || { x: 0,
- y: 0 },
- yOffset = labelOptions.y,
- // Adjust for label alignment if we use reserveSpace: true (#5286)
- labelOffsetCorrection = (!horiz && !axis.reserveSpaceDefault ?
- -axis.labelOffset * (axis.labelAlign === 'center' ? 0.5 : 1) :
- 0),
- line,
- pos = {};
- if (!defined(yOffset)) {
- if (axis.side === 0) {
- yOffset = label.rotation ? -8 : -label.getBBox().height;
- }
- else if (axis.side === 2) {
- yOffset = rotCorr.y + 8;
- }
- else {
- // #3140, #3140
- yOffset = Math.cos(label.rotation * deg2rad) *
- (rotCorr.y - label.getBBox(false, 0).height / 2);
- }
- }
- x = x +
- labelOptions.x +
- labelOffsetCorrection +
- rotCorr.x -
- (tickmarkOffset && horiz ?
- tickmarkOffset * transA * (reversed ? -1 : 1) :
- 0);
- y = y + yOffset - (tickmarkOffset && !horiz ?
- tickmarkOffset * transA * (reversed ? 1 : -1) : 0);
- // Correct for staggered labels
- if (staggerLines) {
- line = (index / (step || 1) % staggerLines);
- if (axis.opposite) {
- line = staggerLines - line - 1;
- }
- y += line * (axis.labelOffset / staggerLines);
- }
- pos.x = x;
- pos.y = Math.round(y);
- fireEvent(this, 'afterGetLabelPosition', { pos: pos, tickmarkOffset: tickmarkOffset, index: index });
- return pos;
- };
- /**
- * Get the offset height or width of the label
- *
- * @private
- * @function Highcharts.Tick#getLabelSize
- * @return {number}
- */
- Tick.prototype.getLabelSize = function () {
- return this.label ?
- this.label.getBBox()[this.axis.horiz ? 'height' : 'width'] :
- 0;
- };
- /**
- * Extendible method to return the path of the marker
- *
- * @private
- *
- */
- Tick.prototype.getMarkPath = function (x, y, tickLength, tickWidth, horiz, renderer) {
- return renderer.crispLine([[
- 'M',
- x,
- y
- ], [
- 'L',
- x + (horiz ? 0 : -tickLength),
- y + (horiz ? tickLength : 0)
- ]], tickWidth);
- };
- /**
- * Handle the label overflow by adjusting the labels to the left and right
- * edge, or hide them if they collide into the neighbour label.
- *
- * @private
- * @function Highcharts.Tick#handleOverflow
- * @param {Highcharts.PositionObject} xy
- * @return {void}
- */
- Tick.prototype.handleOverflow = function (xy) {
- var tick = this,
- axis = this.axis,
- labelOptions = axis.options.labels,
- pxPos = xy.x,
- chartWidth = axis.chart.chartWidth,
- spacing = axis.chart.spacing,
- leftBound = pick(axis.labelLeft,
- Math.min(axis.pos,
- spacing[3])),
- rightBound = pick(axis.labelRight,
- Math.max(!axis.isRadial ? axis.pos + axis.len : 0,
- chartWidth - spacing[1])),
- label = this.label,
- rotation = this.rotation,
- factor = {
- left: 0,
- center: 0.5,
- right: 1
- }[axis.labelAlign || label.attr('align')],
- labelWidth = label.getBBox().width,
- slotWidth = axis.getSlotWidth(tick),
- modifiedSlotWidth = slotWidth,
- xCorrection = factor,
- goRight = 1,
- leftPos,
- rightPos,
- textWidth,
- css = {};
- // Check if the label overshoots the chart spacing box. If it does, move
- // it. If it now overshoots the slotWidth, add ellipsis.
- if (!rotation &&
- pick(labelOptions.overflow, 'justify') === 'justify') {
- leftPos = pxPos - factor * labelWidth;
- rightPos = pxPos + (1 - factor) * labelWidth;
- if (leftPos < leftBound) {
- modifiedSlotWidth =
- xy.x + modifiedSlotWidth * (1 - factor) - leftBound;
- }
- else if (rightPos > rightBound) {
- modifiedSlotWidth =
- rightBound - xy.x + modifiedSlotWidth * factor;
- goRight = -1;
- }
- modifiedSlotWidth = Math.min(slotWidth, modifiedSlotWidth); // #4177
- if (modifiedSlotWidth < slotWidth && axis.labelAlign === 'center') {
- xy.x += (goRight *
- (slotWidth -
- modifiedSlotWidth -
- xCorrection * (slotWidth - Math.min(labelWidth, modifiedSlotWidth))));
- }
- // If the label width exceeds the available space, set a text width
- // to be picked up below. Also, if a width has been set before, we
- // need to set a new one because the reported labelWidth will be
- // limited by the box (#3938).
- if (labelWidth > modifiedSlotWidth ||
- (axis.autoRotation && (label.styles || {}).width)) {
- textWidth = modifiedSlotWidth;
- }
- // Add ellipsis to prevent rotated labels to be clipped against the edge
- // of the chart
- }
- else if (rotation < 0 &&
- pxPos - factor * labelWidth < leftBound) {
- textWidth = Math.round(pxPos / Math.cos(rotation * deg2rad) - leftBound);
- }
- else if (rotation > 0 &&
- pxPos + factor * labelWidth > rightBound) {
- textWidth = Math.round((chartWidth - pxPos) /
- Math.cos(rotation * deg2rad));
- }
- if (textWidth) {
- if (tick.shortenLabel) {
- tick.shortenLabel();
- }
- else {
- css.width = Math.floor(textWidth) + 'px';
- if (!(labelOptions.style || {}).textOverflow) {
- css.textOverflow = 'ellipsis';
- }
- label.css(css);
- }
- }
- };
- /**
- * Try to replace the label if the same one already exists.
- *
- * @private
- * @function Highcharts.Tick#moveLabel
- * @param {string} str
- * @param {Highcharts.XAxisLabelsOptions} labelOptions
- *
- * @return {void}
- */
- Tick.prototype.moveLabel = function (str, labelOptions) {
- var tick = this,
- label = tick.label,
- moved = false,
- axis = tick.axis,
- labelPos,
- reversed = axis.reversed,
- xPos,
- yPos;
- if (label && label.textStr === str) {
- tick.movedLabel = label;
- moved = true;
- delete tick.label;
- }
- else { // Find a label with the same string
- objectEach(axis.ticks, function (currentTick) {
- if (!moved &&
- !currentTick.isNew &&
- currentTick !== tick &&
- currentTick.label &&
- currentTick.label.textStr === str) {
- tick.movedLabel = currentTick.label;
- moved = true;
- currentTick.labelPos = tick.movedLabel.xy;
- delete currentTick.label;
- }
- });
- }
- // Create new label if the actual one is moved
- if (!moved && (tick.labelPos || label)) {
- labelPos = tick.labelPos || label.xy;
- xPos = axis.horiz ?
- (reversed ? 0 : axis.width + axis.left) : labelPos.x;
- yPos = axis.horiz ?
- labelPos.y : (reversed ? (axis.width + axis.left) : 0);
- tick.movedLabel = tick.createLabel({ x: xPos, y: yPos }, str, labelOptions);
- if (tick.movedLabel) {
- tick.movedLabel.attr({ opacity: 0 });
- }
- }
- };
- /**
- * Put everything in place
- *
- * @private
- * @param {number} index
- * @param {boolean} [old]
- * Use old coordinates to prepare an animation into new position
- * @param {number} [opacity]
- * @return {voids}
- */
- Tick.prototype.render = function (index, old, opacity) {
- var tick = this,
- axis = tick.axis,
- horiz = axis.horiz,
- pos = tick.pos,
- tickmarkOffset = pick(tick.tickmarkOffset,
- axis.tickmarkOffset),
- xy = tick.getPosition(horiz,
- pos,
- tickmarkOffset,
- old),
- x = xy.x,
- y = xy.y,
- reverseCrisp = ((horiz && x === axis.pos + axis.len) ||
- (!horiz && y === axis.pos)) ? -1 : 1; // #1480, #1687
- opacity = pick(opacity, 1);
- this.isActive = true;
- // Create the grid line
- this.renderGridLine(old, opacity, reverseCrisp);
- // create the tick mark
- this.renderMark(xy, opacity, reverseCrisp);
- // the label is created on init - now move it into place
- this.renderLabel(xy, old, opacity, index);
- tick.isNew = false;
- fireEvent(this, 'afterRender');
- };
- /**
- * Renders the gridLine.
- *
- * @private
- * @param {boolean} old Whether or not the tick is old
- * @param {number} opacity The opacity of the grid line
- * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1
- * @return {void}
- */
- Tick.prototype.renderGridLine = function (old, opacity, reverseCrisp) {
- var tick = this, axis = tick.axis, options = axis.options, gridLine = tick.gridLine, gridLinePath, attribs = {}, pos = tick.pos, type = tick.type, tickmarkOffset = pick(tick.tickmarkOffset, axis.tickmarkOffset), renderer = axis.chart.renderer, gridPrefix = type ? type + 'Grid' : 'grid', gridLineWidth = options[gridPrefix + 'LineWidth'], gridLineColor = options[gridPrefix + 'LineColor'], dashStyle = options[gridPrefix + 'LineDashStyle'];
- if (!gridLine) {
- if (!axis.chart.styledMode) {
- attribs.stroke = gridLineColor;
- attribs['stroke-width'] = gridLineWidth;
- if (dashStyle) {
- attribs.dashstyle = dashStyle;
- }
- }
- if (!type) {
- attribs.zIndex = 1;
- }
- if (old) {
- opacity = 0;
- }
- /**
- * The rendered grid line of the tick.
- * @name Highcharts.Tick#gridLine
- * @type {Highcharts.SVGElement|undefined}
- */
- tick.gridLine = gridLine = renderer.path()
- .attr(attribs)
- .addClass('highcharts-' + (type ? type + '-' : '') + 'grid-line')
- .add(axis.gridGroup);
- }
- if (gridLine) {
- gridLinePath = axis.getPlotLinePath({
- value: pos + tickmarkOffset,
- lineWidth: gridLine.strokeWidth() * reverseCrisp,
- force: 'pass',
- old: old
- });
- // If the parameter 'old' is set, the current call will be followed
- // by another call, therefore do not do any animations this time
- if (gridLinePath) {
- gridLine[old || tick.isNew ? 'attr' : 'animate']({
- d: gridLinePath,
- opacity: opacity
- });
- }
- }
- };
- /**
- * Renders the tick mark.
- *
- * @private
- * @param {Highcharts.PositionObject} xy The position vector of the mark
- * @param {number} opacity The opacity of the mark
- * @param {number} reverseCrisp Modifier for avoiding overlapping 1 or -1
- * @return {void}
- */
- Tick.prototype.renderMark = function (xy, opacity, reverseCrisp) {
- var tick = this, axis = tick.axis, options = axis.options, renderer = axis.chart.renderer, type = tick.type, tickPrefix = type ? type + 'Tick' : 'tick', tickSize = axis.tickSize(tickPrefix), mark = tick.mark, isNewMark = !mark, x = xy.x, y = xy.y, tickWidth = pick(options[tickPrefix + 'Width'], !type && axis.isXAxis ? 1 : 0), // X axis defaults to 1
- tickColor = options[tickPrefix + 'Color'];
- if (tickSize) {
- // negate the length
- if (axis.opposite) {
- tickSize[0] = -tickSize[0];
- }
- // First time, create it
- if (isNewMark) {
- /**
- * The rendered mark of the tick.
- * @name Highcharts.Tick#mark
- * @type {Highcharts.SVGElement|undefined}
- */
- tick.mark = mark = renderer.path()
- .addClass('highcharts-' + (type ? type + '-' : '') + 'tick')
- .add(axis.axisGroup);
- if (!axis.chart.styledMode) {
- mark.attr({
- stroke: tickColor,
- 'stroke-width': tickWidth
- });
- }
- }
- mark[isNewMark ? 'attr' : 'animate']({
- d: tick.getMarkPath(x, y, tickSize[0], mark.strokeWidth() * reverseCrisp, axis.horiz, renderer),
- opacity: opacity
- });
- }
- };
- /**
- * Renders the tick label.
- * Note: The label should already be created in init(), so it should only
- * have to be moved into place.
- *
- * @private
- * @param {Highcharts.PositionObject} xy The position vector of the label
- * @param {boolean} old Whether or not the tick is old
- * @param {number} opacity The opacity of the label
- * @param {number} index The index of the tick
- * @return {void}
- */
- Tick.prototype.renderLabel = function (xy, old, opacity, index) {
- var tick = this,
- axis = tick.axis,
- horiz = axis.horiz,
- options = axis.options,
- label = tick.label,
- labelOptions = options.labels,
- step = labelOptions.step,
- tickmarkOffset = pick(tick.tickmarkOffset,
- axis.tickmarkOffset),
- show = true,
- x = xy.x,
- y = xy.y;
- if (label && isNumber(x)) {
- label.xy = xy = tick.getLabelPosition(x, y, label, horiz, labelOptions, tickmarkOffset, index, step);
- // Apply show first and show last. If the tick is both first and
- // last, it is a single centered tick, in which case we show the
- // label anyway (#2100).
- if ((tick.isFirst &&
- !tick.isLast &&
- !pick(options.showFirstLabel, 1)) ||
- (tick.isLast &&
- !tick.isFirst &&
- !pick(options.showLastLabel, 1))) {
- show = false;
- // Handle label overflow and show or hide accordingly
- }
- else if (horiz &&
- !labelOptions.step &&
- !labelOptions.rotation &&
- !old &&
- opacity !== 0) {
- tick.handleOverflow(xy);
- }
- // apply step
- if (step && index % step) {
- // show those indices dividable by step
- show = false;
- }
- // Set the new position, and show or hide
- if (show && isNumber(xy.y)) {
- xy.opacity = opacity;
- label[tick.isNewLabel ? 'attr' : 'animate'](xy);
- tick.isNewLabel = false;
- }
- else {
- label.attr('y', -9999); // #1338
- tick.isNewLabel = true;
- }
- }
- };
- /**
- * Replace labels with the moved ones to perform animation. Additionally
- * destroy unused labels.
- *
- * @private
- * @function Highcharts.Tick#replaceMovedLabel
- * @return {void}
- */
- Tick.prototype.replaceMovedLabel = function () {
- var tick = this,
- label = tick.label,
- axis = tick.axis,
- reversed = axis.reversed,
- x,
- y;
- // Animate and destroy
- if (label && !tick.isNew) {
- x = axis.horiz ? (reversed ? axis.left : axis.width + axis.left) : label.xy.x;
- y = axis.horiz ?
- label.xy.y :
- (reversed ? axis.width + axis.top : axis.top);
- label.animate({ x: x, y: y, opacity: 0 }, void 0, label.destroy);
- delete tick.label;
- }
- axis.isDirty = true;
- tick.label = tick.movedLabel;
- delete tick.movedLabel;
- };
- return Tick;
- }());
- H.Tick = Tick;
- return H.Tick;
- });
- _registerModule(_modules, 'Core/Time.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Highcharts, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Normalized interval.
- *
- * @interface Highcharts.TimeNormalizedObject
- */ /**
- * The count.
- *
- * @name Highcharts.TimeNormalizedObject#count
- * @type {number}
- */ /**
- * The interval in axis values (ms).
- *
- * @name Highcharts.TimeNormalizedObject#unitRange
- * @type {number}
- */
- /**
- * Function of an additional date format specifier.
- *
- * @callback Highcharts.TimeFormatCallbackFunction
- *
- * @param {number} timestamp
- * The time to format.
- *
- * @return {string}
- * The formatted portion of the date.
- */
- /**
- * Additonal time tick information.
- *
- * @interface Highcharts.TimeTicksInfoObject
- * @extends Highcharts.TimeNormalizedObject
- */ /**
- * @name Highcharts.TimeTicksInfoObject#higherRanks
- * @type {Array<string>}
- */ /**
- * @name Highcharts.TimeTicksInfoObject#totalRange
- * @type {number}
- */
- /**
- * Time ticks.
- *
- * @interface Highcharts.AxisTickPositionsArray
- * @extends global.Array<number>
- */ /**
- * @name Highcharts.AxisTickPositionsArray#info
- * @type {Highcharts.TimeTicksInfoObject|undefined}
- */
- /**
- * A callback to return the time zone offset for a given datetime. It
- * takes the timestamp in terms of milliseconds since January 1 1970,
- * and returns the timezone offset in minutes. This provides a hook
- * for drawing time based charts in specific time zones using their
- * local DST crossover dates, with the help of external libraries.
- *
- * @callback Highcharts.TimezoneOffsetCallbackFunction
- *
- * @param {number} timestamp
- * Timestamp in terms of milliseconds since January 1 1970.
- *
- * @return {number}
- * Timezone offset in minutes.
- */
- /**
- * Allows to manually load the `moment.js` library from Highcharts options
- * instead of the `window`.
- * In case of loading the library from a `script` tag,
- * this option is not needed, it will be loaded from there by default.
- *
- * @type {function}
- * @since 8.2.0
- * @apioption time.moment
- */
- var defined = U.defined,
- error = U.error,
- extend = U.extend,
- isObject = U.isObject,
- merge = U.merge,
- objectEach = U.objectEach,
- pad = U.pad,
- pick = U.pick,
- splat = U.splat,
- timeUnits = U.timeUnits;
- var H = Highcharts,
- win = H.win;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Time class. Time settings are applied in general for each page using
- * `Highcharts.setOptions`, or individually for each Chart item through the
- * [time](https://api.highcharts.com/highcharts/time) options set.
- *
- * The Time object is available from {@link Highcharts.Chart#time},
- * which refers to `Highcharts.time` if no individual time settings are
- * applied.
- *
- * @example
- * // Apply time settings globally
- * Highcharts.setOptions({
- * time: {
- * timezone: 'Europe/London'
- * }
- * });
- *
- * // Apply time settings by instance
- * var chart = Highcharts.chart('container', {
- * time: {
- * timezone: 'America/New_York'
- * },
- * series: [{
- * data: [1, 4, 3, 5]
- * }]
- * });
- *
- * // Use the Time object
- * console.log(
- * 'Current time in New York',
- * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now())
- * );
- *
- * @since 6.0.5
- *
- * @class
- * @name Highcharts.Time
- *
- * @param {Highcharts.TimeOptions} options
- * Time options as defined in [chart.options.time](/highcharts/time).
- */
- var Time = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Time(options) {
- /* *
- *
- * Properties
- *
- * */
- this.options = {};
- this.useUTC = false;
- this.variableTimezone = false;
- this.Date = win.Date;
- /**
- * Get the time zone offset based on the current timezone information as
- * set in the global options.
- *
- * @function Highcharts.Time#getTimezoneOffset
- *
- * @param {number} timestamp
- * The JavaScript timestamp to inspect.
- *
- * @return {number}
- * The timezone offset in minutes compared to UTC.
- */
- this.getTimezoneOffset = this.timezoneOffsetFunction();
- this.update(options);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Time units used in `Time.get` and `Time.set`
- *
- * @typedef {"Date"|"Day"|"FullYear"|"Hours"|"Milliseconds"|"Minutes"|"Month"|"Seconds"} Highcharts.TimeUnitValue
- */
- /**
- * Get the value of a date object in given units, and subject to the Time
- * object's current timezone settings. This function corresponds directly to
- * JavaScripts `Date.getXXX / Date.getUTCXXX`, so instead of calling
- * `date.getHours()` or `date.getUTCHours()` we will call
- * `time.get('Hours')`.
- *
- * @function Highcharts.Time#get
- *
- * @param {Highcharts.TimeUnitValue} unit
- * @param {Date} date
- *
- * @return {number}
- * The given time unit
- */
- Time.prototype.get = function (unit, date) {
- if (this.variableTimezone || this.timezoneOffset) {
- var realMs = date.getTime();
- var ms = realMs - this.getTimezoneOffset(date);
- date.setTime(ms); // Temporary adjust to timezone
- var ret = date['getUTC' + unit]();
- date.setTime(realMs); // Reset
- return ret;
- }
- // UTC time with no timezone handling
- if (this.useUTC) {
- return date['getUTC' + unit]();
- }
- // Else, local time
- return date['get' + unit]();
- };
- /**
- * Set the value of a date object in given units, and subject to the Time
- * object's current timezone settings. This function corresponds directly to
- * JavaScripts `Date.setXXX / Date.setUTCXXX`, so instead of calling
- * `date.setHours(0)` or `date.setUTCHours(0)` we will call
- * `time.set('Hours', 0)`.
- *
- * @function Highcharts.Time#set
- *
- * @param {Highcharts.TimeUnitValue} unit
- * @param {Date} date
- * @param {number} value
- *
- * @return {number}
- * The epoch milliseconds of the updated date
- */
- Time.prototype.set = function (unit, date, value) {
- // UTC time with timezone handling
- if (this.variableTimezone || this.timezoneOffset) {
- // For lower order time units, just set it directly using UTC
- // time
- if (unit === 'Milliseconds' ||
- unit === 'Seconds' ||
- unit === 'Minutes') {
- return date['setUTC' + unit](value);
- }
- // Higher order time units need to take the time zone into
- // account
- // Adjust by timezone
- var offset = this.getTimezoneOffset(date);
- var ms = date.getTime() - offset;
- date.setTime(ms);
- date['setUTC' + unit](value);
- var newOffset = this.getTimezoneOffset(date);
- ms = date.getTime() + newOffset;
- return date.setTime(ms);
- }
- // UTC time with no timezone handling
- if (this.useUTC) {
- return date['setUTC' + unit](value);
- }
- // Else, local time
- return date['set' + unit](value);
- };
- /**
- * Update the Time object with current options. It is called internally on
- * initializing Highcharts, after running `Highcharts.setOptions` and on
- * `Chart.update`.
- *
- * @private
- * @function Highcharts.Time#update
- *
- * @param {Highcharts.TimeOptions} options
- *
- * @return {void}
- */
- Time.prototype.update = function (options) {
- var useUTC = pick(options && options.useUTC,
- true),
- time = this;
- this.options = options = merge(true, this.options || {}, options);
- // Allow using a different Date class
- this.Date = options.Date || win.Date || Date;
- this.useUTC = useUTC;
- this.timezoneOffset = (useUTC && options.timezoneOffset);
- this.getTimezoneOffset = this.timezoneOffsetFunction();
- /*
- * The time object has options allowing for variable time zones, meaning
- * the axis ticks or series data needs to consider this.
- */
- this.variableTimezone = !!(!useUTC ||
- options.getTimezoneOffset ||
- options.timezone);
- };
- /**
- * Make a time and returns milliseconds. Interprets the inputs as UTC time,
- * local time or a specific timezone time depending on the current time
- * settings.
- *
- * @function Highcharts.Time#makeTime
- *
- * @param {number} year
- * The year
- *
- * @param {number} month
- * The month. Zero-based, so January is 0.
- *
- * @param {number} [date=1]
- * The day of the month
- *
- * @param {number} [hours=0]
- * The hour of the day, 0-23.
- *
- * @param {number} [minutes=0]
- * The minutes
- *
- * @param {number} [seconds=0]
- * The seconds
- *
- * @return {number}
- * The time in milliseconds since January 1st 1970.
- */
- Time.prototype.makeTime = function (year, month, date, hours, minutes, seconds) {
- var d,
- offset,
- newOffset;
- if (this.useUTC) {
- d = this.Date.UTC.apply(0, arguments);
- offset = this.getTimezoneOffset(d);
- d += offset;
- newOffset = this.getTimezoneOffset(d);
- if (offset !== newOffset) {
- d += newOffset - offset;
- // A special case for transitioning from summer time to winter time.
- // When the clock is set back, the same time is repeated twice, i.e.
- // 02:30 am is repeated since the clock is set back from 3 am to
- // 2 am. We need to make the same time as local Date does.
- }
- else if (offset - 36e5 === this.getTimezoneOffset(d - 36e5) &&
- !H.isSafari) {
- d -= 36e5;
- }
- }
- else {
- d = new this.Date(year, month, pick(date, 1), pick(hours, 0), pick(minutes, 0), pick(seconds, 0)).getTime();
- }
- return d;
- };
- /**
- * Sets the getTimezoneOffset function. If the `timezone` option is set, a
- * default getTimezoneOffset function with that timezone is returned. If
- * a `getTimezoneOffset` option is defined, it is returned. If neither are
- * specified, the function using the `timezoneOffset` option or 0 offset is
- * returned.
- *
- * @private
- * @function Highcharts.Time#timezoneOffsetFunction
- *
- * @return {Function}
- * A getTimezoneOffset function
- */
- Time.prototype.timezoneOffsetFunction = function () {
- var time = this,
- options = this.options,
- moment = options.moment || win.moment;
- if (!this.useUTC) {
- return function (timestamp) {
- return new Date(timestamp.toString()).getTimezoneOffset() * 60000;
- };
- }
- if (options.timezone) {
- if (!moment) {
- // getTimezoneOffset-function stays undefined because it depends
- // on Moment.js
- error(25);
- }
- else {
- return function (timestamp) {
- return -moment.tz(timestamp, options.timezone).utcOffset() * 60000;
- };
- }
- }
- // If not timezone is set, look for the getTimezoneOffset callback
- if (this.useUTC && options.getTimezoneOffset) {
- return function (timestamp) {
- return options.getTimezoneOffset(timestamp.valueOf()) * 60000;
- };
- }
- // Last, use the `timezoneOffset` option if set
- return function () {
- return (time.timezoneOffset || 0) * 60000;
- };
- };
- /**
- * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970)
- * into a human readable date string. The available format keys are listed
- * below. Additional formats can be given in the
- * {@link Highcharts.dateFormats} hook.
- *
- * Supported format keys:
- * - `%a`: Short weekday, like 'Mon'
- * - `%A`: Long weekday, like 'Monday'
- * - `%d`: Two digit day of the month, 01 to 31
- * - `%e`: Day of the month, 1 through 31
- * - `%w`: Day of the week, 0 through 6
- * - `%b`: Short month, like 'Jan'
- * - `%B`: Long month, like 'January'
- * - `%m`: Two digit month number, 01 through 12
- * - `%y`: Two digits year, like 09 for 2009
- * - `%Y`: Four digits year, like 2009
- * - `%H`: Two digits hours in 24h format, 00 through 23
- * - `%k`: Hours in 24h format, 0 through 23
- * - `%I`: Two digits hours in 12h format, 00 through 11
- * - `%l`: Hours in 12h format, 1 through 12
- * - `%M`: Two digits minutes, 00 through 59
- * - `%p`: Upper case AM or PM
- * - `%P`: Lower case AM or PM
- * - `%S`: Two digits seconds, 00 through 59
- * - `%L`: Milliseconds (naming from Ruby)
- *
- * @example
- * const time = new Highcharts.Time();
- * const s = time.dateFormat('%Y-%m-%d %H:%M:%S', Date.UTC(2020, 0, 1));
- * console.log(s); // => 2020-01-01 00:00:00
- *
- * @function Highcharts.Time#dateFormat
- *
- * @param {string} format
- * The desired format where various time representations are
- * prefixed with %.
- *
- * @param {number} timestamp
- * The JavaScript timestamp.
- *
- * @param {boolean} [capitalize=false]
- * Upper case first letter in the return.
- *
- * @return {string}
- * The formatted date.
- */
- Time.prototype.dateFormat = function (format, timestamp, capitalize) {
- var _a;
- if (!defined(timestamp) || isNaN(timestamp)) {
- return ((_a = H.defaultOptions.lang) === null || _a === void 0 ? void 0 : _a.invalidDate) || '';
- }
- format = pick(format, '%Y-%m-%d %H:%M:%S');
- var time = this, date = new this.Date(timestamp),
- // get the basic time values
- hours = this.get('Hours', date), day = this.get('Day', date), dayOfMonth = this.get('Date', date), month = this.get('Month', date), fullYear = this.get('FullYear', date), lang = H.defaultOptions.lang, langWeekdays = lang === null || lang === void 0 ? void 0 : lang.weekdays, shortWeekdays = lang === null || lang === void 0 ? void 0 : lang.shortWeekdays,
- // List all format keys. Custom formats can be added from the
- // outside.
- replacements = extend({
- // Day
- // Short weekday, like 'Mon'
- a: shortWeekdays ?
- shortWeekdays[day] :
- langWeekdays[day].substr(0, 3),
- // Long weekday, like 'Monday'
- A: langWeekdays[day],
- // Two digit day of the month, 01 to 31
- d: pad(dayOfMonth),
- // Day of the month, 1 through 31
- e: pad(dayOfMonth, 2, ' '),
- // Day of the week, 0 through 6
- w: day,
- // Week (none implemented)
- // 'W': weekNumber(),
- // Month
- // Short month, like 'Jan'
- b: lang.shortMonths[month],
- // Long month, like 'January'
- B: lang.months[month],
- // Two digit month number, 01 through 12
- m: pad(month + 1),
- // Month number, 1 through 12 (#8150)
- o: month + 1,
- // Year
- // Two digits year, like 09 for 2009
- y: fullYear.toString().substr(2, 2),
- // Four digits year, like 2009
- Y: fullYear,
- // Time
- // Two digits hours in 24h format, 00 through 23
- H: pad(hours),
- // Hours in 24h format, 0 through 23
- k: hours,
- // Two digits hours in 12h format, 00 through 11
- I: pad((hours % 12) || 12),
- // Hours in 12h format, 1 through 12
- l: (hours % 12) || 12,
- // Two digits minutes, 00 through 59
- M: pad(this.get('Minutes', date)),
- // Upper case AM or PM
- p: hours < 12 ? 'AM' : 'PM',
- // Lower case AM or PM
- P: hours < 12 ? 'am' : 'pm',
- // Two digits seconds, 00 through 59
- S: pad(date.getSeconds()),
- // Milliseconds (naming from Ruby)
- L: pad(Math.floor(timestamp % 1000), 3)
- }, H.dateFormats);
- // Do the replaces
- objectEach(replacements, function (val, key) {
- // Regex would do it in one line, but this is faster
- while (format.indexOf('%' + key) !== -1) {
- format = format.replace('%' + key, typeof val === 'function' ? val.call(time, timestamp) : val);
- }
- });
- // Optionally capitalize the string and return
- return capitalize ?
- (format.substr(0, 1).toUpperCase() +
- format.substr(1)) :
- format;
- };
- /**
- * Resolve legacy formats of dateTimeLabelFormats (strings and arrays) into
- * an object.
- * @private
- * @param {string|Array<T>|Highcharts.Dictionary<T>} f - General format description
- * @return {Highcharts.Dictionary<T>} - The object definition
- */
- Time.prototype.resolveDTLFormat = function (f) {
- if (!isObject(f, true)) { // check for string or array
- f = splat(f);
- return {
- main: f[0],
- from: f[1],
- to: f[2]
- };
- }
- return f;
- };
- /**
- * Return an array with time positions distributed on round time values
- * right and right after min and max. Used in datetime axes as well as for
- * grouping data on a datetime axis.
- *
- * @function Highcharts.Time#getTimeTicks
- *
- * @param {Highcharts.TimeNormalizedObject} normalizedInterval
- * The interval in axis values (ms) and the count
- *
- * @param {number} [min]
- * The minimum in axis values
- *
- * @param {number} [max]
- * The maximum in axis values
- *
- * @param {number} [startOfWeek=1]
- *
- * @return {Highcharts.AxisTickPositionsArray}
- */
- Time.prototype.getTimeTicks = function (normalizedInterval, min, max, startOfWeek) {
- var time = this,
- Date = time.Date,
- tickPositions = [],
- i,
- higherRanks = {},
- minYear, // used in months and years as a basis for Date.UTC()
- // When crossing DST, use the max. Resolves #6278.
- minDate = new Date(min),
- interval = normalizedInterval.unitRange,
- count = normalizedInterval.count || 1,
- variableDayLength,
- minDay;
- startOfWeek = pick(startOfWeek, 1);
- if (defined(min)) { // #1300
- time.set('Milliseconds', minDate, interval >= timeUnits.second ?
- 0 : // #3935
- count * Math.floor(time.get('Milliseconds', minDate) / count)); // #3652, #3654
- if (interval >= timeUnits.second) { // second
- time.set('Seconds', minDate, interval >= timeUnits.minute ?
- 0 : // #3935
- count * Math.floor(time.get('Seconds', minDate) / count));
- }
- if (interval >= timeUnits.minute) { // minute
- time.set('Minutes', minDate, interval >= timeUnits.hour ?
- 0 :
- count * Math.floor(time.get('Minutes', minDate) / count));
- }
- if (interval >= timeUnits.hour) { // hour
- time.set('Hours', minDate, interval >= timeUnits.day ?
- 0 :
- count * Math.floor(time.get('Hours', minDate) / count));
- }
- if (interval >= timeUnits.day) { // day
- time.set('Date', minDate, interval >= timeUnits.month ?
- 1 :
- Math.max(1, count * Math.floor(time.get('Date', minDate) / count)));
- }
- if (interval >= timeUnits.month) { // month
- time.set('Month', minDate, interval >= timeUnits.year ? 0 :
- count * Math.floor(time.get('Month', minDate) / count));
- minYear = time.get('FullYear', minDate);
- }
- if (interval >= timeUnits.year) { // year
- minYear -= minYear % count;
- time.set('FullYear', minDate, minYear);
- }
- // week is a special case that runs outside the hierarchy
- if (interval === timeUnits.week) {
- // get start of current week, independent of count
- minDay = time.get('Day', minDate);
- time.set('Date', minDate, (time.get('Date', minDate) -
- minDay + startOfWeek +
- // We don't want to skip days that are before
- // startOfWeek (#7051)
- (minDay < startOfWeek ? -7 : 0)));
- }
- // Get basics for variable time spans
- minYear = time.get('FullYear', minDate);
- var minMonth = time.get('Month', minDate), minDateDate = time.get('Date', minDate), minHours = time.get('Hours', minDate);
- // Redefine min to the floored/rounded minimum time (#7432)
- min = minDate.getTime();
- // Handle local timezone offset
- if (time.variableTimezone) {
- // Detect whether we need to take the DST crossover into
- // consideration. If we're crossing over DST, the day length may
- // be 23h or 25h and we need to compute the exact clock time for
- // each tick instead of just adding hours. This comes at a cost,
- // so first we find out if it is needed (#4951).
- variableDayLength = (
- // Long range, assume we're crossing over.
- max - min > 4 * timeUnits.month ||
- // Short range, check if min and max are in different time
- // zones.
- time.getTimezoneOffset(min) !==
- time.getTimezoneOffset(max));
- }
- // Iterate and add tick positions at appropriate values
- var t = minDate.getTime();
- i = 1;
- while (t < max) {
- tickPositions.push(t);
- // if the interval is years, use Date.UTC to increase years
- if (interval === timeUnits.year) {
- t = time.makeTime(minYear + i * count, 0);
- // if the interval is months, use Date.UTC to increase months
- }
- else if (interval === timeUnits.month) {
- t = time.makeTime(minYear, minMonth + i * count);
- // if we're using global time, the interval is not fixed as it
- // jumps one hour at the DST crossover
- }
- else if (variableDayLength &&
- (interval === timeUnits.day || interval === timeUnits.week)) {
- t = time.makeTime(minYear, minMonth, minDateDate +
- i * count * (interval === timeUnits.day ? 1 : 7));
- }
- else if (variableDayLength &&
- interval === timeUnits.hour &&
- count > 1) {
- // make sure higher ranks are preserved across DST (#6797,
- // #7621)
- t = time.makeTime(minYear, minMonth, minDateDate, minHours + i * count);
- // else, the interval is fixed and we use simple addition
- }
- else {
- t += interval * count;
- }
- i++;
- }
- // push the last time
- tickPositions.push(t);
- // Handle higher ranks. Mark new days if the time is on midnight
- // (#950, #1649, #1760, #3349). Use a reasonable dropout threshold
- // to prevent looping over dense data grouping (#6156).
- if (interval <= timeUnits.hour && tickPositions.length < 10000) {
- tickPositions.forEach(function (t) {
- if (
- // Speed optimization, no need to run dateFormat unless
- // we're on a full or half hour
- t % 1800000 === 0 &&
- // Check for local or global midnight
- time.dateFormat('%H%M%S%L', t) === '000000000') {
- higherRanks[t] = 'day';
- }
- });
- }
- }
- // record information on the chosen unit - for dynamic label formatter
- tickPositions.info = extend(normalizedInterval, {
- higherRanks: higherRanks,
- totalRange: interval * count
- });
- return tickPositions;
- };
- return Time;
- }());
- H.Time = Time;
- return H.Time;
- });
- _registerModule(_modules, 'Core/Options.js', [_modules['Core/Globals.js'], _modules['Core/Time.js'], _modules['Core/Color.js'], _modules['Core/Utilities.js']], function (H, Time, Color, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * @typedef {"plotBox"|"spacingBox"} Highcharts.ButtonRelativeToValue
- */
- /**
- * Gets fired when a series is added to the chart after load time, using the
- * `addSeries` method. Returning `false` prevents the series from being added.
- *
- * @callback Highcharts.ChartAddSeriesCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {Highcharts.ChartAddSeriesEventObject} event
- * The event that occured.
- */
- /**
- * Contains common event information. Through the `options` property you can
- * access the series options that were passed to the `addSeries` method.
- *
- * @interface Highcharts.ChartAddSeriesEventObject
- */ /**
- * The series options that were passed to the `addSeries` method.
- * @name Highcharts.ChartAddSeriesEventObject#options
- * @type {Highcharts.SeriesOptionsType}
- */ /**
- * Prevents the default behaviour of the event.
- * @name Highcharts.ChartAddSeriesEventObject#preventDefault
- * @type {Function}
- */ /**
- * The event target.
- * @name Highcharts.ChartAddSeriesEventObject#target
- * @type {Highcharts.Chart}
- */ /**
- * The event type.
- * @name Highcharts.ChartAddSeriesEventObject#type
- * @type {"addSeries"}
- */
- /**
- * Gets fired when clicking on the plot background.
- *
- * @callback Highcharts.ChartClickCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {Highcharts.PointerEventObject} event
- * The event that occured.
- */
- /**
- * Contains an axes of the clicked spot.
- *
- * @interface Highcharts.ChartClickEventAxisObject
- */ /**
- * Axis at the clicked spot.
- * @name Highcharts.ChartClickEventAxisObject#axis
- * @type {Highcharts.Axis}
- */ /**
- * Axis value at the clicked spot.
- * @name Highcharts.ChartClickEventAxisObject#value
- * @type {number}
- */
- /**
- * Contains information about the clicked spot on the chart. Remember the unit
- * of a datetime axis is milliseconds since 1970-01-01 00:00:00.
- *
- * @interface Highcharts.ChartClickEventObject
- * @extends Highcharts.PointerEventObject
- */ /**
- * Information about the x-axis on the clicked spot.
- * @name Highcharts.ChartClickEventObject#xAxis
- * @type {Array<Highcharts.ChartClickEventAxisObject>}
- */ /**
- * Information about the y-axis on the clicked spot.
- * @name Highcharts.ChartClickEventObject#yAxis
- * @type {Array<Highcharts.ChartClickEventAxisObject>}
- */ /**
- * Information about the z-axis on the clicked spot.
- * @name Highcharts.ChartClickEventObject#zAxis
- * @type {Array<Highcharts.ChartClickEventAxisObject>|undefined}
- */
- /**
- * Gets fired when the chart is finished loading.
- *
- * @callback Highcharts.ChartLoadCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {global.Event} event
- * The event that occured.
- */
- /**
- * Fires when the chart is redrawn, either after a call to `chart.redraw()` or
- * after an axis, series or point is modified with the `redraw` option set to
- * `true`.
- *
- * @callback Highcharts.ChartRedrawCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {global.Event} event
- * The event that occured.
- */
- /**
- * Gets fired after initial load of the chart (directly after the `load` event),
- * and after each redraw (directly after the `redraw` event).
- *
- * @callback Highcharts.ChartRenderCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {global.Event} event
- * The event that occured.
- */
- /**
- * Gets fired when an area of the chart has been selected. The default action
- * for the selection event is to zoom the chart to the selected area. It can be
- * prevented by calling `event.preventDefault()` or return false.
- *
- * @callback Highcharts.ChartSelectionCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * The chart on which the event occured.
- *
- * @param {global.ChartSelectionContextObject} event
- * Event informations
- *
- * @return {boolean|undefined}
- * Return false to prevent the default action, usually zoom.
- */
- /**
- * The primary axes are `xAxis[0]` and `yAxis[0]`. Remember the unit of a
- * datetime axis is milliseconds since 1970-01-01 00:00:00.
- *
- * @interface Highcharts.ChartSelectionContextObject
- * @extends global.Event
- */ /**
- * Arrays containing the axes of each dimension and each axis' min and max
- * values.
- * @name Highcharts.ChartSelectionContextObject#xAxis
- * @type {Array<Highcharts.ChartSelectionAxisContextObject>}
- */ /**
- * Arrays containing the axes of each dimension and each axis' min and max
- * values.
- * @name Highcharts.ChartSelectionContextObject#yAxis
- * @type {Array<Highcharts.ChartSelectionAxisContextObject>}
- */
- /**
- * Axis context of the selection.
- *
- * @interface Highcharts.ChartSelectionAxisContextObject
- */ /**
- * The selected Axis.
- * @name Highcharts.ChartSelectionAxisContextObject#axis
- * @type {Highcharts.Axis}
- */ /**
- * The maximum axis value, either automatic or set manually.
- * @name Highcharts.ChartSelectionAxisContextObject#max
- * @type {number}
- */ /**
- * The minimum axis value, either automatic or set manually.
- * @name Highcharts.ChartSelectionAxisContextObject#min
- * @type {number}
- */
- var color = Color.parse;
- var merge = U.merge;
- var isTouchDevice = H.isTouchDevice,
- svg = H.svg;
- /* ************************************************************************** *
- * Handle the options *
- * ************************************************************************** */
- /**
- * Global default settings.
- *
- * @name Highcharts.defaultOptions
- * @type {Highcharts.Options}
- */ /**
- * @optionparent
- */
- H.defaultOptions = {
- /**
- * An array containing the default colors for the chart's series. When
- * all colors are used, new colors are pulled from the start again.
- *
- * Default colors can also be set on a series or series.type basis,
- * see [column.colors](#plotOptions.column.colors),
- * [pie.colors](#plotOptions.pie.colors).
- *
- * In styled mode, the colors option doesn't exist. Instead, colors
- * are defined in CSS and applied either through series or point class
- * names, or through the [chart.colorCount](#chart.colorCount) option.
- *
- *
- * ### Legacy
- *
- * In Highcharts 3.x, the default colors were:
- * ```js
- * colors: ['#2f7ed8', '#0d233a', '#8bbc21', '#910000', '#1aadce',
- * '#492970', '#f28f43', '#77a1e5', '#c42525', '#a6c96a']
- * ```
- *
- * In Highcharts 2.x, the default colors were:
- * ```js
- * colors: ['#4572A7', '#AA4643', '#89A54E', '#80699B', '#3D96AE',
- * '#DB843D', '#92A8CD', '#A47D7C', '#B5CA92']
- * ```
- *
- * @sample {highcharts} highcharts/chart/colors/
- * Assign a global color theme
- *
- * @type {Array<Highcharts.ColorString>}
- * @default ["#7cb5ec", "#434348", "#90ed7d", "#f7a35c", "#8085e9",
- * "#f15c80", "#e4d354", "#2b908f", "#f45b5b", "#91e8e1"]
- */
- colors: '#7cb5ec #434348 #90ed7d #f7a35c #8085e9 #f15c80 #e4d354 #2b908f #f45b5b #91e8e1'.split(' '),
- /**
- * Styled mode only. Configuration object for adding SVG definitions for
- * reusable elements. See [gradients, shadows and
- * patterns](https://www.highcharts.com/docs/chart-design-and-style/gradients-shadows-and-patterns)
- * for more information and code examples.
- *
- * @type {*}
- * @since 5.0.0
- * @apioption defs
- */
- /**
- * @ignore-option
- */
- symbols: ['circle', 'diamond', 'square', 'triangle', 'triangle-down'],
- /**
- * The language object is global and it can't be set on each chart
- * initialization. Instead, use `Highcharts.setOptions` to set it before any
- * chart is initialized.
- *
- * ```js
- * Highcharts.setOptions({
- * lang: {
- * months: [
- * 'Janvier', 'Février', 'Mars', 'Avril',
- * 'Mai', 'Juin', 'Juillet', 'Août',
- * 'Septembre', 'Octobre', 'Novembre', 'Décembre'
- * ],
- * weekdays: [
- * 'Dimanche', 'Lundi', 'Mardi', 'Mercredi',
- * 'Jeudi', 'Vendredi', 'Samedi'
- * ]
- * }
- * });
- * ```
- */
- lang: {
- /**
- * The loading text that appears when the chart is set into the loading
- * state following a call to `chart.showLoading`.
- */
- loading: 'Loading...',
- /**
- * An array containing the months names. Corresponds to the `%B` format
- * in `Highcharts.dateFormat()`.
- *
- * @type {Array<string>}
- * @default ["January", "February", "March", "April", "May", "June",
- * "July", "August", "September", "October", "November",
- * "December"]
- */
- months: [
- 'January', 'February', 'March', 'April', 'May', 'June', 'July',
- 'August', 'September', 'October', 'November', 'December'
- ],
- /**
- * An array containing the months names in abbreviated form. Corresponds
- * to the `%b` format in `Highcharts.dateFormat()`.
- *
- * @type {Array<string>}
- * @default ["Jan", "Feb", "Mar", "Apr", "May", "Jun",
- * "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
- */
- shortMonths: [
- 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul',
- 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'
- ],
- /**
- * An array containing the weekday names.
- *
- * @type {Array<string>}
- * @default ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday",
- * "Friday", "Saturday"]
- */
- weekdays: [
- 'Sunday', 'Monday', 'Tuesday', 'Wednesday',
- 'Thursday', 'Friday', 'Saturday'
- ],
- /**
- * Short week days, starting Sunday. If not specified, Highcharts uses
- * the first three letters of the `lang.weekdays` option.
- *
- * @sample highcharts/lang/shortweekdays/
- * Finnish two-letter abbreviations
- *
- * @type {Array<string>}
- * @since 4.2.4
- * @apioption lang.shortWeekdays
- */
- /**
- * What to show in a date field for invalid dates. Defaults to an empty
- * string.
- *
- * @type {string}
- * @since 4.1.8
- * @product highcharts highstock
- * @apioption lang.invalidDate
- */
- /**
- * The title appearing on hovering the zoom in button. The text itself
- * defaults to "+" and can be changed in the button options.
- *
- * @type {string}
- * @default Zoom in
- * @product highmaps
- * @apioption lang.zoomIn
- */
- /**
- * The title appearing on hovering the zoom out button. The text itself
- * defaults to "-" and can be changed in the button options.
- *
- * @type {string}
- * @default Zoom out
- * @product highmaps
- * @apioption lang.zoomOut
- */
- /**
- * The default decimal point used in the `Highcharts.numberFormat`
- * method unless otherwise specified in the function arguments.
- *
- * @since 1.2.2
- */
- decimalPoint: '.',
- /**
- * [Metric prefixes](https://en.wikipedia.org/wiki/Metric_prefix) used
- * to shorten high numbers in axis labels. Replacing any of the
- * positions with `null` causes the full number to be written. Setting
- * `numericSymbols` to `null` disables shortening altogether.
- *
- * @sample {highcharts} highcharts/lang/numericsymbols/
- * Replacing the symbols with text
- * @sample {highstock} highcharts/lang/numericsymbols/
- * Replacing the symbols with text
- *
- * @type {Array<string>}
- * @default ["k", "M", "G", "T", "P", "E"]
- * @since 2.3.0
- */
- numericSymbols: ['k', 'M', 'G', 'T', 'P', 'E'],
- /**
- * The magnitude of [numericSymbols](#lang.numericSymbol) replacements.
- * Use 10000 for Japanese, Korean and various Chinese locales, which
- * use symbols for 10^4, 10^8 and 10^12.
- *
- * @sample highcharts/lang/numericsymbolmagnitude/
- * 10000 magnitude for Japanese
- *
- * @type {number}
- * @default 1000
- * @since 5.0.3
- * @apioption lang.numericSymbolMagnitude
- */
- /**
- * The text for the label appearing when a chart is zoomed.
- *
- * @since 1.2.4
- */
- resetZoom: 'Reset zoom',
- /**
- * The tooltip title for the label appearing when a chart is zoomed.
- *
- * @since 1.2.4
- */
- resetZoomTitle: 'Reset zoom level 1:1',
- /**
- * The default thousands separator used in the `Highcharts.numberFormat`
- * method unless otherwise specified in the function arguments. Defaults
- * to a single space character, which is recommended in
- * [ISO 31-0](https://en.wikipedia.org/wiki/ISO_31-0#Numbers) and works
- * across Anglo-American and continental European languages.
- *
- * @default \u0020
- * @since 1.2.2
- */
- thousandsSep: ' '
- },
- /**
- * Global options that don't apply to each chart. These options, like
- * the `lang` options, must be set using the `Highcharts.setOptions`
- * method.
- *
- * ```js
- * Highcharts.setOptions({
- * global: {
- * useUTC: false
- * }
- * });
- * ```
- */
- /**
- * _Canvg rendering for Android 2.x is removed as of Highcharts 5.0\.
- * Use the [libURL](#exporting.libURL) option to configure exporting._
- *
- * The URL to the additional file to lazy load for Android 2.x devices.
- * These devices don't support SVG, so we download a helper file that
- * contains [canvg](https://github.com/canvg/canvg), its dependency
- * rbcolor, and our own CanVG Renderer class. To avoid hotlinking to
- * our site, you can install canvas-tools.js on your own server and
- * change this option accordingly.
- *
- * @deprecated
- *
- * @type {string}
- * @default https://code.highcharts.com/{version}/modules/canvas-tools.js
- * @product highcharts highmaps
- * @apioption global.canvasToolsURL
- */
- /**
- * This option is deprecated since v6.0.5. Instead, use
- * [time.useUTC](#time.useUTC) that supports individual time settings
- * per chart.
- *
- * @deprecated
- *
- * @type {boolean}
- * @apioption global.useUTC
- */
- /**
- * This option is deprecated since v6.0.5. Instead, use
- * [time.Date](#time.Date) that supports individual time settings
- * per chart.
- *
- * @deprecated
- *
- * @type {Function}
- * @product highcharts highstock
- * @apioption global.Date
- */
- /**
- * This option is deprecated since v6.0.5. Instead, use
- * [time.getTimezoneOffset](#time.getTimezoneOffset) that supports
- * individual time settings per chart.
- *
- * @deprecated
- *
- * @type {Function}
- * @product highcharts highstock
- * @apioption global.getTimezoneOffset
- */
- /**
- * This option is deprecated since v6.0.5. Instead, use
- * [time.timezone](#time.timezone) that supports individual time
- * settings per chart.
- *
- * @deprecated
- *
- * @type {string}
- * @product highcharts highstock
- * @apioption global.timezone
- */
- /**
- * This option is deprecated since v6.0.5. Instead, use
- * [time.timezoneOffset](#time.timezoneOffset) that supports individual
- * time settings per chart.
- *
- * @deprecated
- *
- * @type {number}
- * @product highcharts highstock
- * @apioption global.timezoneOffset
- */
- global: {},
- /**
- * Time options that can apply globally or to individual charts. These
- * settings affect how `datetime` axes are laid out, how tooltips are
- * formatted, how series
- * [pointIntervalUnit](#plotOptions.series.pointIntervalUnit) works and how
- * the Highstock range selector handles time.
- *
- * The common use case is that all charts in the same Highcharts object
- * share the same time settings, in which case the global settings are set
- * using `setOptions`.
- *
- * ```js
- * // Apply time settings globally
- * Highcharts.setOptions({
- * time: {
- * timezone: 'Europe/London'
- * }
- * });
- * // Apply time settings by instance
- * var chart = Highcharts.chart('container', {
- * time: {
- * timezone: 'America/New_York'
- * },
- * series: [{
- * data: [1, 4, 3, 5]
- * }]
- * });
- *
- * // Use the Time object
- * console.log(
- * 'Current time in New York',
- * chart.time.dateFormat('%Y-%m-%d %H:%M:%S', Date.now())
- * );
- * ```
- *
- * Since v6.0.5, the time options were moved from the `global` obect to the
- * `time` object, and time options can be set on each individual chart.
- *
- * @sample {highcharts|highstock}
- * highcharts/time/timezone/
- * Set the timezone globally
- * @sample {highcharts}
- * highcharts/time/individual/
- * Set the timezone per chart instance
- * @sample {highstock}
- * stock/time/individual/
- * Set the timezone per chart instance
- *
- * @since 6.0.5
- * @optionparent time
- */
- time: {
- /**
- * A custom `Date` class for advanced date handling. For example,
- * [JDate](https://github.com/tahajahangir/jdate) can be hooked in to
- * handle Jalali dates.
- *
- * @type {*}
- * @since 4.0.4
- * @product highcharts highstock gantt
- */
- Date: void 0,
- /**
- * A callback to return the time zone offset for a given datetime. It
- * takes the timestamp in terms of milliseconds since January 1 1970,
- * and returns the timezone offset in minutes. This provides a hook
- * for drawing time based charts in specific time zones using their
- * local DST crossover dates, with the help of external libraries.
- *
- * @see [global.timezoneOffset](#global.timezoneOffset)
- *
- * @sample {highcharts|highstock} highcharts/time/gettimezoneoffset/
- * Use moment.js to draw Oslo time regardless of browser locale
- *
- * @type {Highcharts.TimezoneOffsetCallbackFunction}
- * @since 4.1.0
- * @product highcharts highstock gantt
- */
- getTimezoneOffset: void 0,
- /**
- * Requires [moment.js](https://momentjs.com/). If the timezone option
- * is specified, it creates a default
- * [getTimezoneOffset](#time.getTimezoneOffset) function that looks
- * up the specified timezone in moment.js. If moment.js is not included,
- * this throws a Highcharts error in the console, but does not crash the
- * chart.
- *
- * @see [getTimezoneOffset](#time.getTimezoneOffset)
- *
- * @sample {highcharts|highstock} highcharts/time/timezone/
- * Europe/Oslo
- *
- * @type {string}
- * @since 5.0.7
- * @product highcharts highstock gantt
- */
- timezone: void 0,
- /**
- * The timezone offset in minutes. Positive values are west, negative
- * values are east of UTC, as in the ECMAScript
- * [getTimezoneOffset](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/getTimezoneOffset)
- * method. Use this to display UTC based data in a predefined time zone.
- *
- * @see [time.getTimezoneOffset](#time.getTimezoneOffset)
- *
- * @sample {highcharts|highstock} highcharts/time/timezoneoffset/
- * Timezone offset
- *
- * @since 3.0.8
- * @product highcharts highstock gantt
- */
- timezoneOffset: 0,
- /**
- * Whether to use UTC time for axis scaling, tickmark placement and
- * time display in `Highcharts.dateFormat`. Advantages of using UTC
- * is that the time displays equally regardless of the user agent's
- * time zone settings. Local time can be used when the data is loaded
- * in real time or when correct Daylight Saving Time transitions are
- * required.
- *
- * @sample {highcharts} highcharts/time/useutc-true/
- * True by default
- * @sample {highcharts} highcharts/time/useutc-false/
- * False
- */
- useUTC: true
- },
- /**
- * General options for the chart.
- */
- chart: {
- /**
- * Default `mapData` for all series. If set to a string, it functions
- * as an index into the `Highcharts.maps` array. Otherwise it is
- * interpreted as map data.
- *
- * @see [mapData](#series.map.mapData)
- *
- * @sample maps/demo/geojson
- * Loading geoJSON data
- * @sample maps/chart/topojson
- * Loading topoJSON converted to geoJSON
- *
- * @type {string|Array<*>|Highcharts.GeoJSON}
- * @since 5.0.0
- * @product highmaps
- * @apioption chart.map
- */
- /**
- * Set lat/lon transformation definitions for the chart. If not defined,
- * these are extracted from the map data.
- *
- * @type {*}
- * @since 5.0.0
- * @product highmaps
- * @apioption chart.mapTransforms
- */
- /**
- * When using multiple axis, the ticks of two or more opposite axes
- * will automatically be aligned by adding ticks to the axis or axes
- * with the least ticks, as if `tickAmount` were specified.
- *
- * This can be prevented by setting `alignTicks` to false. If the grid
- * lines look messy, it's a good idea to hide them for the secondary
- * axis by setting `gridLineWidth` to 0.
- *
- * If `startOnTick` or `endOnTick` in an Axis options are set to false,
- * then the `alignTicks ` will be disabled for the Axis.
- *
- * Disabled for logarithmic axes.
- *
- * @sample {highcharts} highcharts/chart/alignticks-true/
- * True by default
- * @sample {highcharts} highcharts/chart/alignticks-false/
- * False
- * @sample {highstock} stock/chart/alignticks-true/
- * True by default
- * @sample {highstock} stock/chart/alignticks-false/
- * False
- *
- * @type {boolean}
- * @default true
- * @product highcharts highstock gantt
- * @apioption chart.alignTicks
- */
- /**
- * Set the overall animation for all chart updating. Animation can be
- * disabled throughout the chart by setting it to false here. It can
- * be overridden for each individual API method as a function parameter.
- * The only animation not affected by this option is the initial series
- * animation, see [plotOptions.series.animation](
- * #plotOptions.series.animation).
- *
- * The animation can either be set as a boolean or a configuration
- * object. If `true`, it will use the 'swing' jQuery easing and a
- * duration of 500 ms. If used as a configuration object, the following
- * properties are supported:
- *
- * - `defer`: The animation delay time in milliseconds.
- *
- * - `duration`: The duration of the animation in milliseconds.
- *
- * - `easing`: A string reference to an easing function set on the
- * `Math` object. See
- * [the easing demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-animation-easing/).
- *
- * When zooming on a series with less than 100 points, the chart redraw
- * will be done with animation, but in case of more data points, it is
- * necessary to set this option to ensure animation on zoom.
- *
- * @sample {highcharts} highcharts/chart/animation-none/
- * Updating with no animation
- * @sample {highcharts} highcharts/chart/animation-duration/
- * With a longer duration
- * @sample {highcharts} highcharts/chart/animation-easing/
- * With a jQuery UI easing
- * @sample {highmaps} maps/chart/animation-none/
- * Updating with no animation
- * @sample {highmaps} maps/chart/animation-duration/
- * With a longer duration
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @default undefined
- * @apioption chart.animation
- */
- /**
- * A CSS class name to apply to the charts container `div`, allowing
- * unique CSS styling for each chart.
- *
- * @type {string}
- * @apioption chart.className
- */
- /**
- * Event listeners for the chart.
- *
- * @apioption chart.events
- */
- /**
- * Fires when a series is added to the chart after load time, using the
- * `addSeries` method. One parameter, `event`, is passed to the
- * function, containing common event information. Through
- * `event.options` you can access the series options that were passed to
- * the `addSeries` method. Returning false prevents the series from
- * being added.
- *
- * @sample {highcharts} highcharts/chart/events-addseries/
- * Alert on add series
- * @sample {highstock} stock/chart/events-addseries/
- * Alert on add series
- *
- * @type {Highcharts.ChartAddSeriesCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Chart
- * @apioption chart.events.addSeries
- */
- /**
- * Fires when clicking on the plot background. One parameter, `event`,
- * is passed to the function, containing common event information.
- *
- * Information on the clicked spot can be found through `event.xAxis`
- * and `event.yAxis`, which are arrays containing the axes of each
- * dimension and each axis' value at the clicked spot. The primary axes
- * are `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a
- * datetime axis is milliseconds since 1970-01-01 00:00:00.
- *
- * ```js
- * click: function(e) {
- * console.log(
- * Highcharts.dateFormat('%Y-%m-%d %H:%M:%S', e.xAxis[0].value),
- * e.yAxis[0].value
- * )
- * }
- * ```
- *
- * @sample {highcharts} highcharts/chart/events-click/
- * Alert coordinates on click
- * @sample {highcharts} highcharts/chart/events-container/
- * Alternatively, attach event to container
- * @sample {highstock} stock/chart/events-click/
- * Alert coordinates on click
- * @sample {highstock} highcharts/chart/events-container/
- * Alternatively, attach event to container
- * @sample {highmaps} maps/chart/events-click/
- * Record coordinates on click
- * @sample {highmaps} highcharts/chart/events-container/
- * Alternatively, attach event to container
- *
- * @type {Highcharts.ChartClickCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Chart
- * @apioption chart.events.click
- */
- /**
- * Fires when the chart is finished loading. Since v4.2.2, it also waits
- * for images to be loaded, for example from point markers. One
- * parameter, `event`, is passed to the function, containing common
- * event information.
- *
- * There is also a second parameter to the chart constructor where a
- * callback function can be passed to be executed on chart.load.
- *
- * @sample {highcharts} highcharts/chart/events-load/
- * Alert on chart load
- * @sample {highstock} stock/chart/events-load/
- * Alert on chart load
- * @sample {highmaps} maps/chart/events-load/
- * Add series on chart load
- *
- * @type {Highcharts.ChartLoadCallbackFunction}
- * @context Highcharts.Chart
- * @apioption chart.events.load
- */
- /**
- * Fires when the chart is redrawn, either after a call to
- * `chart.redraw()` or after an axis, series or point is modified with
- * the `redraw` option set to `true`. One parameter, `event`, is passed
- * to the function, containing common event information.
- *
- * @sample {highcharts} highcharts/chart/events-redraw/
- * Alert on chart redraw
- * @sample {highstock} stock/chart/events-redraw/
- * Alert on chart redraw when adding a series or moving the
- * zoomed range
- * @sample {highmaps} maps/chart/events-redraw/
- * Set subtitle on chart redraw
- *
- * @type {Highcharts.ChartRedrawCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Chart
- * @apioption chart.events.redraw
- */
- /**
- * Fires after initial load of the chart (directly after the `load`
- * event), and after each redraw (directly after the `redraw` event).
- *
- * @type {Highcharts.ChartRenderCallbackFunction}
- * @since 5.0.7
- * @context Highcharts.Chart
- * @apioption chart.events.render
- */
- /**
- * Fires when an area of the chart has been selected. Selection is
- * enabled by setting the chart's zoomType. One parameter, `event`, is
- * passed to the function, containing common event information. The
- * default action for the selection event is to zoom the chart to the
- * selected area. It can be prevented by calling
- * `event.preventDefault()` or return false.
- *
- * Information on the selected area can be found through `event.xAxis`
- * and `event.yAxis`, which are arrays containing the axes of each
- * dimension and each axis' min and max values. The primary axes are
- * `event.xAxis[0]` and `event.yAxis[0]`. Remember the unit of a
- * datetime axis is milliseconds since 1970-01-01 00:00:00.
- *
- * ```js
- * selection: function(event) {
- * // log the min and max of the primary, datetime x-axis
- * console.log(
- * Highcharts.dateFormat(
- * '%Y-%m-%d %H:%M:%S',
- * event.xAxis[0].min
- * ),
- * Highcharts.dateFormat(
- * '%Y-%m-%d %H:%M:%S',
- * event.xAxis[0].max
- * )
- * );
- * // log the min and max of the y axis
- * console.log(event.yAxis[0].min, event.yAxis[0].max);
- * }
- * ```
- *
- * @sample {highcharts} highcharts/chart/events-selection/
- * Report on selection and reset
- * @sample {highcharts} highcharts/chart/events-selection-points/
- * Select a range of points through a drag selection
- * @sample {highstock} stock/chart/events-selection/
- * Report on selection and reset
- * @sample {highstock} highcharts/chart/events-selection-points/
- * Select a range of points through a drag selection
- * (Highcharts)
- *
- * @type {Highcharts.ChartSelectionCallbackFunction}
- * @apioption chart.events.selection
- */
- /**
- * The margin between the outer edge of the chart and the plot area.
- * The numbers in the array designate top, right, bottom and left
- * respectively. Use the options `marginTop`, `marginRight`,
- * `marginBottom` and `marginLeft` for shorthand setting of one option.
- *
- * By default there is no margin. The actual space is dynamically
- * calculated from the offset of axis labels, axis title, title,
- * subtitle and legend in addition to the `spacingTop`, `spacingRight`,
- * `spacingBottom` and `spacingLeft` options.
- *
- * @sample {highcharts} highcharts/chart/margins-zero/
- * Zero margins
- * @sample {highstock} stock/chart/margin-zero/
- * Zero margins
- *
- * @type {number|Array<number>}
- * @apioption chart.margin
- */
- /**
- * The margin between the bottom outer edge of the chart and the plot
- * area. Use this to set a fixed pixel value for the margin as opposed
- * to the default dynamic margin. See also `spacingBottom`.
- *
- * @sample {highcharts} highcharts/chart/marginbottom/
- * 100px bottom margin
- * @sample {highstock} stock/chart/marginbottom/
- * 100px bottom margin
- * @sample {highmaps} maps/chart/margin/
- * 100px margins
- *
- * @type {number}
- * @since 2.0
- * @apioption chart.marginBottom
- */
- /**
- * The margin between the left outer edge of the chart and the plot
- * area. Use this to set a fixed pixel value for the margin as opposed
- * to the default dynamic margin. See also `spacingLeft`.
- *
- * @sample {highcharts} highcharts/chart/marginleft/
- * 150px left margin
- * @sample {highstock} stock/chart/marginleft/
- * 150px left margin
- * @sample {highmaps} maps/chart/margin/
- * 100px margins
- *
- * @type {number}
- * @since 2.0
- * @apioption chart.marginLeft
- */
- /**
- * The margin between the right outer edge of the chart and the plot
- * area. Use this to set a fixed pixel value for the margin as opposed
- * to the default dynamic margin. See also `spacingRight`.
- *
- * @sample {highcharts} highcharts/chart/marginright/
- * 100px right margin
- * @sample {highstock} stock/chart/marginright/
- * 100px right margin
- * @sample {highmaps} maps/chart/margin/
- * 100px margins
- *
- * @type {number}
- * @since 2.0
- * @apioption chart.marginRight
- */
- /**
- * The margin between the top outer edge of the chart and the plot area.
- * Use this to set a fixed pixel value for the margin as opposed to
- * the default dynamic margin. See also `spacingTop`.
- *
- * @sample {highcharts} highcharts/chart/margintop/ 100px top margin
- * @sample {highstock} stock/chart/margintop/
- * 100px top margin
- * @sample {highmaps} maps/chart/margin/
- * 100px margins
- *
- * @type {number}
- * @since 2.0
- * @apioption chart.marginTop
- */
- /**
- * Callback function to override the default function that formats all
- * the numbers in the chart. Returns a string with the formatted number.
- *
- * @sample highcharts/members/highcharts-numberformat
- * Arabic digits in Highcharts
- * @type {Highcharts.NumberFormatterCallbackFunction}
- * @since 8.0.0
- * @apioption chart.numberFormatter
- */
- /**
- * Allows setting a key to switch between zooming and panning. Can be
- * one of `alt`, `ctrl`, `meta` (the command key on Mac and Windows
- * key on Windows) or `shift`. The keys are mapped directly to the key
- * properties of the click event argument (`event.altKey`,
- * `event.ctrlKey`, `event.metaKey` and `event.shiftKey`).
- *
- * @type {string}
- * @since 4.0.3
- * @product highcharts gantt
- * @validvalue ["alt", "ctrl", "meta", "shift"]
- * @apioption chart.panKey
- */
- /**
- * Allow panning in a chart. Best used with [panKey](#chart.panKey)
- * to combine zooming and panning.
- *
- * On touch devices, when the [tooltip.followTouchMove](
- * #tooltip.followTouchMove) option is `true` (default), panning
- * requires two fingers. To allow panning with one finger, set
- * `followTouchMove` to `false`.
- *
- * @sample {highcharts} highcharts/chart/pankey/ Zooming and panning
- * @sample {highstock} stock/chart/panning/ Zooming and xy panning
- *
- * @product highcharts highstock gantt
- * @apioption chart.panning
- */
- /**
- * Enable or disable chart panning.
- *
- * @type {boolean}
- * @default {highcharts} false
- * @default {highstock} true
- * @apioption chart.panning.enabled
- */
- /**
- * Decides in what dimensions the user can pan the chart. Can be
- * one of `x`, `y`, or `xy`.
- *
- * @sample {highcharts} highcharts/chart/panning-type
- * Zooming and xy panning
- *
- * @type {string}
- * @validvalue ["x", "y", "xy"]
- * @default x
- * @apioption chart.panning.type
- */
- /**
- * Equivalent to [zoomType](#chart.zoomType), but for multitouch
- * gestures only. By default, the `pinchType` is the same as the
- * `zoomType` setting. However, pinching can be enabled separately in
- * some cases, for example in stock charts where a mouse drag pans the
- * chart, while pinching is enabled. When [tooltip.followTouchMove](
- * #tooltip.followTouchMove) is true, pinchType only applies to
- * two-finger touches.
- *
- * @type {string}
- * @default {highcharts} undefined
- * @default {highstock} x
- * @since 3.0
- * @product highcharts highstock gantt
- * @validvalue ["x", "y", "xy"]
- * @apioption chart.pinchType
- */
- /**
- * Whether to apply styled mode. When in styled mode, no presentational
- * attributes or CSS are applied to the chart SVG. Instead, CSS rules
- * are required to style the chart. The default style sheet is
- * available from `https://code.highcharts.com/css/highcharts.css`.
- *
- * @type {boolean}
- * @default false
- * @since 7.0
- * @apioption chart.styledMode
- */
- styledMode: false,
- /**
- * The corner radius of the outer chart border.
- *
- * @sample {highcharts} highcharts/chart/borderradius/
- * 20px radius
- * @sample {highstock} stock/chart/border/
- * 10px radius
- * @sample {highmaps} maps/chart/border/
- * Border options
- *
- */
- borderRadius: 0,
- /**
- * In styled mode, this sets how many colors the class names
- * should rotate between. With ten colors, series (or points) are
- * given class names like `highcharts-color-0`, `highcharts-color-0`
- * [...] `highcharts-color-9`. The equivalent in non-styled mode
- * is to set colors using the [colors](#colors) setting.
- *
- * @since 5.0.0
- */
- colorCount: 10,
- /**
- * Alias of `type`.
- *
- * @sample {highcharts} highcharts/chart/defaultseriestype/
- * Bar
- *
- * @deprecated
- *
- * @product highcharts
- */
- defaultSeriesType: 'line',
- /**
- * If true, the axes will scale to the remaining visible series once
- * one series is hidden. If false, hiding and showing a series will
- * not affect the axes or the other series. For stacks, once one series
- * within the stack is hidden, the rest of the stack will close in
- * around it even if the axis is not affected.
- *
- * @sample {highcharts} highcharts/chart/ignorehiddenseries-true/
- * True by default
- * @sample {highcharts} highcharts/chart/ignorehiddenseries-false/
- * False
- * @sample {highcharts} highcharts/chart/ignorehiddenseries-true-stacked/
- * True with stack
- * @sample {highstock} stock/chart/ignorehiddenseries-true/
- * True by default
- * @sample {highstock} stock/chart/ignorehiddenseries-false/
- * False
- *
- * @since 1.2.0
- * @product highcharts highstock gantt
- */
- ignoreHiddenSeries: true,
- /**
- * Whether to invert the axes so that the x axis is vertical and y axis
- * is horizontal. When `true`, the x axis is [reversed](#xAxis.reversed)
- * by default.
- *
- * @productdesc {highcharts}
- * If a bar series is present in the chart, it will be inverted
- * automatically. Inverting the chart doesn't have an effect if there
- * are no cartesian series in the chart, or if the chart is
- * [polar](#chart.polar).
- *
- * @sample {highcharts} highcharts/chart/inverted/
- * Inverted line
- * @sample {highstock} stock/navigator/inverted/
- * Inverted stock chart
- *
- * @type {boolean}
- * @default false
- * @product highcharts highstock gantt
- * @apioption chart.inverted
- */
- /**
- * The distance between the outer edge of the chart and the content,
- * like title or legend, or axis title and labels if present. The
- * numbers in the array designate top, right, bottom and left
- * respectively. Use the options spacingTop, spacingRight, spacingBottom
- * and spacingLeft options for shorthand setting of one option.
- *
- * @type {Array<number>}
- * @see [chart.margin](#chart.margin)
- * @default [10, 10, 15, 10]
- * @since 3.0.6
- */
- spacing: [10, 10, 15, 10],
- /**
- * The button that appears after a selection zoom, allowing the user
- * to reset zoom.
- */
- resetZoomButton: {
- /**
- * What frame the button placement should be related to. Can be
- * either `plotBox` or `spacingBox`.
- *
- * @sample {highcharts} highcharts/chart/resetzoombutton-relativeto/
- * Relative to the chart
- * @sample {highstock} highcharts/chart/resetzoombutton-relativeto/
- * Relative to the chart
- *
- * @type {Highcharts.ButtonRelativeToValue}
- * @default plot
- * @since 2.2
- * @apioption chart.resetZoomButton.relativeTo
- */
- /**
- * A collection of attributes for the button. The object takes SVG
- * attributes like `fill`, `stroke`, `stroke-width` or `r`, the
- * border radius. The theme also supports `style`, a collection of
- * CSS properties for the text. Equivalent attributes for the hover
- * state are given in `theme.states.hover`.
- *
- * @sample {highcharts} highcharts/chart/resetzoombutton-theme/
- * Theming the button
- * @sample {highstock} highcharts/chart/resetzoombutton-theme/
- * Theming the button
- *
- * @type {Highcharts.SVGAttributes}
- * @since 2.2
- */
- theme: {
- /** @internal */
- zIndex: 6
- },
- /**
- * The position of the button.
- *
- * @sample {highcharts} highcharts/chart/resetzoombutton-position/
- * Above the plot area
- * @sample {highstock} highcharts/chart/resetzoombutton-position/
- * Above the plot area
- * @sample {highmaps} highcharts/chart/resetzoombutton-position/
- * Above the plot area
- *
- * @type {Highcharts.AlignObject}
- * @since 2.2
- */
- position: {
- /**
- * The horizontal alignment of the button.
- */
- align: 'right',
- /**
- * The horizontal offset of the button.
- */
- x: -10,
- /**
- * The vertical alignment of the button.
- *
- * @type {Highcharts.VerticalAlignValue}
- * @default top
- * @apioption chart.resetZoomButton.position.verticalAlign
- */
- /**
- * The vertical offset of the button.
- */
- y: 10
- }
- },
- /**
- * The pixel width of the plot area border.
- *
- * @sample {highcharts} highcharts/chart/plotborderwidth/
- * 1px border
- * @sample {highstock} stock/chart/plotborder/
- * 2px border
- * @sample {highmaps} maps/chart/plotborder/
- * Plot border options
- *
- * @type {number}
- * @default 0
- * @apioption chart.plotBorderWidth
- */
- /**
- * Whether to apply a drop shadow to the plot area. Requires that
- * plotBackgroundColor be set. The shadow can be an object configuration
- * containing `color`, `offsetX`, `offsetY`, `opacity` and `width`.
- *
- * @sample {highcharts} highcharts/chart/plotshadow/
- * Plot shadow
- * @sample {highstock} stock/chart/plotshadow/
- * Plot shadow
- * @sample {highmaps} maps/chart/plotborder/
- * Plot border options
- *
- * @type {boolean|Highcharts.CSSObject}
- * @default false
- * @apioption chart.plotShadow
- */
- /**
- * When true, cartesian charts like line, spline, area and column are
- * transformed into the polar coordinate system. This produces _polar
- * charts_, also known as _radar charts_.
- *
- * @sample {highcharts} highcharts/demo/polar/
- * Polar chart
- * @sample {highcharts} highcharts/demo/polar-wind-rose/
- * Wind rose, stacked polar column chart
- * @sample {highcharts} highcharts/demo/polar-spider/
- * Spider web chart
- * @sample {highcharts} highcharts/parallel-coordinates/polar/
- * Star plot, multivariate data in a polar chart
- *
- * @type {boolean}
- * @default false
- * @since 2.3.0
- * @product highcharts
- * @requires highcharts-more
- * @apioption chart.polar
- */
- /**
- * Whether to reflow the chart to fit the width of the container div
- * on resizing the window.
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * True by default
- * @sample {highcharts} highcharts/chart/reflow-false/
- * False
- * @sample {highstock} stock/chart/reflow-true/
- * True by default
- * @sample {highstock} stock/chart/reflow-false/
- * False
- * @sample {highmaps} maps/chart/reflow-true/
- * True by default
- * @sample {highmaps} maps/chart/reflow-false/
- * False
- *
- * @type {boolean}
- * @default true
- * @since 2.1
- * @apioption chart.reflow
- */
- /**
- * The HTML element where the chart will be rendered. If it is a string,
- * the element by that id is used. The HTML element can also be passed
- * by direct reference, or as the first argument of the chart
- * constructor, in which case the option is not needed.
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * String
- * @sample {highcharts} highcharts/chart/renderto-object/
- * Object reference
- * @sample {highcharts} highcharts/chart/renderto-jquery/
- * Object reference through jQuery
- * @sample {highstock} stock/chart/renderto-string/
- * String
- * @sample {highstock} stock/chart/renderto-object/
- * Object reference
- * @sample {highstock} stock/chart/renderto-jquery/
- * Object reference through jQuery
- *
- * @type {string|Highcharts.HTMLDOMElement}
- * @apioption chart.renderTo
- */
- /**
- * The background color of the marker square when selecting (zooming
- * in on) an area of the chart.
- *
- * @see In styled mode, the selection marker fill is set with the
- * `.highcharts-selection-marker` class.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default rgba(51,92,173,0.25)
- * @since 2.1.7
- * @apioption chart.selectionMarkerFill
- */
- /**
- * Whether to apply a drop shadow to the outer chart area. Requires
- * that backgroundColor be set. The shadow can be an object
- * configuration containing `color`, `offsetX`, `offsetY`, `opacity` and
- * `width`.
- *
- * @sample {highcharts} highcharts/chart/shadow/
- * Shadow
- * @sample {highstock} stock/chart/shadow/
- * Shadow
- * @sample {highmaps} maps/chart/border/
- * Chart border and shadow
- *
- * @type {boolean|Highcharts.CSSObject}
- * @default false
- * @apioption chart.shadow
- */
- /**
- * Whether to show the axes initially. This only applies to empty charts
- * where series are added dynamically, as axes are automatically added
- * to cartesian series.
- *
- * @sample {highcharts} highcharts/chart/showaxes-false/
- * False by default
- * @sample {highcharts} highcharts/chart/showaxes-true/
- * True
- *
- * @type {boolean}
- * @since 1.2.5
- * @product highcharts gantt
- * @apioption chart.showAxes
- */
- /**
- * The space between the bottom edge of the chart and the content (plot
- * area, axis title and labels, title, subtitle or legend in top
- * position).
- *
- * @sample {highcharts} highcharts/chart/spacingbottom/
- * Spacing bottom set to 100
- * @sample {highstock} stock/chart/spacingbottom/
- * Spacing bottom set to 100
- * @sample {highmaps} maps/chart/spacing/
- * Spacing 100 all around
- *
- * @type {number}
- * @default 15
- * @since 2.1
- * @apioption chart.spacingBottom
- */
- /**
- * The space between the left edge of the chart and the content (plot
- * area, axis title and labels, title, subtitle or legend in top
- * position).
- *
- * @sample {highcharts} highcharts/chart/spacingleft/
- * Spacing left set to 100
- * @sample {highstock} stock/chart/spacingleft/
- * Spacing left set to 100
- * @sample {highmaps} maps/chart/spacing/
- * Spacing 100 all around
- *
- * @type {number}
- * @default 10
- * @since 2.1
- * @apioption chart.spacingLeft
- */
- /**
- * The space between the right edge of the chart and the content (plot
- * area, axis title and labels, title, subtitle or legend in top
- * position).
- *
- * @sample {highcharts} highcharts/chart/spacingright-100/
- * Spacing set to 100
- * @sample {highcharts} highcharts/chart/spacingright-legend/
- * Legend in right position with default spacing
- * @sample {highstock} stock/chart/spacingright/
- * Spacing set to 100
- * @sample {highmaps} maps/chart/spacing/
- * Spacing 100 all around
- *
- * @type {number}
- * @default 10
- * @since 2.1
- * @apioption chart.spacingRight
- */
- /**
- * The space between the top edge of the chart and the content (plot
- * area, axis title and labels, title, subtitle or legend in top
- * position).
- *
- * @sample {highcharts} highcharts/chart/spacingtop-100/
- * A top spacing of 100
- * @sample {highcharts} highcharts/chart/spacingtop-10/
- * Floating chart title makes the plot area align to the default
- * spacingTop of 10.
- * @sample {highstock} stock/chart/spacingtop/
- * A top spacing of 100
- * @sample {highmaps} maps/chart/spacing/
- * Spacing 100 all around
- *
- * @type {number}
- * @default 10
- * @since 2.1
- * @apioption chart.spacingTop
- */
- /**
- * Additional CSS styles to apply inline to the container `div`. Note
- * that since the default font styles are applied in the renderer, it
- * is ignorant of the individual chart options and must be set globally.
- *
- * @see In styled mode, general chart styles can be set with the
- * `.highcharts-root` class.
- * @sample {highcharts} highcharts/chart/style-serif-font/
- * Using a serif type font
- * @sample {highcharts} highcharts/css/em/
- * Styled mode with relative font sizes
- * @sample {highstock} stock/chart/style/
- * Using a serif type font
- * @sample {highmaps} maps/chart/style-serif-font/
- * Using a serif type font
- *
- * @type {Highcharts.CSSObject}
- * @default {"fontFamily": "\"Lucida Grande\", \"Lucida Sans Unicode\", Verdana, Arial, Helvetica, sans-serif","fontSize":"12px"}
- * @apioption chart.style
- */
- /**
- * The default series type for the chart. Can be any of the chart types
- * listed under [plotOptions](#plotOptions) and [series](#series) or can
- * be a series provided by an additional module.
- *
- * In TypeScript this option has no effect in sense of typing and
- * instead the `type` option must always be set in the series.
- *
- * @sample {highcharts} highcharts/chart/type-bar/
- * Bar
- * @sample {highstock} stock/chart/type/
- * Areaspline
- * @sample {highmaps} maps/chart/type-mapline/
- * Mapline
- *
- * @type {string}
- * @default {highcharts} line
- * @default {highstock} line
- * @default {highmaps} map
- * @since 2.1.0
- * @apioption chart.type
- */
- /**
- * Decides in what dimensions the user can zoom by dragging the mouse.
- * Can be one of `x`, `y` or `xy`.
- *
- * @see [panKey](#chart.panKey)
- *
- * @sample {highcharts} highcharts/chart/zoomtype-none/
- * None by default
- * @sample {highcharts} highcharts/chart/zoomtype-x/
- * X
- * @sample {highcharts} highcharts/chart/zoomtype-y/
- * Y
- * @sample {highcharts} highcharts/chart/zoomtype-xy/
- * Xy
- * @sample {highstock} stock/demo/basic-line/
- * None by default
- * @sample {highstock} stock/chart/zoomtype-x/
- * X
- * @sample {highstock} stock/chart/zoomtype-y/
- * Y
- * @sample {highstock} stock/chart/zoomtype-xy/
- * Xy
- *
- * @type {string}
- * @product highcharts highstock gantt
- * @validvalue ["x", "y", "xy"]
- * @apioption chart.zoomType
- */
- /**
- * An explicit width for the chart. By default (when `null`) the width
- * is calculated from the offset width of the containing element.
- *
- * @sample {highcharts} highcharts/chart/width/
- * 800px wide
- * @sample {highstock} stock/chart/width/
- * 800px wide
- * @sample {highmaps} maps/chart/size/
- * Chart with explicit size
- *
- * @type {null|number|string}
- */
- width: null,
- /**
- * An explicit height for the chart. If a _number_, the height is
- * given in pixels. If given a _percentage string_ (for example
- * `'56%'`), the height is given as the percentage of the actual chart
- * width. This allows for preserving the aspect ratio across responsive
- * sizes.
- *
- * By default (when `null`) the height is calculated from the offset
- * height of the containing element, or 400 pixels if the containing
- * element's height is 0.
- *
- * @sample {highcharts} highcharts/chart/height/
- * 500px height
- * @sample {highstock} stock/chart/height/
- * 300px height
- * @sample {highmaps} maps/chart/size/
- * Chart with explicit size
- * @sample highcharts/chart/height-percent/
- * Highcharts with percentage height
- *
- * @type {null|number|string}
- */
- height: null,
- /**
- * The color of the outer chart border.
- *
- * @see In styled mode, the stroke is set with the
- * `.highcharts-background` class.
- *
- * @sample {highcharts} highcharts/chart/bordercolor/
- * Brown border
- * @sample {highstock} stock/chart/border/
- * Brown border
- * @sample {highmaps} maps/chart/border/
- * Border options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- borderColor: '#335cad',
- /**
- * The pixel width of the outer chart border.
- *
- * @see In styled mode, the stroke is set with the
- * `.highcharts-background` class.
- *
- * @sample {highcharts} highcharts/chart/borderwidth/
- * 5px border
- * @sample {highstock} stock/chart/border/
- * 2px border
- * @sample {highmaps} maps/chart/border/
- * Border options
- *
- * @type {number}
- * @default 0
- * @apioption chart.borderWidth
- */
- /**
- * The background color or gradient for the outer chart area.
- *
- * @see In styled mode, the background is set with the
- * `.highcharts-background` class.
- *
- * @sample {highcharts} highcharts/chart/backgroundcolor-color/
- * Color
- * @sample {highcharts} highcharts/chart/backgroundcolor-gradient/
- * Gradient
- * @sample {highstock} stock/chart/backgroundcolor-color/
- * Color
- * @sample {highstock} stock/chart/backgroundcolor-gradient/
- * Gradient
- * @sample {highmaps} maps/chart/backgroundcolor-color/
- * Color
- * @sample {highmaps} maps/chart/backgroundcolor-gradient/
- * Gradient
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- backgroundColor: '#ffffff',
- /**
- * The background color or gradient for the plot area.
- *
- * @see In styled mode, the plot background is set with the
- * `.highcharts-plot-background` class.
- *
- * @sample {highcharts} highcharts/chart/plotbackgroundcolor-color/
- * Color
- * @sample {highcharts} highcharts/chart/plotbackgroundcolor-gradient/
- * Gradient
- * @sample {highstock} stock/chart/plotbackgroundcolor-color/
- * Color
- * @sample {highstock} stock/chart/plotbackgroundcolor-gradient/
- * Gradient
- * @sample {highmaps} maps/chart/plotbackgroundcolor-color/
- * Color
- * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/
- * Gradient
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption chart.plotBackgroundColor
- */
- /**
- * The URL for an image to use as the plot background. To set an image
- * as the background for the entire chart, set a CSS background image
- * to the container element. Note that for the image to be applied to
- * exported charts, its URL needs to be accessible by the export server.
- *
- * @see In styled mode, a plot background image can be set with the
- * `.highcharts-plot-background` class and a [custom pattern](
- * https://www.highcharts.com/docs/chart-design-and-style/
- * gradients-shadows-and-patterns).
- *
- * @sample {highcharts} highcharts/chart/plotbackgroundimage/
- * Skies
- * @sample {highstock} stock/chart/plotbackgroundimage/
- * Skies
- *
- * @type {string}
- * @apioption chart.plotBackgroundImage
- */
- /**
- * The color of the inner chart or plot area border.
- *
- * @see In styled mode, a plot border stroke can be set with the
- * `.highcharts-plot-border` class.
- *
- * @sample {highcharts} highcharts/chart/plotbordercolor/
- * Blue border
- * @sample {highstock} stock/chart/plotborder/
- * Blue border
- * @sample {highmaps} maps/chart/plotborder/
- * Plot border options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- plotBorderColor: '#cccccc'
- },
- /**
- * The chart's main title.
- *
- * @sample {highmaps} maps/title/title/
- * Title options demonstrated
- */
- title: {
- /**
- * When the title is floating, the plot area will not move to make space
- * for it.
- *
- * @sample {highcharts} highcharts/chart/zoomtype-none/
- * False by default
- * @sample {highcharts} highcharts/title/floating/
- * True - title on top of the plot area
- * @sample {highstock} stock/chart/title-floating/
- * True - title on top of the plot area
- *
- * @type {boolean}
- * @default false
- * @since 2.1
- * @apioption title.floating
- */
- /**
- * CSS styles for the title. Use this for font styling, but use `align`,
- * `x` and `y` for text alignment.
- *
- * In styled mode, the title style is given in the `.highcharts-title`
- * class.
- *
- * @sample {highcharts} highcharts/title/style/
- * Custom color and weight
- * @sample {highstock} stock/chart/title-style/
- * Custom color and weight
- * @sample highcharts/css/titles/
- * Styled mode
- *
- * @type {Highcharts.CSSObject}
- * @default {highcharts|highmaps} { "color": "#333333", "fontSize": "18px" }
- * @default {highstock} { "color": "#333333", "fontSize": "16px" }
- * @apioption title.style
- */
- /**
- * Whether to
- * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the text.
- *
- * @type {boolean}
- * @default false
- * @apioption title.useHTML
- */
- /**
- * The vertical alignment of the title. Can be one of `"top"`,
- * `"middle"` and `"bottom"`. When a value is given, the title behaves
- * as if [floating](#title.floating) were `true`.
- *
- * @sample {highcharts} highcharts/title/verticalalign/
- * Chart title in bottom right corner
- * @sample {highstock} stock/chart/title-verticalalign/
- * Chart title in bottom right corner
- *
- * @type {Highcharts.VerticalAlignValue}
- * @since 2.1
- * @apioption title.verticalAlign
- */
- /**
- * The x position of the title relative to the alignment within
- * `chart.spacingLeft` and `chart.spacingRight`.
- *
- * @sample {highcharts} highcharts/title/align/
- * Aligned to the plot area (x = 70px = margin left - spacing
- * left)
- * @sample {highstock} stock/chart/title-align/
- * Aligned to the plot area (x = 50px = margin left - spacing
- * left)
- *
- * @type {number}
- * @default 0
- * @since 2.0
- * @apioption title.x
- */
- /**
- * The y position of the title relative to the alignment within
- * [chart.spacingTop](#chart.spacingTop) and [chart.spacingBottom](
- * #chart.spacingBottom). By default it depends on the font size.
- *
- * @sample {highcharts} highcharts/title/y/
- * Title inside the plot area
- * @sample {highstock} stock/chart/title-verticalalign/
- * Chart title in bottom right corner
- *
- * @type {number}
- * @since 2.0
- * @apioption title.y
- */
- /**
- * The title of the chart. To disable the title, set the `text` to
- * `undefined`.
- *
- * @sample {highcharts} highcharts/title/text/
- * Custom title
- * @sample {highstock} stock/chart/title-text/
- * Custom title
- *
- * @default {highcharts|highmaps} Chart title
- * @default {highstock} undefined
- */
- text: 'Chart title',
- /**
- * The horizontal alignment of the title. Can be one of "left", "center"
- * and "right".
- *
- * @sample {highcharts} highcharts/title/align/
- * Aligned to the plot area (x = 70px = margin left - spacing
- * left)
- * @sample {highstock} stock/chart/title-align/
- * Aligned to the plot area (x = 50px = margin left - spacing
- * left)
- *
- * @type {Highcharts.AlignValue}
- * @since 2.0
- */
- align: 'center',
- /**
- * The margin between the title and the plot area, or if a subtitle
- * is present, the margin between the subtitle and the plot area.
- *
- * @sample {highcharts} highcharts/title/margin-50/
- * A chart title margin of 50
- * @sample {highcharts} highcharts/title/margin-subtitle/
- * The same margin applied with a subtitle
- * @sample {highstock} stock/chart/title-margin/
- * A chart title margin of 50
- *
- * @since 2.1
- */
- margin: 15,
- /**
- * Adjustment made to the title width, normally to reserve space for
- * the exporting burger menu.
- *
- * @sample highcharts/title/widthadjust/
- * Wider menu, greater padding
- *
- * @since 4.2.5
- */
- widthAdjust: -44
- },
- /**
- * The chart's subtitle. This can be used both to display a subtitle below
- * the main title, and to display random text anywhere in the chart. The
- * subtitle can be updated after chart initialization through the
- * `Chart.setTitle` method.
- *
- * @sample {highmaps} maps/title/subtitle/
- * Subtitle options demonstrated
- */
- subtitle: {
- /**
- * When the subtitle is floating, the plot area will not move to make
- * space for it.
- *
- * @sample {highcharts} highcharts/subtitle/floating/
- * Floating title and subtitle
- * @sample {highstock} stock/chart/subtitle-footnote
- * Footnote floating at bottom right of plot area
- *
- * @type {boolean}
- * @default false
- * @since 2.1
- * @apioption subtitle.floating
- */
- /**
- * CSS styles for the title.
- *
- * In styled mode, the subtitle style is given in the
- * `.highcharts-subtitle` class.
- *
- * @sample {highcharts} highcharts/subtitle/style/
- * Custom color and weight
- * @sample {highcharts} highcharts/css/titles/
- * Styled mode
- * @sample {highstock} stock/chart/subtitle-style
- * Custom color and weight
- * @sample {highstock} highcharts/css/titles/
- * Styled mode
- * @sample {highmaps} highcharts/css/titles/
- * Styled mode
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "#666666"}
- * @apioption subtitle.style
- */
- /**
- * Whether to
- * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the text.
- *
- * @type {boolean}
- * @default false
- * @apioption subtitle.useHTML
- */
- /**
- * The vertical alignment of the title. Can be one of `"top"`,
- * `"middle"` and `"bottom"`. When middle, the subtitle behaves as
- * floating.
- *
- * @sample {highcharts} highcharts/subtitle/verticalalign/
- * Footnote at the bottom right of plot area
- * @sample {highstock} stock/chart/subtitle-footnote
- * Footnote at the bottom right of plot area
- *
- * @type {Highcharts.VerticalAlignValue}
- * @since 2.1
- * @apioption subtitle.verticalAlign
- */
- /**
- * The x position of the subtitle relative to the alignment within
- * `chart.spacingLeft` and `chart.spacingRight`.
- *
- * @sample {highcharts} highcharts/subtitle/align/
- * Footnote at right of plot area
- * @sample {highstock} stock/chart/subtitle-footnote
- * Footnote at the bottom right of plot area
- *
- * @type {number}
- * @default 0
- * @since 2.0
- * @apioption subtitle.x
- */
- /**
- * The y position of the subtitle relative to the alignment within
- * `chart.spacingTop` and `chart.spacingBottom`. By default the subtitle
- * is laid out below the title unless the title is floating.
- *
- * @sample {highcharts} highcharts/subtitle/verticalalign/
- * Footnote at the bottom right of plot area
- * @sample {highstock} stock/chart/subtitle-footnote
- * Footnote at the bottom right of plot area
- *
- * @type {number}
- * @since 2.0
- * @apioption subtitle.y
- */
- /**
- * The subtitle of the chart.
- *
- * @sample {highcharts|highstock} highcharts/subtitle/text/
- * Custom subtitle
- * @sample {highcharts|highstock} highcharts/subtitle/text-formatted/
- * Formatted and linked text.
- */
- text: '',
- /**
- * The horizontal alignment of the subtitle. Can be one of "left",
- * "center" and "right".
- *
- * @sample {highcharts} highcharts/subtitle/align/
- * Footnote at right of plot area
- * @sample {highstock} stock/chart/subtitle-footnote
- * Footnote at bottom right of plot area
- *
- * @type {Highcharts.AlignValue}
- * @since 2.0
- */
- align: 'center',
- /**
- * Adjustment made to the subtitle width, normally to reserve space
- * for the exporting burger menu.
- *
- * @see [title.widthAdjust](#title.widthAdjust)
- *
- * @sample highcharts/title/widthadjust/
- * Wider menu, greater padding
- *
- * @since 4.2.5
- */
- widthAdjust: -44
- },
- /**
- * The chart's caption, which will render below the chart and will be part
- * of exported charts. The caption can be updated after chart initialization
- * through the `Chart.update` or `Chart.caption.update` methods.
- *
- * @sample highcharts/caption/text/
- * A chart with a caption
- * @since 7.2.0
- */
- caption: {
- /**
- * When the caption is floating, the plot area will not move to make
- * space for it.
- *
- * @type {boolean}
- * @default false
- * @apioption caption.floating
- */
- /**
- * The margin between the caption and the plot area.
- */
- margin: 15,
- /**
- * CSS styles for the caption.
- *
- * In styled mode, the caption style is given in the
- * `.highcharts-caption` class.
- *
- * @sample {highcharts} highcharts/css/titles/
- * Styled mode
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "#666666"}
- * @apioption caption.style
- */
- /**
- * Whether to
- * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the text.
- *
- * @type {boolean}
- * @default false
- * @apioption caption.useHTML
- */
- /**
- * The x position of the caption relative to the alignment within
- * `chart.spacingLeft` and `chart.spacingRight`.
- *
- * @type {number}
- * @default 0
- * @apioption caption.x
- */
- /**
- * The y position of the caption relative to the alignment within
- * `chart.spacingTop` and `chart.spacingBottom`.
- *
- * @type {number}
- * @apioption caption.y
- */
- /**
- * The caption text of the chart.
- *
- * @sample {highcharts} highcharts/caption/text/
- * Custom caption
- */
- text: '',
- /**
- * The horizontal alignment of the caption. Can be one of "left",
- * "center" and "right".
- *
- * @type {Highcharts.AlignValue}
- */
- align: 'left',
- /**
- * The vertical alignment of the caption. Can be one of `"top"`,
- * `"middle"` and `"bottom"`. When middle, the caption behaves as
- * floating.
- *
- * @type {Highcharts.VerticalAlignValue}
- */
- verticalAlign: 'bottom'
- },
- /**
- * The plotOptions is a wrapper object for config objects for each series
- * type. The config objects for each series can also be overridden for
- * each series item as given in the series array.
- *
- * Configuration options for the series are given in three levels. Options
- * for all series in a chart are given in the [plotOptions.series](
- * #plotOptions.series) object. Then options for all series of a specific
- * type are given in the plotOptions of that type, for example
- * `plotOptions.line`. Next, options for one single series are given in
- * [the series array](#series).
- */
- plotOptions: {},
- /**
- * HTML labels that can be positioned anywhere in the chart area.
- *
- * This option is deprecated since v7.1.2. Instead, use
- * [annotations](#annotations) that support labels.
- *
- * @deprecated
- * @product highcharts highstock
- */
- labels: {
- /**
- * An HTML label that can be positioned anywhere in the chart area.
- *
- * @deprecated
- * @type {Array<*>}
- * @apioption labels.items
- */
- /**
- * Inner HTML or text for the label.
- *
- * @deprecated
- * @type {string}
- * @apioption labels.items.html
- */
- /**
- * CSS styles for each label. To position the label, use left and top
- * like this:
- * ```js
- * style: {
- * left: '100px',
- * top: '100px'
- * }
- * ```
- *
- * @deprecated
- * @type {Highcharts.CSSObject}
- * @apioption labels.items.style
- */
- /**
- * Shared CSS styles for all labels.
- *
- * @deprecated
- * @type {Highcharts.CSSObject}
- * @default {"color": "#333333", "position": "absolute"}
- */
- style: {
- /**
- * @ignore-option
- */
- position: 'absolute',
- /**
- * @ignore-option
- */
- color: '#333333'
- }
- },
- /**
- * The legend is a box containing a symbol and name for each series
- * item or point item in the chart. Each series (or points in case
- * of pie charts) is represented by a symbol and its name in the legend.
- *
- * It is possible to override the symbol creator function and create
- * [custom legend symbols](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/studies/legend-custom-symbol/).
- *
- * @productdesc {highmaps}
- * A Highmaps legend by default contains one legend item per series, but if
- * a `colorAxis` is defined, the axis will be displayed in the legend.
- * Either as a gradient, or as multiple legend items for `dataClasses`.
- */
- legend: {
- /**
- * The background color of the legend.
- *
- * @see In styled mode, the legend background fill can be applied with
- * the `.highcharts-legend-box` class.
- *
- * @sample {highcharts} highcharts/legend/backgroundcolor/
- * Yellowish background
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/border-background/
- * Border and background options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption legend.backgroundColor
- */
- /**
- * The width of the drawn border around the legend.
- *
- * @see In styled mode, the legend border stroke width can be applied
- * with the `.highcharts-legend-box` class.
- *
- * @sample {highcharts} highcharts/legend/borderwidth/
- * 2px border width
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/border-background/
- * Border and background options
- *
- * @type {number}
- * @default 0
- * @apioption legend.borderWidth
- */
- /**
- * Enable or disable the legend. There is also a series-specific option,
- * [showInLegend](#plotOptions.series.showInLegend), that can hide the
- * series from the legend. In some series types this is `false` by
- * default, so it must set to `true` in order to show the legend for the
- * series.
- *
- * @sample {highcharts} highcharts/legend/enabled-false/ Legend disabled
- * @sample {highstock} stock/legend/align/ Various legend options
- * @sample {highmaps} maps/legend/enabled-false/ Legend disabled
- *
- * @default {highstock} false
- * @default {highmaps} true
- * @default {gantt} false
- */
- enabled: true,
- /**
- * The horizontal alignment of the legend box within the chart area.
- * Valid values are `left`, `center` and `right`.
- *
- * In the case that the legend is aligned in a corner position, the
- * `layout` option will determine whether to place it above/below
- * or on the side of the plot area.
- *
- * @sample {highcharts} highcharts/legend/align/
- * Legend at the right of the chart
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/alignment/
- * Legend alignment
- *
- * @type {Highcharts.AlignValue}
- * @since 2.0
- */
- align: 'center',
- /**
- * If the [layout](legend.layout) is `horizontal` and the legend items
- * span over two lines or more, whether to align the items into vertical
- * columns. Setting this to `false` makes room for more items, but will
- * look more messy.
- *
- * @since 6.1.0
- */
- alignColumns: true,
- /**
- * When the legend is floating, the plot area ignores it and is allowed
- * to be placed below it.
- *
- * @sample {highcharts} highcharts/legend/floating-false/
- * False by default
- * @sample {highcharts} highcharts/legend/floating-true/
- * True
- * @sample {highmaps} maps/legend/alignment/
- * Floating legend
- *
- * @type {boolean}
- * @default false
- * @since 2.1
- * @apioption legend.floating
- */
- /**
- * The layout of the legend items. Can be one of `horizontal` or
- * `vertical` or `proximate`. When `proximate`, the legend items will be
- * placed as close as possible to the graphs they're representing,
- * except in inverted charts or when the legend position doesn't allow
- * it.
- *
- * @sample {highcharts} highcharts/legend/layout-horizontal/
- * Horizontal by default
- * @sample {highcharts} highcharts/legend/layout-vertical/
- * Vertical
- * @sample highcharts/legend/layout-proximate
- * Labels proximate to the data
- * @sample {highstock} stock/legend/layout-horizontal/
- * Horizontal by default
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * Vertical with data classes
- * @sample {highmaps} maps/legend/layout-vertical/
- * Vertical with color axis gradient
- *
- * @validvalue ["horizontal", "vertical", "proximate"]
- */
- layout: 'horizontal',
- /**
- * In a legend with horizontal layout, the itemDistance defines the
- * pixel distance between each item.
- *
- * @sample {highcharts} highcharts/legend/layout-horizontal/
- * 50px item distance
- * @sample {highstock} highcharts/legend/layout-horizontal/
- * 50px item distance
- *
- * @type {number}
- * @default {highcharts} 20
- * @default {highstock} 20
- * @default {highmaps} 8
- * @since 3.0.3
- * @apioption legend.itemDistance
- */
- /**
- * The pixel bottom margin for each legend item.
- *
- * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- *
- * @type {number}
- * @default 0
- * @since 2.2.0
- * @apioption legend.itemMarginBottom
- */
- /**
- * The pixel top margin for each legend item.
- *
- * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- *
- * @type {number}
- * @default 0
- * @since 2.2.0
- * @apioption legend.itemMarginTop
- */
- /**
- * The width for each legend item. By default the items are laid out
- * successively. In a [horizontal layout](legend.layout), if the items
- * are laid out across two rows or more, they will be vertically aligned
- * depending on the [legend.alignColumns](legend.alignColumns) option.
- *
- * @sample {highcharts} highcharts/legend/itemwidth-default/
- * Undefined by default
- * @sample {highcharts} highcharts/legend/itemwidth-80/
- * 80 for aligned legend items
- *
- * @type {number}
- * @since 2.0
- * @apioption legend.itemWidth
- */
- /**
- * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting)
- * for each legend label. Available variables relates to properties on
- * the series, or the point in case of pies.
- *
- * @type {string}
- * @default {name}
- * @since 1.3
- * @apioption legend.labelFormat
- */
- /* eslint-disable valid-jsdoc */
- /**
- * Callback function to format each of the series' labels. The `this`
- * keyword refers to the series object, or the point object in case of
- * pie charts. By default the series or point name is printed.
- *
- * @productdesc {highmaps}
- * In Highmaps the context can also be a data class in case of a
- * `colorAxis`.
- *
- * @sample {highcharts} highcharts/legend/labelformatter/
- * Add text
- * @sample {highmaps} maps/legend/labelformatter/
- * Data classes with label formatter
- *
- * @type {Highcharts.FormatterCallbackFunction<Point|Series>}
- */
- labelFormatter: function () {
- /** eslint-enable valid-jsdoc */
- return this.name;
- },
- /**
- * Line height for the legend items. Deprecated as of 2.1\. Instead,
- * the line height for each item can be set using
- * `itemStyle.lineHeight`, and the padding between items using
- * `itemMarginTop` and `itemMarginBottom`.
- *
- * @sample {highcharts} highcharts/legend/lineheight/
- * Setting padding
- *
- * @deprecated
- *
- * @type {number}
- * @default 16
- * @since 2.0
- * @product highcharts gantt
- * @apioption legend.lineHeight
- */
- /**
- * If the plot area sized is calculated automatically and the legend is
- * not floating, the legend margin is the space between the legend and
- * the axis labels or plot area.
- *
- * @sample {highcharts} highcharts/legend/margin-default/
- * 12 pixels by default
- * @sample {highcharts} highcharts/legend/margin-30/
- * 30 pixels
- *
- * @type {number}
- * @default 12
- * @since 2.1
- * @apioption legend.margin
- */
- /**
- * Maximum pixel height for the legend. When the maximum height is
- * extended, navigation will show.
- *
- * @type {number}
- * @since 2.3.0
- * @apioption legend.maxHeight
- */
- /**
- * The color of the drawn border around the legend.
- *
- * @see In styled mode, the legend border stroke can be applied with the
- * `.highcharts-legend-box` class.
- *
- * @sample {highcharts} highcharts/legend/bordercolor/
- * Brown border
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/border-background/
- * Border and background options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- borderColor: '#999999',
- /**
- * The border corner radius of the legend.
- *
- * @sample {highcharts} highcharts/legend/borderradius-default/
- * Square by default
- * @sample {highcharts} highcharts/legend/borderradius-round/
- * 5px rounded
- * @sample {highmaps} maps/legend/border-background/
- * Border and background options
- */
- borderRadius: 0,
- /**
- * Options for the paging or navigation appearing when the legend is
- * overflown. Navigation works well on screen, but not in static
- * exported images. One way of working around that is to
- * [increase the chart height in
- * export](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/legend/navigation-enabled-false/).
- */
- navigation: {
- /**
- * How to animate the pages when navigating up or down. A value of
- * `true` applies the default navigation given in the
- * `chart.animation` option. Additional options can be given as an
- * object containing values for easing and duration.
- *
- * @sample {highcharts} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- * @sample {highstock} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @default true
- * @since 2.2.4
- * @apioption legend.navigation.animation
- */
- /**
- * The pixel size of the up and down arrows in the legend paging
- * navigation.
- *
- * @sample {highcharts} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- * @sample {highstock} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- *
- * @type {number}
- * @default 12
- * @since 2.2.4
- * @apioption legend.navigation.arrowSize
- */
- /**
- * Whether to enable the legend navigation. In most cases, disabling
- * the navigation results in an unwanted overflow.
- *
- * See also the [adapt chart to legend](
- * https://www.highcharts.com/products/plugin-registry/single/8/Adapt-Chart-To-Legend)
- * plugin for a solution to extend the chart height to make room for
- * the legend, optionally in exported charts only.
- *
- * @type {boolean}
- * @default true
- * @since 4.2.4
- * @apioption legend.navigation.enabled
- */
- /**
- * Text styles for the legend page navigation.
- *
- * @see In styled mode, the navigation items are styled with the
- * `.highcharts-legend-navigation` class.
- *
- * @sample {highcharts} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- * @sample {highstock} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- *
- * @type {Highcharts.CSSObject}
- * @since 2.2.4
- * @apioption legend.navigation.style
- */
- /**
- * The color for the active up or down arrow in the legend page
- * navigation.
- *
- * @see In styled mode, the active arrow be styled with the
- * `.highcharts-legend-nav-active` class.
- *
- * @sample {highcharts} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- * @sample {highstock} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 2.2.4
- */
- activeColor: '#003399',
- /**
- * The color of the inactive up or down arrow in the legend page
- * navigation. .
- *
- * @see In styled mode, the inactive arrow be styled with the
- * `.highcharts-legend-nav-inactive` class.
- *
- * @sample {highcharts} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- * @sample {highstock} highcharts/legend/navigation/
- * Legend page navigation demonstrated
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 2.2.4
- */
- inactiveColor: '#cccccc'
- },
- /**
- * The inner padding of the legend box.
- *
- * @sample {highcharts|highstock} highcharts/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- *
- * @type {number}
- * @default 8
- * @since 2.2.0
- * @apioption legend.padding
- */
- /**
- * Whether to reverse the order of the legend items compared to the
- * order of the series or points as defined in the configuration object.
- *
- * @see [yAxis.reversedStacks](#yAxis.reversedStacks),
- * [series.legendIndex](#series.legendIndex)
- *
- * @sample {highcharts} highcharts/legend/reversed/
- * Stacked bar with reversed legend
- *
- * @type {boolean}
- * @default false
- * @since 1.2.5
- * @apioption legend.reversed
- */
- /**
- * Whether to show the symbol on the right side of the text rather than
- * the left side. This is common in Arabic and Hebraic.
- *
- * @sample {highcharts} highcharts/legend/rtl/
- * Symbol to the right
- *
- * @type {boolean}
- * @default false
- * @since 2.2
- * @apioption legend.rtl
- */
- /**
- * CSS styles for the legend area. In the 1.x versions the position
- * of the legend area was determined by CSS. In 2.x, the position is
- * determined by properties like `align`, `verticalAlign`, `x` and `y`,
- * but the styles are still parsed for backwards compatibility.
- *
- * @deprecated
- *
- * @type {Highcharts.CSSObject}
- * @product highcharts highstock
- * @apioption legend.style
- */
- /**
- * CSS styles for each legend item. Only a subset of CSS is supported,
- * notably those options related to text. The default `textOverflow`
- * property makes long texts truncate. Set it to `undefined` to wrap
- * text instead. A `width` property can be added to control the text
- * width.
- *
- * @see In styled mode, the legend items can be styled with the
- * `.highcharts-legend-item` class.
- *
- * @sample {highcharts} highcharts/legend/itemstyle/
- * Bold black text
- * @sample {highmaps} maps/legend/itemstyle/
- * Item text styles
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "#333333", "cursor": "pointer", "fontSize": "12px", "fontWeight": "bold", "textOverflow": "ellipsis"}
- */
- itemStyle: {
- /**
- * @ignore
- */
- color: '#333333',
- /**
- * @ignore
- */
- cursor: 'pointer',
- /**
- * @ignore
- */
- fontSize: '12px',
- /**
- * @ignore
- */
- fontWeight: 'bold',
- /**
- * @ignore
- */
- textOverflow: 'ellipsis'
- },
- /**
- * CSS styles for each legend item in hover mode. Only a subset of
- * CSS is supported, notably those options related to text. Properties
- * are inherited from `style` unless overridden here.
- *
- * @see In styled mode, the hovered legend items can be styled with
- * the `.highcharts-legend-item:hover` pesudo-class.
- *
- * @sample {highcharts} highcharts/legend/itemhoverstyle/
- * Red on hover
- * @sample {highmaps} maps/legend/itemstyle/
- * Item text styles
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "#000000"}
- */
- itemHoverStyle: {
- /**
- * @ignore
- */
- color: '#000000'
- },
- /**
- * CSS styles for each legend item when the corresponding series or
- * point is hidden. Only a subset of CSS is supported, notably those
- * options related to text. Properties are inherited from `style`
- * unless overridden here.
- *
- * @see In styled mode, the hidden legend items can be styled with
- * the `.highcharts-legend-item-hidden` class.
- *
- * @sample {highcharts} highcharts/legend/itemhiddenstyle/
- * Darker gray color
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "#cccccc"}
- */
- itemHiddenStyle: {
- /**
- * @ignore
- */
- color: '#cccccc'
- },
- /**
- * Whether to apply a drop shadow to the legend. A `backgroundColor`
- * also needs to be applied for this to take effect. The shadow can be
- * an object configuration containing `color`, `offsetX`, `offsetY`,
- * `opacity` and `width`.
- *
- * @sample {highcharts} highcharts/legend/shadow/
- * White background and drop shadow
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/border-background/
- * Border and background options
- *
- * @type {boolean|Highcharts.CSSObject}
- */
- shadow: false,
- /**
- * Default styling for the checkbox next to a legend item when
- * `showCheckbox` is true.
- *
- * @type {Highcharts.CSSObject}
- * @default {"width": "13px", "height": "13px", "position":"absolute"}
- */
- itemCheckboxStyle: {
- /**
- * @ignore
- */
- position: 'absolute',
- /**
- * @ignore
- */
- width: '13px',
- /**
- * @ignore
- */
- height: '13px'
- },
- // itemWidth: undefined,
- /**
- * When this is true, the legend symbol width will be the same as
- * the symbol height, which in turn defaults to the font size of the
- * legend items.
- *
- * @since 5.0.0
- */
- squareSymbol: true,
- /**
- * The pixel height of the symbol for series types that use a rectangle
- * in the legend. Defaults to the font size of legend items.
- *
- * @productdesc {highmaps}
- * In Highmaps, when the symbol is the gradient of a vertical color
- * axis, the height defaults to 200.
- *
- * @sample {highmaps} maps/legend/layout-vertical-sized/
- * Sized vertical gradient
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * No distance between data classes
- *
- * @type {number}
- * @since 3.0.8
- * @apioption legend.symbolHeight
- */
- /**
- * The border radius of the symbol for series types that use a rectangle
- * in the legend. Defaults to half the `symbolHeight`.
- *
- * @sample {highcharts} highcharts/legend/symbolradius/
- * Round symbols
- * @sample {highstock} highcharts/legend/symbolradius/
- * Round symbols
- * @sample {highmaps} highcharts/legend/symbolradius/
- * Round symbols
- *
- * @type {number}
- * @since 3.0.8
- * @apioption legend.symbolRadius
- */
- /**
- * The pixel width of the legend item symbol. When the `squareSymbol`
- * option is set, this defaults to the `symbolHeight`, otherwise 16.
- *
- * @productdesc {highmaps}
- * In Highmaps, when the symbol is the gradient of a horizontal color
- * axis, the width defaults to 200.
- *
- * @sample {highcharts} highcharts/legend/symbolwidth/
- * Greater symbol width and padding
- * @sample {highmaps} maps/legend/padding-itemmargin/
- * Padding and item margins demonstrated
- * @sample {highmaps} maps/legend/layout-vertical-sized/
- * Sized vertical gradient
- *
- * @type {number}
- * @apioption legend.symbolWidth
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the legend item texts.
- *
- * Prior to 4.1.7, when using HTML, [legend.navigation](
- * #legend.navigation) was disabled.
- *
- * @type {boolean}
- * @default false
- * @apioption legend.useHTML
- */
- /**
- * The width of the legend box. If a number is set, it translates to
- * pixels. Since v7.0.2 it allows setting a percent string of the full
- * chart width, for example `40%`.
- *
- * Defaults to the full chart width for legends below or above the
- * chart, half the chart width for legends to the left and right.
- *
- * @sample {highcharts} highcharts/legend/width/
- * Aligned to the plot area
- * @sample {highcharts} highcharts/legend/width-percent/
- * A percent of the chart width
- *
- * @type {number|string}
- * @since 2.0
- * @apioption legend.width
- */
- /**
- * The pixel padding between the legend item symbol and the legend
- * item text.
- *
- * @sample {highcharts} highcharts/legend/symbolpadding/
- * Greater symbol width and padding
- */
- symbolPadding: 5,
- /**
- * The vertical alignment of the legend box. Can be one of `top`,
- * `middle` or `bottom`. Vertical position can be further determined
- * by the `y` option.
- *
- * In the case that the legend is aligned in a corner position, the
- * `layout` option will determine whether to place it above/below
- * or on the side of the plot area.
- *
- * When the [layout](#legend.layout) option is `proximate`, the
- * `verticalAlign` option doesn't apply.
- *
- * @sample {highcharts} highcharts/legend/verticalalign/
- * Legend 100px from the top of the chart
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/alignment/
- * Legend alignment
- *
- * @type {Highcharts.VerticalAlignValue}
- * @since 2.0
- */
- verticalAlign: 'bottom',
- // width: undefined,
- /**
- * The x offset of the legend relative to its horizontal alignment
- * `align` within chart.spacingLeft and chart.spacingRight. Negative
- * x moves it to the left, positive x moves it to the right.
- *
- * @sample {highcharts} highcharts/legend/width/
- * Aligned to the plot area
- *
- * @since 2.0
- */
- x: 0,
- /**
- * The vertical offset of the legend relative to it's vertical alignment
- * `verticalAlign` within chart.spacingTop and chart.spacingBottom.
- * Negative y moves it up, positive y moves it down.
- *
- * @sample {highcharts} highcharts/legend/verticalalign/
- * Legend 100px from the top of the chart
- * @sample {highstock} stock/legend/align/
- * Various legend options
- * @sample {highmaps} maps/legend/alignment/
- * Legend alignment
- *
- * @since 2.0
- */
- y: 0,
- /**
- * A title to be added on top of the legend.
- *
- * @sample {highcharts} highcharts/legend/title/
- * Legend title
- * @sample {highmaps} maps/legend/alignment/
- * Legend with title
- *
- * @since 3.0
- */
- title: {
- /**
- * A text or HTML string for the title.
- *
- * @type {string}
- * @since 3.0
- * @apioption legend.title.text
- */
- /**
- * Generic CSS styles for the legend title.
- *
- * @see In styled mode, the legend title is styled with the
- * `.highcharts-legend-title` class.
- *
- * @type {Highcharts.CSSObject}
- * @default {"fontWeight": "bold"}
- * @since 3.0
- */
- style: {
- /**
- * @ignore
- */
- fontWeight: 'bold'
- }
- }
- },
- /**
- * The loading options control the appearance of the loading screen
- * that covers the plot area on chart operations. This screen only
- * appears after an explicit call to `chart.showLoading()`. It is a
- * utility for developers to communicate to the end user that something
- * is going on, for example while retrieving new data via an XHR connection.
- * The "Loading..." text itself is not part of this configuration
- * object, but part of the `lang` object.
- */
- loading: {
- /**
- * The duration in milliseconds of the fade out effect.
- *
- * @sample highcharts/loading/hideduration/
- * Fade in and out over a second
- *
- * @type {number}
- * @default 100
- * @since 1.2.0
- * @apioption loading.hideDuration
- */
- /**
- * The duration in milliseconds of the fade in effect.
- *
- * @sample highcharts/loading/hideduration/
- * Fade in and out over a second
- *
- * @type {number}
- * @default 100
- * @since 1.2.0
- * @apioption loading.showDuration
- */
- /**
- * CSS styles for the loading label `span`.
- *
- * @see In styled mode, the loading label is styled with the
- * `.highcharts-loading-inner` class.
- *
- * @sample {highcharts|highmaps} highcharts/loading/labelstyle/
- * Vertically centered
- * @sample {highstock} stock/loading/general/
- * Label styles
- *
- * @type {Highcharts.CSSObject}
- * @default {"fontWeight": "bold", "position": "relative", "top": "45%"}
- * @since 1.2.0
- */
- labelStyle: {
- /**
- * @ignore
- */
- fontWeight: 'bold',
- /**
- * @ignore
- */
- position: 'relative',
- /**
- * @ignore
- */
- top: '45%'
- },
- /**
- * CSS styles for the loading screen that covers the plot area.
- *
- * In styled mode, the loading label is styled with the
- * `.highcharts-loading` class.
- *
- * @sample {highcharts|highmaps} highcharts/loading/style/
- * Gray plot area, white text
- * @sample {highstock} stock/loading/general/
- * Gray plot area, white text
- *
- * @type {Highcharts.CSSObject}
- * @default {"position": "absolute", "backgroundColor": "#ffffff", "opacity": 0.5, "textAlign": "center"}
- * @since 1.2.0
- */
- style: {
- /**
- * @ignore
- */
- position: 'absolute',
- /**
- * @ignore
- */
- backgroundColor: '#ffffff',
- /**
- * @ignore
- */
- opacity: 0.5,
- /**
- * @ignore
- */
- textAlign: 'center'
- }
- },
- /**
- * Options for the tooltip that appears when the user hovers over a
- * series or point.
- *
- * @declare Highcharts.TooltipOptions
- */
- tooltip: {
- /**
- * The color of the tooltip border. When `undefined`, the border takes
- * the color of the corresponding series or point.
- *
- * @sample {highcharts} highcharts/tooltip/bordercolor-default/
- * Follow series by default
- * @sample {highcharts} highcharts/tooltip/bordercolor-black/
- * Black border
- * @sample {highstock} stock/tooltip/general/
- * Styled tooltip
- * @sample {highmaps} maps/tooltip/background-border/
- * Background and border demo
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption tooltip.borderColor
- */
- /**
- * A CSS class name to apply to the tooltip's container div,
- * allowing unique CSS styling for each chart.
- *
- * @type {string}
- * @apioption tooltip.className
- */
- /**
- * Since 4.1, the crosshair definitions are moved to the Axis object
- * in order for a better separation from the tooltip. See
- * [xAxis.crosshair](#xAxis.crosshair).
- *
- * @sample {highcharts} highcharts/tooltip/crosshairs-x/
- * Enable a crosshair for the x value
- *
- * @deprecated
- *
- * @type {*}
- * @default true
- * @apioption tooltip.crosshairs
- */
- /**
- * Distance from point to tooltip in pixels.
- *
- * @type {number}
- * @default 16
- * @apioption tooltip.distance
- */
- /**
- * Whether the tooltip should follow the mouse as it moves across
- * columns, pie slices and other point types with an extent.
- * By default it behaves this way for pie, polygon, map, sankey
- * and wordcloud series by override in the `plotOptions`
- * for those series types.
- *
- * For touch moves to behave the same way, [followTouchMove](
- * #tooltip.followTouchMove) must be `true` also.
- *
- * @type {boolean}
- * @default {highcharts} false
- * @default {highstock} false
- * @default {highmaps} true
- * @since 3.0
- * @apioption tooltip.followPointer
- */
- /**
- * Whether the tooltip should update as the finger moves on a touch
- * device. If this is `true` and [chart.panning](#chart.panning) is
- * set,`followTouchMove` will take over one-finger touches, so the user
- * needs to use two fingers for zooming and panning.
- *
- * Note the difference to [followPointer](#tooltip.followPointer) that
- * only defines the _position_ of the tooltip. If `followPointer` is
- * false in for example a column series, the tooltip will show above or
- * below the column, but as `followTouchMove` is true, the tooltip will
- * jump from column to column as the user swipes across the plot area.
- *
- * @type {boolean}
- * @default {highcharts} true
- * @default {highstock} true
- * @default {highmaps} false
- * @since 3.0.1
- * @apioption tooltip.followTouchMove
- */
- /**
- * Callback function to format the text of the tooltip from scratch. In
- * case of single or [shared](#tooltip.shared) tooltips, a string should
- * be returned. In case of [split](#tooltip.split) tooltips, it should
- * return an array where the first item is the header, and subsequent
- * items are mapped to the points. Return `false` to disable tooltip for
- * a specific point on series.
- *
- * A subset of HTML is supported. Unless `useHTML` is true, the HTML of
- * the tooltip is parsed and converted to SVG, therefore this isn't a
- * complete HTML renderer. The following HTML tags are supported: `b`,
- * `br`, `em`, `i`, `span`, `strong`. Spans can be styled with a `style`
- * attribute, but only text-related CSS, that is shared with SVG, is
- * handled.
- *
- * The available data in the formatter differ a bit depending on whether
- * the tooltip is shared or split, or belongs to a single point. In a
- * shared/split tooltip, all properties except `x`, which is common for
- * all points, are kept in an array, `this.points`.
- *
- * Available data are:
- *
- * - **this.percentage (not shared) /**
- * **this.points[i].percentage (shared)**:
- * Stacked series and pies only. The point's percentage of the total.
- *
- * - **this.point (not shared) / this.points[i].point (shared)**:
- * The point object. The point name, if defined, is available through
- * `this.point.name`.
- *
- * - **this.points**:
- * In a shared tooltip, this is an array containing all other
- * properties for each point.
- *
- * - **this.series (not shared) / this.points[i].series (shared)**:
- * The series object. The series name is available through
- * `this.series.name`.
- *
- * - **this.total (not shared) / this.points[i].total (shared)**:
- * Stacked series only. The total value at this point's x value.
- *
- * - **this.x**:
- * The x value. This property is the same regardless of the tooltip
- * being shared or not.
- *
- * - **this.y (not shared) / this.points[i].y (shared)**:
- * The y value.
- *
- * @sample {highcharts} highcharts/tooltip/formatter-simple/
- * Simple string formatting
- * @sample {highcharts} highcharts/tooltip/formatter-shared/
- * Formatting with shared tooltip
- * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/
- * Formatting with split tooltip
- * @sample highcharts/tooltip/formatter-conditional-default/
- * Extending default formatter
- * @sample {highstock} stock/tooltip/formatter/
- * Formatting with shared tooltip
- * @sample {highmaps} maps/tooltip/formatter/
- * String formatting
- *
- * @type {Highcharts.TooltipFormatterCallbackFunction}
- * @apioption tooltip.formatter
- */
- /**
- * Callback function to format the text of the tooltip for
- * visible null points.
- * Works analogously to [formatter](#tooltip.formatter).
- *
- * @sample highcharts/plotoptions/series-nullformat
- * Format data label and tooltip for null point.
- *
- * @type {Highcharts.TooltipFormatterCallbackFunction}
- * @apioption tooltip.nullFormatter
- */
- /**
- * The number of milliseconds to wait until the tooltip is hidden when
- * mouse out from a point or chart.
- *
- * @type {number}
- * @default 500
- * @since 3.0
- * @apioption tooltip.hideDelay
- */
- /**
- * Whether to allow the tooltip to render outside the chart's SVG
- * element box. By default (`false`), the tooltip is rendered within the
- * chart's SVG element, which results in the tooltip being aligned
- * inside the chart area. For small charts, this may result in clipping
- * or overlapping. When `true`, a separate SVG element is created and
- * overlaid on the page, allowing the tooltip to be aligned inside the
- * page itself.
- *
- * Defaults to `true` if `chart.scrollablePlotArea` is activated,
- * otherwise `false`.
- *
- * @sample highcharts/tooltip/outside
- * Small charts with tooltips outside
- *
- * @type {boolean|undefined}
- * @default undefined
- * @since 6.1.1
- * @apioption tooltip.outside
- */
- /**
- * A callback function for formatting the HTML output for a single point
- * in the tooltip. Like the `pointFormat` string, but with more
- * flexibility.
- *
- * @type {Highcharts.FormatterCallbackFunction<Highcharts.Point>}
- * @since 4.1.0
- * @context Highcharts.Point
- * @apioption tooltip.pointFormatter
- */
- /**
- * A callback function to place the tooltip in a default position. The
- * callback receives three parameters: `labelWidth`, `labelHeight` and
- * `point`, where point contains values for `plotX` and `plotY` telling
- * where the reference point is in the plot area. Add `chart.plotLeft`
- * and `chart.plotTop` to get the full coordinates.
- *
- * Since v7, when [tooltip.split](#tooltip.split) option is enabled,
- * positioner is called for each of the boxes separately, including
- * xAxis header. xAxis header is not a point, instead `point` argument
- * contains info:
- * `{ plotX: Number, plotY: Number, isHeader: Boolean }`
- *
- *
- * The return should be an object containing x and y values, for example
- * `{ x: 100, y: 100 }`.
- *
- * @sample {highcharts} highcharts/tooltip/positioner/
- * A fixed tooltip position
- * @sample {highstock} stock/tooltip/positioner/
- * A fixed tooltip position on top of the chart
- * @sample {highmaps} maps/tooltip/positioner/
- * A fixed tooltip position
- * @sample {highstock} stock/tooltip/split-positioner/
- * Split tooltip with fixed positions
- * @sample {highstock} stock/tooltip/positioner-scrollable-plotarea/
- * Scrollable plot area combined with tooltip positioner
- *
- * @type {Highcharts.TooltipPositionerCallbackFunction}
- * @since 2.2.4
- * @apioption tooltip.positioner
- */
- /**
- * The name of a symbol to use for the border around the tooltip. Can
- * be one of: `"callout"`, `"circle"`, or `"square"`. When
- * [tooltip.split](#tooltip.split)
- * option is enabled, shape is applied to all boxes except header, which
- * is controlled by
- * [tooltip.headerShape](#tooltip.headerShape).
- *
- * Custom callbacks for symbol path generation can also be added to
- * `Highcharts.SVGRenderer.prototype.symbols` the same way as for
- * [series.marker.symbol](plotOptions.line.marker.symbol).
- *
- * @type {Highcharts.TooltipShapeValue}
- * @default callout
- * @since 4.0
- * @apioption tooltip.shape
- */
- /**
- * The name of a symbol to use for the border around the tooltip
- * header. Applies only when [tooltip.split](#tooltip.split) is
- * enabled.
- *
- * Custom callbacks for symbol path generation can also be added to
- * `Highcharts.SVGRenderer.prototype.symbols` the same way as for
- * [series.marker.symbol](plotOptions.line.marker.symbol).
- *
- * @see [tooltip.shape](#tooltip.shape)
- *
- * @sample {highstock} stock/tooltip/split-positioner/
- * Different shapes for header and split boxes
- *
- * @type {Highcharts.TooltipShapeValue}
- * @default callout
- * @validvalue ["callout", "square"]
- * @since 7.0
- * @apioption tooltip.headerShape
- */
- /**
- * When the tooltip is shared, the entire plot area will capture mouse
- * movement or touch events. Tooltip texts for series types with ordered
- * data (not pie, scatter, flags etc) will be shown in a single bubble.
- * This is recommended for single series charts and for tablet/mobile
- * optimized charts.
- *
- * See also [tooltip.split](#tooltip.split), that is better suited for
- * charts with many series, especially line-type series. The
- * `tooltip.split` option takes precedence over `tooltip.shared`.
- *
- * @sample {highcharts} highcharts/tooltip/shared-false/
- * False by default
- * @sample {highcharts} highcharts/tooltip/shared-true/
- * True
- * @sample {highcharts} highcharts/tooltip/shared-x-crosshair/
- * True with x axis crosshair
- * @sample {highcharts} highcharts/tooltip/shared-true-mixed-types/
- * True with mixed series types
- *
- * @type {boolean}
- * @default false
- * @since 2.1
- * @product highcharts highstock
- * @apioption tooltip.shared
- */
- /**
- * Split the tooltip into one label per series, with the header close
- * to the axis. This is recommended over [shared](#tooltip.shared)
- * tooltips for charts with multiple line series, generally making them
- * easier to read. This option takes precedence over `tooltip.shared`.
- *
- * @productdesc {highstock} In Highstock, tooltips are split by default
- * since v6.0.0. Stock charts typically contain multi-dimension points
- * and multiple panes, making split tooltips the preferred layout over
- * the previous `shared` tooltip.
- *
- * @sample highcharts/tooltip/split/
- * Split tooltip
- * @sample {highcharts|highstock} highcharts/tooltip/formatter-split/
- * Split tooltip and custom formatter callback
- *
- * @type {boolean}
- * @default {highcharts} false
- * @default {highstock} true
- * @since 5.0.0
- * @product highcharts highstock
- * @apioption tooltip.split
- */
- /**
- * Prevents the tooltip from switching or closing, when touched or
- * pointed.
- *
- * @sample highcharts/tooltip/stickoncontact/
- * Tooltip sticks on pointer contact
- *
- * @type {boolean}
- * @since 8.0.1
- * @apioption tooltip.stickOnContact
- */
- /**
- * Use HTML to render the contents of the tooltip instead of SVG. Using
- * HTML allows advanced formatting like tables and images in the
- * tooltip. It is also recommended for rtl languages as it works around
- * rtl bugs in early Firefox.
- *
- * @sample {highcharts|highstock} highcharts/tooltip/footerformat/
- * A table for value alignment
- * @sample {highcharts|highstock} highcharts/tooltip/fullhtml/
- * Full HTML tooltip
- * @sample {highmaps} maps/tooltip/usehtml/
- * Pure HTML tooltip
- *
- * @type {boolean}
- * @default false
- * @since 2.2
- * @apioption tooltip.useHTML
- */
- /**
- * How many decimals to show in each series' y value. This is
- * overridable in each series' tooltip options object. The default is to
- * preserve all decimals.
- *
- * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- * @sample {highmaps} maps/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- *
- * @type {number}
- * @since 2.2
- * @apioption tooltip.valueDecimals
- */
- /**
- * A string to prepend to each series' y value. Overridable in each
- * series' tooltip options object.
- *
- * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- * @sample {highmaps} maps/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- *
- * @type {string}
- * @since 2.2
- * @apioption tooltip.valuePrefix
- */
- /**
- * A string to append to each series' y value. Overridable in each
- * series' tooltip options object.
- *
- * @sample {highcharts|highstock} highcharts/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- * @sample {highmaps} maps/tooltip/valuedecimals/
- * Set decimals, prefix and suffix for the value
- *
- * @type {string}
- * @since 2.2
- * @apioption tooltip.valueSuffix
- */
- /**
- * The format for the date in the tooltip header if the X axis is a
- * datetime axis. The default is a best guess based on the smallest
- * distance between points in the chart.
- *
- * @sample {highcharts} highcharts/tooltip/xdateformat/
- * A different format
- *
- * @type {string}
- * @product highcharts highstock gantt
- * @apioption tooltip.xDateFormat
- */
- /**
- * How many decimals to show for the `point.change` value when the
- * `series.compare` option is set. This is overridable in each series'
- * tooltip options object. The default is to preserve all decimals.
- *
- * @type {number}
- * @since 1.0.1
- * @product highstock
- * @apioption tooltip.changeDecimals
- */
- /**
- * Enable or disable the tooltip.
- *
- * @sample {highcharts} highcharts/tooltip/enabled/
- * Disabled
- * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/
- * Disable tooltip and show values on chart instead
- */
- enabled: true,
- /**
- * Enable or disable animation of the tooltip.
- *
- * @type {boolean}
- * @default true
- * @since 2.3.0
- */
- animation: svg,
- /**
- * The radius of the rounded border corners.
- *
- * @sample {highcharts} highcharts/tooltip/bordercolor-default/
- * 5px by default
- * @sample {highcharts} highcharts/tooltip/borderradius-0/
- * Square borders
- * @sample {highmaps} maps/tooltip/background-border/
- * Background and border demo
- */
- borderRadius: 3,
- /**
- * For series on a datetime axes, the date format in the tooltip's
- * header will by default be guessed based on the closest data points.
- * This member gives the default string representations used for
- * each unit. For an overview of the replacement codes, see
- * [dateFormat](/class-reference/Highcharts#dateFormat).
- *
- * @see [xAxis.dateTimeLabelFormats](#xAxis.dateTimeLabelFormats)
- *
- * @type {Highcharts.Dictionary<string>}
- * @product highcharts highstock gantt
- */
- dateTimeLabelFormats: {
- /** @internal */
- millisecond: '%A, %b %e, %H:%M:%S.%L',
- /** @internal */
- second: '%A, %b %e, %H:%M:%S',
- /** @internal */
- minute: '%A, %b %e, %H:%M',
- /** @internal */
- hour: '%A, %b %e, %H:%M',
- /** @internal */
- day: '%A, %b %e, %Y',
- /** @internal */
- week: 'Week from %A, %b %e, %Y',
- /** @internal */
- month: '%B %Y',
- /** @internal */
- year: '%Y'
- },
- /**
- * A string to append to the tooltip format.
- *
- * @sample {highcharts} highcharts/tooltip/footerformat/
- * A table for value alignment
- * @sample {highmaps} maps/tooltip/format/
- * Format demo
- *
- * @since 2.2
- */
- footerFormat: '',
- /**
- * Padding inside the tooltip, in pixels.
- *
- * @since 5.0.0
- */
- padding: 8,
- /**
- * Proximity snap for graphs or single points. It defaults to 10 for
- * mouse-powered devices and 25 for touch devices.
- *
- * Note that in most cases the whole plot area captures the mouse
- * movement, and in these cases `tooltip.snap` doesn't make sense. This
- * applies when [stickyTracking](#plotOptions.series.stickyTracking)
- * is `true` (default) and when the tooltip is [shared](#tooltip.shared)
- * or [split](#tooltip.split).
- *
- * @sample {highcharts} highcharts/tooltip/bordercolor-default/
- * 10 px by default
- * @sample {highcharts} highcharts/tooltip/snap-50/
- * 50 px on graph
- *
- * @type {number}
- * @default 10/25
- * @since 1.2.0
- * @product highcharts highstock
- */
- snap: isTouchDevice ? 25 : 10,
- /**
- * The HTML of the tooltip header line. Variables are enclosed by
- * curly brackets. Available variables are `point.key`, `series.name`,
- * `series.color` and other members from the `point` and `series`
- * objects. The `point.key` variable contains the category name, x
- * value or datetime string depending on the type of axis. For datetime
- * axes, the `point.key` date format can be set using
- * `tooltip.xDateFormat`.
- *
- * @sample {highcharts} highcharts/tooltip/footerformat/
- * An HTML table in the tooltip
- * @sample {highstock} highcharts/tooltip/footerformat/
- * An HTML table in the tooltip
- * @sample {highmaps} maps/tooltip/format/
- * Format demo
- *
- * @type {string}
- * @apioption tooltip.headerFormat
- */
- headerFormat: '<span style="font-size: 10px">{point.key}</span><br/>',
- /**
- * The HTML of the null point's line in the tooltip. Works analogously
- * to [pointFormat](#tooltip.pointFormat).
- *
- * @sample {highcharts} highcharts/plotoptions/series-nullformat
- * Format data label and tooltip for null point.
- *
- * @type {string}
- * @apioption tooltip.nullFormat
- */
- /**
- * The HTML of the point's line in the tooltip. Variables are enclosed
- * by curly brackets. Available variables are point.x, point.y, series.
- * name and series.color and other properties on the same form.
- * Furthermore, `point.y` can be extended by the `tooltip.valuePrefix`
- * and `tooltip.valueSuffix` variables. This can also be overridden for
- * each series, which makes it a good hook for displaying units.
- *
- * In styled mode, the dot is colored by a class name rather
- * than the point color.
- *
- * @sample {highcharts} highcharts/tooltip/pointformat/
- * A different point format with value suffix
- * @sample {highmaps} maps/tooltip/format/
- * Format demo
- *
- * @type {string}
- * @since 2.2
- * @apioption tooltip.pointFormat
- */
- pointFormat: '<span style="color:{point.color}">\u25CF</span> {series.name}: <b>{point.y}</b><br/>',
- /**
- * The background color or gradient for the tooltip.
- *
- * In styled mode, the stroke width is set in the
- * `.highcharts-tooltip-box` class.
- *
- * @sample {highcharts} highcharts/tooltip/backgroundcolor-solid/
- * Yellowish background
- * @sample {highcharts} highcharts/tooltip/backgroundcolor-gradient/
- * Gradient
- * @sample {highcharts} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- * @sample {highstock} stock/tooltip/general/
- * Custom tooltip
- * @sample {highstock} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- * @sample {highmaps} maps/tooltip/background-border/
- * Background and border demo
- * @sample {highmaps} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- backgroundColor: color('#f7f7f7')
- .setOpacity(0.85).get(),
- /**
- * The pixel width of the tooltip border.
- *
- * In styled mode, the stroke width is set in the
- * `.highcharts-tooltip-box` class.
- *
- * @sample {highcharts} highcharts/tooltip/bordercolor-default/
- * 2px by default
- * @sample {highcharts} highcharts/tooltip/borderwidth/
- * No border (shadow only)
- * @sample {highcharts} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- * @sample {highstock} stock/tooltip/general/
- * Custom tooltip
- * @sample {highstock} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- * @sample {highmaps} maps/tooltip/background-border/
- * Background and border demo
- * @sample {highmaps} highcharts/css/tooltip-border-background/
- * Tooltip in styled mode
- */
- borderWidth: 1,
- /**
- * Whether to apply a drop shadow to the tooltip.
- *
- * @sample {highcharts} highcharts/tooltip/bordercolor-default/
- * True by default
- * @sample {highcharts} highcharts/tooltip/shadow/
- * False
- * @sample {highmaps} maps/tooltip/positioner/
- * Fixed tooltip position, border and shadow disabled
- *
- * @type {boolean|Highcharts.ShadowOptionsObject}
- */
- shadow: true,
- /**
- * CSS styles for the tooltip. The tooltip can also be styled through
- * the CSS class `.highcharts-tooltip`.
- *
- * Note that the default `pointerEvents` style makes the tooltip ignore
- * mouse events, so in order to use clickable tooltips, this value must
- * be set to `auto`.
- *
- * @sample {highcharts} highcharts/tooltip/style/
- * Greater padding, bold text
- *
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @internal */
- color: '#333333',
- /** @internal */
- cursor: 'default',
- /** @internal */
- fontSize: '12px',
- /** @internal */
- whiteSpace: 'nowrap'
- }
- },
- /**
- * Highchart by default puts a credits label in the lower right corner
- * of the chart. This can be changed using these options.
- */
- credits: {
- /**
- * Credits for map source to be concatenated with conventional credit
- * text. By default this is a format string that collects copyright
- * information from the map if available.
- *
- * @see [mapTextFull](#credits.mapTextFull)
- * @see [text](#credits.text)
- *
- * @type {string}
- * @default \u00a9 <a href="{geojson.copyrightUrl}">{geojson.copyrightShort}</a>
- * @since 4.2.2
- * @product highmaps
- * @apioption credits.mapText
- */
- /**
- * Detailed credits for map source to be displayed on hover of credits
- * text. By default this is a format string that collects copyright
- * information from the map if available.
- *
- * @see [mapText](#credits.mapText)
- * @see [text](#credits.text)
- *
- * @type {string}
- * @default {geojson.copyright}
- * @since 4.2.2
- * @product highmaps
- * @apioption credits.mapTextFull
- */
- /**
- * Whether to show the credits text.
- *
- * @sample {highcharts} highcharts/credits/enabled-false/
- * Credits disabled
- * @sample {highstock} stock/credits/enabled/
- * Credits disabled
- * @sample {highmaps} maps/credits/enabled-false/
- * Credits disabled
- */
- enabled: true,
- /**
- * The URL for the credits label.
- *
- * @sample {highcharts} highcharts/credits/href/
- * Custom URL and text
- * @sample {highmaps} maps/credits/customized/
- * Custom URL and text
- */
- href: 'https://www.highcharts.com?credits',
- /**
- * Position configuration for the credits label.
- *
- * @sample {highcharts} highcharts/credits/position-left/
- * Left aligned
- * @sample {highcharts} highcharts/credits/position-left/
- * Left aligned
- * @sample {highmaps} maps/credits/customized/
- * Left aligned
- * @sample {highmaps} maps/credits/customized/
- * Left aligned
- *
- * @type {Highcharts.AlignObject}
- * @since 2.1
- */
- position: {
- /** @internal */
- align: 'right',
- /** @internal */
- x: -10,
- /** @internal */
- verticalAlign: 'bottom',
- /** @internal */
- y: -5
- },
- /**
- * CSS styles for the credits label.
- *
- * @see In styled mode, credits styles can be set with the
- * `.highcharts-credits` class.
- *
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @internal */
- cursor: 'pointer',
- /** @internal */
- color: '#999999',
- /** @internal */
- fontSize: '9px'
- },
- /**
- * The text for the credits label.
- *
- * @productdesc {highmaps}
- * If a map is loaded as GeoJSON, the text defaults to
- * `Highcharts @ {map-credits}`. Otherwise, it defaults to
- * `Highcharts.com`.
- *
- * @sample {highcharts} highcharts/credits/href/
- * Custom URL and text
- * @sample {highmaps} maps/credits/customized/
- * Custom URL and text
- */
- text: 'Highcharts.com'
- }
- };
- /* eslint-disable spaced-comment */
- '';
- /**
- * Global `Time` object with default options. Since v6.0.5, time settings can be
- * applied individually for each chart. If no individual settings apply, this
- * `Time` object is shared by all instances.
- *
- * @name Highcharts.time
- * @type {Highcharts.Time}
- */
- H.time = new Time(merge(H.defaultOptions.global, H.defaultOptions.time));
- /**
- * Formats a JavaScript date timestamp (milliseconds since Jan 1st 1970) into a
- * human readable date string. The format is a subset of the formats for PHP's
- * [strftime](https://www.php.net/manual/en/function.strftime.php) function.
- * Additional formats can be given in the {@link Highcharts.dateFormats} hook.
- *
- * Since v6.0.5, all internal dates are formatted through the
- * {@link Highcharts.Chart#time} instance to respect chart-level time settings.
- * The `Highcharts.dateFormat` function only reflects global time settings set
- * with `setOptions`.
- *
- * Supported format keys:
- * - `%a`: Short weekday, like 'Mon'
- * - `%A`: Long weekday, like 'Monday'
- * - `%d`: Two digit day of the month, 01 to 31
- * - `%e`: Day of the month, 1 through 31
- * - `%w`: Day of the week, 0 through 6
- * - `%b`: Short month, like 'Jan'
- * - `%B`: Long month, like 'January'
- * - `%m`: Two digit month number, 01 through 12
- * - `%y`: Two digits year, like 09 for 2009
- * - `%Y`: Four digits year, like 2009
- * - `%H`: Two digits hours in 24h format, 00 through 23
- * - `%k`: Hours in 24h format, 0 through 23
- * - `%I`: Two digits hours in 12h format, 00 through 11
- * - `%l`: Hours in 12h format, 1 through 12
- * - `%M`: Two digits minutes, 00 through 59
- * - `%p`: Upper case AM or PM
- * - `%P`: Lower case AM or PM
- * - `%S`: Two digits seconds, 00 through 59
- * - `%L`: Milliseconds (naming from Ruby)
- *
- * @function Highcharts.dateFormat
- *
- * @param {string} format
- * The desired format where various time representations are prefixed
- * with `%`.
- *
- * @param {number} timestamp
- * The JavaScript timestamp.
- *
- * @param {boolean} [capitalize=false]
- * Upper case first letter in the return.
- *
- * @return {string}
- * The formatted date.
- */
- H.dateFormat = function (format, timestamp, capitalize) {
- return H.time.dateFormat(format, timestamp, capitalize);
- };
- var optionsModule = {
- dateFormat: H.dateFormat,
- defaultOptions: H.defaultOptions,
- time: H.time
- };
- return optionsModule;
- });
- _registerModule(_modules, 'Core/Axis/Axis.js', [_modules['Core/Color.js'], _modules['Core/Globals.js'], _modules['Core/Axis/Tick.js'], _modules['Core/Utilities.js'], _modules['Core/Options.js']], function (Color, H, Tick, U, O) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- animObject = U.animObject,
- arrayMax = U.arrayMax,
- arrayMin = U.arrayMin,
- clamp = U.clamp,
- correctFloat = U.correctFloat,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- error = U.error,
- extend = U.extend,
- fireEvent = U.fireEvent,
- format = U.format,
- getMagnitude = U.getMagnitude,
- isArray = U.isArray,
- isFunction = U.isFunction,
- isNumber = U.isNumber,
- isString = U.isString,
- merge = U.merge,
- normalizeTickInterval = U.normalizeTickInterval,
- objectEach = U.objectEach,
- pick = U.pick,
- relativeLength = U.relativeLength,
- removeEvent = U.removeEvent,
- splat = U.splat,
- syncTimeout = U.syncTimeout;
- /**
- * Options for the path on the Axis to be calculated.
- * @interface Highcharts.AxisPlotLinePathOptionsObject
- */ /**
- * Axis value.
- * @name Highcharts.AxisPlotLinePathOptionsObject#value
- * @type {number|undefined}
- */ /**
- * Line width used for calculation crisp line coordinates. Defaults to 1.
- * @name Highcharts.AxisPlotLinePathOptionsObject#lineWidth
- * @type {number|undefined}
- */ /**
- * If `false`, the function will return null when it falls outside the axis
- * bounds. If `true`, the function will return a path aligned to the plot area
- * sides if it falls outside. If `pass`, it will return a path outside.
- * @name Highcharts.AxisPlotLinePathOptionsObject#force
- * @type {string|boolean|undefined}
- */ /**
- * Used in Highstock. When `true`, plot paths (crosshair, plotLines, gridLines)
- * will be rendered on all axes when defined on the first axis.
- * @name Highcharts.AxisPlotLinePathOptionsObject#acrossPanes
- * @type {boolean|undefined}
- */ /**
- * Use old coordinates (for resizing and rescaling).
- * If not set, defaults to `false`.
- * @name Highcharts.AxisPlotLinePathOptionsObject#old
- * @type {boolean|undefined}
- */ /**
- * If given, return the plot line path of a pixel position on the axis.
- * @name Highcharts.AxisPlotLinePathOptionsObject#translatedValue
- * @type {number|undefined}
- */ /**
- * Used in Polar axes. Reverse the positions for concatenation of polygonal
- * plot bands
- * @name Highcharts.AxisPlotLinePathOptionsObject#reverse
- * @type {boolean|undefined}
- */
- /**
- * Options for crosshairs on axes.
- *
- * @product highstock
- *
- * @typedef {Highcharts.XAxisCrosshairOptions|Highcharts.YAxisCrosshairOptions} Highcharts.AxisCrosshairOptions
- */
- /**
- * @typedef {"navigator"|"pan"|"rangeSelectorButton"|"rangeSelectorInput"|"scrollbar"|"traverseUpButton"|"zoom"} Highcharts.AxisExtremesTriggerValue
- */
- /**
- * @callback Highcharts.AxisEventCallbackFunction
- *
- * @param {Highcharts.Axis} this
- */
- /**
- * @callback Highcharts.AxisLabelsFormatterCallbackFunction
- *
- * @param {Highcharts.AxisLabelsFormatterContextObject<number>} this
- *
- * @param {Highcharts.AxisLabelsFormatterContextObject<string>} that
- *
- * @return {string}
- */
- /**
- * @interface Highcharts.AxisLabelsFormatterContextObject<T>
- */ /**
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#axis
- * @type {Highcharts.Axis}
- */ /**
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#chart
- * @type {Highcharts.Chart}
- */ /**
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#isFirst
- * @type {boolean}
- */ /**
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#isLast
- * @type {boolean}
- */ /**
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#pos
- * @type {number}
- */ /**
- * This can be either a numeric value or a category string.
- * @name Highcharts.AxisLabelsFormatterContextObject<T>#value
- * @type {T}
- */
- /**
- * Options for axes.
- *
- * @typedef {Highcharts.XAxisOptions|Highcharts.YAxisOptions|Highcharts.ZAxisOptions} Highcharts.AxisOptions
- */
- /**
- * @callback Highcharts.AxisPointBreakEventCallbackFunction
- *
- * @param {Highcharts.Axis} this
- *
- * @param {Highcharts.AxisPointBreakEventObject} evt
- */
- /**
- * @interface Highcharts.AxisPointBreakEventObject
- */ /**
- * @name Highcharts.AxisPointBreakEventObject#brk
- * @type {Highcharts.Dictionary<number>}
- */ /**
- * @name Highcharts.AxisPointBreakEventObject#point
- * @type {Highcharts.Point}
- */ /**
- * @name Highcharts.AxisPointBreakEventObject#preventDefault
- * @type {Function}
- */ /**
- * @name Highcharts.AxisPointBreakEventObject#target
- * @type {Highcharts.SVGElement}
- */ /**
- * @name Highcharts.AxisPointBreakEventObject#type
- * @type {"pointBreak"|"pointInBreak"}
- */
- /**
- * @callback Highcharts.AxisSetExtremesEventCallbackFunction
- *
- * @param {Highcharts.Axis} this
- *
- * @param {Highcharts.AxisSetExtremesEventObject} evt
- */
- /**
- * @interface Highcharts.AxisSetExtremesEventObject
- * @extends Highcharts.ExtremesObject
- */ /**
- * @name Highcharts.AxisSetExtremesEventObject#preventDefault
- * @type {Function}
- */ /**
- * @name Highcharts.AxisSetExtremesEventObject#target
- * @type {Highcharts.SVGElement}
- */ /**
- * @name Highcharts.AxisSetExtremesEventObject#trigger
- * @type {Highcharts.AxisExtremesTriggerValue|string}
- */ /**
- * @name Highcharts.AxisSetExtremesEventObject#type
- * @type {"setExtremes"}
- */
- /**
- * @callback Highcharts.AxisTickPositionerCallbackFunction
- *
- * @param {Highcharts.Axis} this
- *
- * @return {Highcharts.AxisTickPositionsArray}
- */
- /**
- * @interface Highcharts.AxisTickPositionsArray
- * @augments Array<number>
- */
- /**
- * @typedef {"high"|"low"|"middle"} Highcharts.AxisTitleAlignValue
- */
- /**
- * @typedef {Highcharts.XAxisTitleOptions|Highcharts.YAxisTitleOptions|Highcharts.ZAxisTitleOptions} Highcharts.AxisTitleOptions
- */
- /**
- * @typedef {"linear"|"logarithmic"|"datetime"|"category"|"treegrid"} Highcharts.AxisTypeValue
- */
- /**
- * The returned object literal from the {@link Highcharts.Axis#getExtremes}
- * function.
- *
- * @interface Highcharts.ExtremesObject
- */ /**
- * The maximum value of the axis' associated series.
- * @name Highcharts.ExtremesObject#dataMax
- * @type {number}
- */ /**
- * The minimum value of the axis' associated series.
- * @name Highcharts.ExtremesObject#dataMin
- * @type {number}
- */ /**
- * The maximum axis value, either automatic or set manually. If the `max` option
- * is not set, `maxPadding` is 0 and `endOnTick` is false, this value will be
- * the same as `dataMax`.
- * @name Highcharts.ExtremesObject#max
- * @type {number}
- */ /**
- * The minimum axis value, either automatic or set manually. If the `min` option
- * is not set, `minPadding` is 0 and `startOnTick` is false, this value will be
- * the same as `dataMin`.
- * @name Highcharts.ExtremesObject#min
- * @type {number}
- */ /**
- * The user defined maximum, either from the `max` option or from a zoom or
- * `setExtremes` action.
- * @name Highcharts.ExtremesObject#userMax
- * @type {number}
- */ /**
- * The user defined minimum, either from the `min` option or from a zoom or
- * `setExtremes` action.
- * @name Highcharts.ExtremesObject#userMin
- * @type {number}
- */
- /**
- * Formatter function for the text of a crosshair label.
- *
- * @callback Highcharts.XAxisCrosshairLabelFormatterCallbackFunction
- *
- * @param {Highcharts.Axis} this
- * Axis context
- *
- * @param {number} value
- * Y value of the data point
- *
- * @return {string}
- */
- var defaultOptions = O.defaultOptions;
- var deg2rad = H.deg2rad;
- /**
- * Create a new axis object. Called internally when instanciating a new chart or
- * adding axes by {@link Highcharts.Chart#addAxis}.
- *
- * A chart can have from 0 axes (pie chart) to multiples. In a normal, single
- * series cartesian chart, there is one X axis and one Y axis.
- *
- * The X axis or axes are referenced by {@link Highcharts.Chart.xAxis}, which is
- * an array of Axis objects. If there is only one axis, it can be referenced
- * through `chart.xAxis[0]`, and multiple axes have increasing indices. The same
- * pattern goes for Y axes.
- *
- * If you need to get the axes from a series object, use the `series.xAxis` and
- * `series.yAxis` properties. These are not arrays, as one series can only be
- * associated to one X and one Y axis.
- *
- * A third way to reference the axis programmatically is by `id`. Add an `id` in
- * the axis configuration options, and get the axis by
- * {@link Highcharts.Chart#get}.
- *
- * Configuration options for the axes are given in options.xAxis and
- * options.yAxis.
- *
- * @class
- * @name Highcharts.Axis
- *
- * @param {Highcharts.Chart} chart
- * The Chart instance to apply the axis on.
- *
- * @param {Highcharts.AxisOptions} userOptions
- * Axis options.
- */
- var Axis = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Axis(chart, userOptions) {
- this.alternateBands = void 0;
- this.bottom = void 0;
- this.categories = void 0;
- this.chart = void 0;
- this.closestPointRange = void 0;
- this.coll = void 0;
- this.hasNames = void 0;
- this.hasVisibleSeries = void 0;
- this.height = void 0;
- this.isLinked = void 0;
- this.labelEdge = void 0; // @todo
- this.labelFormatter = void 0;
- this.left = void 0;
- this.len = void 0;
- this.max = void 0;
- this.maxLabelLength = void 0;
- this.min = void 0;
- this.minorTickInterval = void 0;
- this.minorTicks = void 0;
- this.minPixelPadding = void 0;
- this.names = void 0;
- this.offset = void 0;
- this.oldMax = void 0;
- this.oldMin = void 0;
- this.options = void 0;
- this.overlap = void 0;
- this.paddedTicks = void 0;
- this.plotLinesAndBands = void 0;
- this.plotLinesAndBandsGroups = void 0;
- this.pointRange = void 0;
- this.pointRangePadding = void 0;
- this.pos = void 0;
- this.positiveValuesOnly = void 0;
- this.right = void 0;
- this.series = void 0;
- this.side = void 0;
- this.tickAmount = void 0;
- this.tickInterval = void 0;
- this.tickmarkOffset = void 0;
- this.tickPositions = void 0;
- this.tickRotCorr = void 0;
- this.ticks = void 0;
- this.top = void 0;
- this.transA = void 0;
- this.transB = void 0;
- this.translationSlope = void 0;
- this.userOptions = void 0;
- this.visible = void 0;
- this.width = void 0;
- this.zoomEnabled = void 0;
- this.init(chart, userOptions);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Overrideable function to initialize the axis.
- *
- * @see {@link Axis}
- *
- * @function Highcharts.Axis#init
- *
- * @param {Highcharts.Chart} chart
- * The Chart instance to apply the axis on.
- *
- * @param {Highcharts.AxisOptions} userOptions
- * Axis options.
- *
- * @fires Highcharts.Axis#event:afterInit
- * @fires Highcharts.Axis#event:init
- */
- Axis.prototype.init = function (chart, userOptions) {
- var isXAxis = userOptions.isX,
- axis = this;
- /**
- * The Chart that the axis belongs to.
- *
- * @name Highcharts.Axis#chart
- * @type {Highcharts.Chart}
- */
- axis.chart = chart;
- /**
- * Whether the axis is horizontal.
- *
- * @name Highcharts.Axis#horiz
- * @type {boolean|undefined}
- */
- axis.horiz = chart.inverted && !axis.isZAxis ? !isXAxis : isXAxis;
- /**
- * Whether the axis is the x-axis.
- *
- * @name Highcharts.Axis#isXAxis
- * @type {boolean|undefined}
- */
- axis.isXAxis = isXAxis;
- /**
- * The collection where the axis belongs, for example `xAxis`, `yAxis`
- * or `colorAxis`. Corresponds to properties on Chart, for example
- * {@link Chart.xAxis}.
- *
- * @name Highcharts.Axis#coll
- * @type {string}
- */
- axis.coll = axis.coll || (isXAxis ? 'xAxis' : 'yAxis');
- fireEvent(this, 'init', { userOptions: userOptions });
- axis.opposite = userOptions.opposite; // needed in setOptions
- /**
- * The side on which the axis is rendered. 0 is top, 1 is right, 2
- * is bottom and 3 is left.
- *
- * @name Highcharts.Axis#side
- * @type {number}
- */
- axis.side = userOptions.side || (axis.horiz ?
- (axis.opposite ? 0 : 2) : // top : bottom
- (axis.opposite ? 1 : 3)); // right : left
- /**
- * Current options for the axis after merge of defaults and user's
- * options.
- *
- * @name Highcharts.Axis#options
- * @type {Highcharts.AxisOptions}
- */
- axis.setOptions(userOptions);
- var options = this.options,
- type = options.type;
- axis.labelFormatter = (options.labels.formatter ||
- // can be overwritten by dynamic format
- axis.defaultLabelFormatter);
- /**
- * User's options for this axis without defaults.
- *
- * @name Highcharts.Axis#userOptions
- * @type {Highcharts.AxisOptions}
- */
- axis.userOptions = userOptions;
- axis.minPixelPadding = 0;
- /**
- * Whether the axis is reversed. Based on the `axis.reversed`,
- * option, but inverted charts have reversed xAxis by default.
- *
- * @name Highcharts.Axis#reversed
- * @type {boolean}
- */
- axis.reversed = options.reversed;
- axis.visible = options.visible !== false;
- axis.zoomEnabled = options.zoomEnabled !== false;
- // Initial categories
- axis.hasNames =
- type === 'category' || options.categories === true;
- /**
- * If categories are present for the axis, names are used instead of
- * numbers for that axis.
- *
- * Since Highcharts 3.0, categories can also be extracted by giving each
- * point a name and setting axis type to `category`. However, if you
- * have multiple series, best practice remains defining the `categories`
- * array.
- *
- * @see [xAxis.categories](/highcharts/xAxis.categories)
- *
- * @name Highcharts.Axis#categories
- * @type {Array<string>}
- * @readonly
- */
- axis.categories = options.categories || axis.hasNames;
- if (!axis.names) { // Preserve on update (#3830)
- axis.names = [];
- axis.names.keys = {};
- }
- // Placeholder for plotlines and plotbands groups
- axis.plotLinesAndBandsGroups = {};
- // Shorthand types
- axis.positiveValuesOnly = !!axis.logarithmic;
- // Flag, if axis is linked to another axis
- axis.isLinked = defined(options.linkedTo);
- /**
- * List of major ticks mapped by postition on axis.
- *
- * @see {@link Highcharts.Tick}
- *
- * @name Highcharts.Axis#ticks
- * @type {Highcharts.Dictionary<Highcharts.Tick>}
- */
- axis.ticks = {};
- axis.labelEdge = [];
- /**
- * List of minor ticks mapped by position on the axis.
- *
- * @see {@link Highcharts.Tick}
- *
- * @name Highcharts.Axis#minorTicks
- * @type {Highcharts.Dictionary<Highcharts.Tick>}
- */
- axis.minorTicks = {};
- // List of plotLines/Bands
- axis.plotLinesAndBands = [];
- // Alternate bands
- axis.alternateBands = {};
- // Axis metrics
- axis.len = 0;
- axis.minRange = axis.userMinRange = options.minRange || options.maxZoom;
- axis.range = options.range;
- axis.offset = options.offset || 0;
- /**
- * The maximum value of the axis. In a logarithmic axis, this is the
- * logarithm of the real value, and the real value can be obtained from
- * {@link Axis#getExtremes}.
- *
- * @name Highcharts.Axis#max
- * @type {number|null}
- */
- axis.max = null;
- /**
- * The minimum value of the axis. In a logarithmic axis, this is the
- * logarithm of the real value, and the real value can be obtained from
- * {@link Axis#getExtremes}.
- *
- * @name Highcharts.Axis#min
- * @type {number|null}
- */
- axis.min = null;
- /**
- * The processed crosshair options.
- *
- * @name Highcharts.Axis#crosshair
- * @type {boolean|Highcharts.AxisCrosshairOptions}
- */
- axis.crosshair = pick(options.crosshair, splat(chart.options.tooltip.crosshairs)[isXAxis ? 0 : 1], false);
- var events = axis.options.events;
- // Register. Don't add it again on Axis.update().
- if (chart.axes.indexOf(axis) === -1) { //
- if (isXAxis) { // #2713
- chart.axes.splice(chart.xAxis.length, 0, axis);
- }
- else {
- chart.axes.push(axis);
- }
- chart[axis.coll].push(axis);
- }
- /**
- * All series associated to the axis.
- *
- * @name Highcharts.Axis#series
- * @type {Array<Highcharts.Series>}
- */
- axis.series = axis.series || []; // populated by Series
- // Reversed axis
- if (chart.inverted &&
- !axis.isZAxis &&
- isXAxis &&
- typeof axis.reversed === 'undefined') {
- axis.reversed = true;
- }
- axis.labelRotation = axis.options.labels.rotation;
- // register event listeners
- objectEach(events, function (event, eventType) {
- if (isFunction(event)) {
- addEvent(axis, eventType, event);
- }
- });
- fireEvent(this, 'afterInit');
- };
- /**
- * Merge and set options.
- *
- * @private
- * @function Highcharts.Axis#setOptions
- *
- * @param {Highcharts.AxisOptions} userOptions
- * Axis options.
- *
- * @fires Highcharts.Axis#event:afterSetOptions
- */
- Axis.prototype.setOptions = function (userOptions) {
- this.options = merge(Axis.defaultOptions, (this.coll === 'yAxis') && Axis.defaultYAxisOptions, [
- Axis.defaultTopAxisOptions,
- Axis.defaultRightAxisOptions,
- Axis.defaultBottomAxisOptions,
- Axis.defaultLeftAxisOptions
- ][this.side], merge(
- // if set in setOptions (#1053):
- defaultOptions[this.coll], userOptions));
- fireEvent(this, 'afterSetOptions', { userOptions: userOptions });
- };
- /**
- * The default label formatter. The context is a special config object for
- * the label. In apps, use the
- * [labels.formatter](https://api.highcharts.com/highcharts/xAxis.labels.formatter)
- * instead, except when a modification is needed.
- *
- * @function Highcharts.Axis#defaultLabelFormatter
- *
- * @param {Highcharts.AxisLabelsFormatterContextObject<number>|Highcharts.AxisLabelsFormatterContextObject<string>} this
- * Formatter context of axis label.
- *
- * @return {string}
- * The formatted label content.
- */
- Axis.prototype.defaultLabelFormatter = function () {
- var axis = this.axis,
- value = isNumber(this.value) ? this.value : NaN,
- time = axis.chart.time,
- categories = axis.categories,
- dateTimeLabelFormat = this.dateTimeLabelFormat,
- lang = defaultOptions.lang,
- numericSymbols = lang.numericSymbols,
- numSymMagnitude = lang.numericSymbolMagnitude || 1000,
- i = numericSymbols && numericSymbols.length,
- multi,
- ret,
- formatOption = axis.options.labels.format,
- // make sure the same symbol is added for all labels on a linear
- // axis
- numericSymbolDetector = axis.logarithmic ?
- Math.abs(value) :
- axis.tickInterval;
- var chart = this.chart;
- var numberFormatter = chart.numberFormatter;
- if (formatOption) {
- ret = format(formatOption, this, chart);
- }
- else if (categories) {
- ret = "" + this.value;
- }
- else if (dateTimeLabelFormat) { // datetime axis
- ret = time.dateFormat(dateTimeLabelFormat, value);
- }
- else if (i && numericSymbolDetector >= 1000) {
- // Decide whether we should add a numeric symbol like k (thousands)
- // or M (millions). If we are to enable this in tooltip or other
- // places as well, we can move this logic to the numberFormatter and
- // enable it by a parameter.
- while (i-- && typeof ret === 'undefined') {
- multi = Math.pow(numSymMagnitude, i + 1);
- if (
- // Only accept a numeric symbol when the distance is more
- // than a full unit. So for example if the symbol is k, we
- // don't accept numbers like 0.5k.
- numericSymbolDetector >= multi &&
- // Accept one decimal before the symbol. Accepts 0.5k but
- // not 0.25k. How does this work with the previous?
- (value * 10) % multi === 0 &&
- numericSymbols[i] !== null &&
- value !== 0) { // #5480
- ret = numberFormatter(value / multi, -1) + numericSymbols[i];
- }
- }
- }
- if (typeof ret === 'undefined') {
- if (Math.abs(value) >= 10000) { // add thousands separators
- ret = numberFormatter(value, -1);
- }
- else { // small numbers
- ret = numberFormatter(value, -1, void 0, ''); // #2466
- }
- }
- return ret;
- };
- /**
- * Get the minimum and maximum for the series of each axis. The function
- * analyzes the axis series and updates `this.dataMin` and `this.dataMax`.
- *
- * @private
- * @function Highcharts.Axis#getSeriesExtremes
- *
- * @fires Highcharts.Axis#event:afterGetSeriesExtremes
- * @fires Highcharts.Axis#event:getSeriesExtremes
- */
- Axis.prototype.getSeriesExtremes = function () {
- var axis = this,
- chart = axis.chart,
- xExtremes;
- fireEvent(this, 'getSeriesExtremes', null, function () {
- axis.hasVisibleSeries = false;
- // Reset properties in case we're redrawing (#3353)
- axis.dataMin = axis.dataMax = axis.threshold = null;
- axis.softThreshold = !axis.isXAxis;
- if (axis.stacking) {
- axis.stacking.buildStacks();
- }
- // loop through this axis' series
- axis.series.forEach(function (series) {
- if (series.visible ||
- !chart.options.chart.ignoreHiddenSeries) {
- var seriesOptions = series.options,
- xData,
- threshold = seriesOptions.threshold,
- seriesDataMin,
- seriesDataMax;
- axis.hasVisibleSeries = true;
- // Validate threshold in logarithmic axes
- if (axis.positiveValuesOnly && threshold <= 0) {
- threshold = null;
- }
- // Get dataMin and dataMax for X axes
- if (axis.isXAxis) {
- xData = series.xData;
- if (xData.length) {
- var isPositive = function (number) { return number > 0; };
- xData = axis.logarithmic ?
- xData.filter(axis.validatePositiveValue) :
- xData;
- xExtremes = series.getXExtremes(xData);
- // If xData contains values which is not numbers,
- // then filter them out. To prevent performance hit,
- // we only do this after we have already found
- // seriesDataMin because in most cases all data is
- // valid. #5234.
- seriesDataMin = xExtremes.min;
- seriesDataMax = xExtremes.max;
- if (!isNumber(seriesDataMin) &&
- // #5010:
- !(seriesDataMin instanceof Date)) {
- xData = xData.filter(isNumber);
- xExtremes = series.getXExtremes(xData);
- // Do it again with valid data
- seriesDataMin = xExtremes.min;
- seriesDataMax = xExtremes.max;
- }
- if (xData.length) {
- axis.dataMin = Math.min(pick(axis.dataMin, seriesDataMin), seriesDataMin);
- axis.dataMax = Math.max(pick(axis.dataMax, seriesDataMax), seriesDataMax);
- }
- }
- // Get dataMin and dataMax for Y axes, as well as handle
- // stacking and processed data
- }
- else {
- // Get this particular series extremes
- var dataExtremes = series.applyExtremes();
- // Get the dataMin and dataMax so far. If percentage is
- // used, the min and max are always 0 and 100. If
- // seriesDataMin and seriesDataMax is null, then series
- // doesn't have active y data, we continue with nulls
- if (isNumber(dataExtremes.dataMin)) {
- seriesDataMin = dataExtremes.dataMin;
- axis.dataMin = Math.min(pick(axis.dataMin, seriesDataMin), seriesDataMin);
- }
- if (isNumber(dataExtremes.dataMax)) {
- seriesDataMax = dataExtremes.dataMax;
- axis.dataMax = Math.max(pick(axis.dataMax, seriesDataMax), seriesDataMax);
- }
- // Adjust to threshold
- if (defined(threshold)) {
- axis.threshold = threshold;
- }
- // If any series has a hard threshold, it takes
- // precedence
- if (!seriesOptions.softThreshold ||
- axis.positiveValuesOnly) {
- axis.softThreshold = false;
- }
- }
- }
- });
- });
- fireEvent(this, 'afterGetSeriesExtremes');
- };
- /**
- * Translate from axis value to pixel position on the chart, or back. Use
- * the `toPixels` and `toValue` functions in applications.
- *
- * @private
- * @function Highcharts.Axis#translate
- *
- * @param {number} val
- * TO-DO: parameter description
- *
- * @param {boolean|null} [backwards]
- * TO-DO: parameter description
- *
- * @param {boolean|null} [cvsCoord]
- * TO-DO: parameter description
- *
- * @param {boolean|null} [old]
- * TO-DO: parameter description
- *
- * @param {boolean} [handleLog]
- * TO-DO: parameter description
- *
- * @param {number} [pointPlacement]
- * TO-DO: parameter description
- *
- * @return {number|undefined}
- */
- Axis.prototype.translate = function (val, backwards, cvsCoord, old, handleLog, pointPlacement) {
- var axis = this.linkedParent || this, // #1417
- sign = 1,
- cvsOffset = 0,
- localA = old ? axis.oldTransA : axis.transA,
- localMin = old ? axis.oldMin : axis.min,
- returnValue = 0,
- minPixelPadding = axis.minPixelPadding,
- doPostTranslate = (axis.isOrdinal ||
- axis.brokenAxis && axis.brokenAxis.hasBreaks ||
- (axis.logarithmic && handleLog)) && axis.lin2val;
- if (!localA) {
- localA = axis.transA;
- }
- // In vertical axes, the canvas coordinates start from 0 at the top like
- // in SVG.
- if (cvsCoord) {
- sign *= -1; // canvas coordinates inverts the value
- cvsOffset = axis.len;
- }
- // Handle reversed axis
- if (axis.reversed) {
- sign *= -1;
- cvsOffset -= sign * (axis.sector || axis.len);
- }
- // From pixels to value
- if (backwards) { // reverse translation
- val = val * sign + cvsOffset;
- val -= minPixelPadding;
- // from chart pixel to value:
- returnValue = val / localA + localMin;
- if (doPostTranslate) { // log and ordinal axes
- returnValue = axis.lin2val(returnValue);
- }
- // From value to pixels
- }
- else {
- if (doPostTranslate) { // log and ordinal axes
- val = axis.val2lin(val);
- }
- returnValue = isNumber(localMin) ?
- (sign * (val - localMin) * localA +
- cvsOffset +
- (sign * minPixelPadding) +
- (isNumber(pointPlacement) ?
- localA * pointPlacement :
- 0)) :
- void 0;
- }
- return returnValue;
- };
- /**
- * Translate a value in terms of axis units into pixels within the chart.
- *
- * @function Highcharts.Axis#toPixels
- *
- * @param {number} value
- * A value in terms of axis units.
- *
- * @param {boolean} paneCoordinates
- * Whether to return the pixel coordinate relative to the chart or just the
- * axis/pane itself.
- *
- * @return {number}
- * Pixel position of the value on the chart or axis.
- */
- Axis.prototype.toPixels = function (value, paneCoordinates) {
- return this.translate(value, false, !this.horiz, null, true) +
- (paneCoordinates ? 0 : this.pos);
- };
- /**
- * Translate a pixel position along the axis to a value in terms of axis
- * units.
- *
- * @function Highcharts.Axis#toValue
- *
- * @param {number} pixel
- * The pixel value coordinate.
- *
- * @param {boolean} [paneCoordinates=false]
- * Whether the input pixel is relative to the chart or just the axis/pane
- * itself.
- *
- * @return {number}
- * The axis value.
- */
- Axis.prototype.toValue = function (pixel, paneCoordinates) {
- return this.translate(pixel - (paneCoordinates ? 0 : this.pos), true, !this.horiz, null, true);
- };
- /**
- * Create the path for a plot line that goes from the given value on
- * this axis, across the plot to the opposite side. Also used internally for
- * grid lines and crosshairs.
- *
- * @function Highcharts.Axis#getPlotLinePath
- *
- * @param {Highcharts.AxisPlotLinePathOptionsObject} options
- * Options for the path.
- *
- * @return {Highcharts.SVGPathArray|null}
- * The SVG path definition for the plot line.
- */
- Axis.prototype.getPlotLinePath = function (options) {
- var axis = this,
- chart = axis.chart,
- axisLeft = axis.left,
- axisTop = axis.top,
- old = options.old,
- value = options.value,
- translatedValue = options.translatedValue,
- lineWidth = options.lineWidth,
- force = options.force,
- x1,
- y1,
- x2,
- y2,
- cHeight = (old && chart.oldChartHeight) || chart.chartHeight,
- cWidth = (old && chart.oldChartWidth) || chart.chartWidth,
- skip,
- transB = axis.transB,
- evt;
- // eslint-disable-next-line valid-jsdoc
- /**
- * Check if x is between a and b. If not, either move to a/b
- * or skip, depending on the force parameter.
- * @private
- */
- function between(x, a, b) {
- if (force !== 'pass' && x < a || x > b) {
- if (force) {
- x = clamp(x, a, b);
- }
- else {
- skip = true;
- }
- }
- return x;
- }
- evt = {
- value: value,
- lineWidth: lineWidth,
- old: old,
- force: force,
- acrossPanes: options.acrossPanes,
- translatedValue: translatedValue
- };
- fireEvent(this, 'getPlotLinePath', evt, function (e) {
- translatedValue = pick(translatedValue, axis.translate(value, null, null, old));
- // Keep the translated value within sane bounds, and avoid Infinity
- // to fail the isNumber test (#7709).
- translatedValue = clamp(translatedValue, -1e5, 1e5);
- x1 = x2 = Math.round(translatedValue + transB);
- y1 = y2 = Math.round(cHeight - translatedValue - transB);
- if (!isNumber(translatedValue)) { // no min or max
- skip = true;
- force = false; // #7175, don't force it when path is invalid
- }
- else if (axis.horiz) {
- y1 = axisTop;
- y2 = cHeight - axis.bottom;
- x1 = x2 = between(x1, axisLeft, axisLeft + axis.width);
- }
- else {
- x1 = axisLeft;
- x2 = cWidth - axis.right;
- y1 = y2 = between(y1, axisTop, axisTop + axis.height);
- }
- e.path = skip && !force ?
- null :
- chart.renderer.crispLine([['M', x1, y1], ['L', x2, y2]], lineWidth || 1);
- });
- return evt.path;
- };
- /**
- * Internal function to et the tick positions of a linear axis to round
- * values like whole tens or every five.
- *
- * @function Highcharts.Axis#getLinearTickPositions
- *
- * @param {number} tickInterval
- * The normalized tick interval.
- *
- * @param {number} min
- * Axis minimum.
- *
- * @param {number} max
- * Axis maximum.
- *
- * @return {Array<number>}
- * An array of axis values where ticks should be placed.
- */
- Axis.prototype.getLinearTickPositions = function (tickInterval, min, max) {
- var pos,
- lastPos,
- roundedMin = correctFloat(Math.floor(min / tickInterval) * tickInterval),
- roundedMax = correctFloat(Math.ceil(max / tickInterval) * tickInterval),
- tickPositions = [],
- precision;
- // When the precision is higher than what we filter out in
- // correctFloat, skip it (#6183).
- if (correctFloat(roundedMin + tickInterval) === roundedMin) {
- precision = 20;
- }
- // For single points, add a tick regardless of the relative position
- // (#2662, #6274)
- if (this.single) {
- return [min];
- }
- // Populate the intermediate values
- pos = roundedMin;
- while (pos <= roundedMax) {
- // Place the tick on the rounded value
- tickPositions.push(pos);
- // Always add the raw tickInterval, not the corrected one.
- pos = correctFloat(pos + tickInterval, precision);
- // If the interval is not big enough in the current min - max range
- // to actually increase the loop variable, we need to break out to
- // prevent endless loop. Issue #619
- if (pos === lastPos) {
- break;
- }
- // Record the last value
- lastPos = pos;
- }
- return tickPositions;
- };
- /**
- * Resolve the new minorTicks/minorTickInterval options into the legacy
- * loosely typed minorTickInterval option.
- *
- * @function Highcharts.Axis#getMinorTickInterval
- *
- * @return {number|"auto"|null}
- */
- Axis.prototype.getMinorTickInterval = function () {
- var options = this.options;
- if (options.minorTicks === true) {
- return pick(options.minorTickInterval, 'auto');
- }
- if (options.minorTicks === false) {
- return null;
- }
- return options.minorTickInterval;
- };
- /**
- * Internal function to return the minor tick positions. For logarithmic
- * axes, the same logic as for major ticks is reused.
- *
- * @function Highcharts.Axis#getMinorTickPositions
- *
- * @return {Array<number>}
- * An array of axis values where ticks should be placed.
- */
- Axis.prototype.getMinorTickPositions = function () {
- var axis = this,
- options = axis.options,
- tickPositions = axis.tickPositions,
- minorTickInterval = axis.minorTickInterval,
- minorTickPositions = [],
- pos,
- pointRangePadding = axis.pointRangePadding || 0,
- min = axis.min - pointRangePadding, // #1498
- max = axis.max + pointRangePadding, // #1498
- range = max - min;
- // If minor ticks get too dense, they are hard to read, and may cause
- // long running script. So we don't draw them.
- if (range && range / minorTickInterval < axis.len / 3) { // #3875
- var logarithmic_1 = axis.logarithmic;
- if (logarithmic_1) {
- // For each interval in the major ticks, compute the minor ticks
- // separately.
- this.paddedTicks.forEach(function (_pos, i, paddedTicks) {
- if (i) {
- minorTickPositions.push.apply(minorTickPositions, logarithmic_1.getLogTickPositions(minorTickInterval, paddedTicks[i - 1], paddedTicks[i], true));
- }
- });
- }
- else if (axis.dateTime &&
- this.getMinorTickInterval() === 'auto') { // #1314
- minorTickPositions = minorTickPositions.concat(axis.getTimeTicks(axis.dateTime.normalizeTimeTickInterval(minorTickInterval), min, max, options.startOfWeek));
- }
- else {
- for (pos = min + (tickPositions[0] - min) % minorTickInterval; pos <= max; pos += minorTickInterval) {
- // Very, very, tight grid lines (#5771)
- if (pos === minorTickPositions[0]) {
- break;
- }
- minorTickPositions.push(pos);
- }
- }
- }
- if (minorTickPositions.length !== 0) {
- axis.trimTicks(minorTickPositions); // #3652 #3743 #1498 #6330
- }
- return minorTickPositions;
- };
- /**
- * Adjust the min and max for the minimum range. Keep in mind that the
- * series data is not yet processed, so we don't have information on data
- * cropping and grouping, or updated `axis.pointRange` or
- * `series.pointRange`. The data can't be processed until we have finally
- * established min and max.
- *
- * @private
- * @function Highcharts.Axis#adjustForMinRange
- */
- Axis.prototype.adjustForMinRange = function () {
- var axis = this,
- options = axis.options,
- min = axis.min,
- max = axis.max,
- log = axis.logarithmic,
- zoomOffset,
- spaceAvailable,
- closestDataRange,
- i,
- distance,
- xData,
- loopLength,
- minArgs,
- maxArgs,
- minRange;
- // Set the automatic minimum range based on the closest point distance
- if (axis.isXAxis &&
- typeof axis.minRange === 'undefined' &&
- !log) {
- if (defined(options.min) || defined(options.max)) {
- axis.minRange = null; // don't do this again
- }
- else {
- // Find the closest distance between raw data points, as opposed
- // to closestPointRange that applies to processed points
- // (cropped and grouped)
- axis.series.forEach(function (series) {
- xData = series.xData;
- loopLength = series.xIncrement ? 1 : xData.length - 1;
- for (i = loopLength; i > 0; i--) {
- distance = xData[i] - xData[i - 1];
- if (typeof closestDataRange === 'undefined' ||
- distance < closestDataRange) {
- closestDataRange = distance;
- }
- }
- });
- axis.minRange = Math.min(closestDataRange * 5, axis.dataMax - axis.dataMin);
- }
- }
- // if minRange is exceeded, adjust
- if (max - min < axis.minRange) {
- spaceAvailable =
- axis.dataMax - axis.dataMin >=
- axis.minRange;
- minRange = axis.minRange;
- zoomOffset = (minRange - max + min) / 2;
- // if min and max options have been set, don't go beyond it
- minArgs = [
- min - zoomOffset,
- pick(options.min, min - zoomOffset)
- ];
- // If space is available, stay within the data range
- if (spaceAvailable) {
- minArgs[2] = axis.logarithmic ?
- axis.logarithmic.log2lin(axis.dataMin) :
- axis.dataMin;
- }
- min = arrayMax(minArgs);
- maxArgs = [
- min + minRange,
- pick(options.max, min + minRange)
- ];
- // If space is availabe, stay within the data range
- if (spaceAvailable) {
- maxArgs[2] = log ?
- log.log2lin(axis.dataMax) :
- axis.dataMax;
- }
- max = arrayMin(maxArgs);
- // now if the max is adjusted, adjust the min back
- if (max - min < minRange) {
- minArgs[0] = max - minRange;
- minArgs[1] = pick(options.min, max - minRange);
- min = arrayMax(minArgs);
- }
- }
- // Record modified extremes
- axis.min = min;
- axis.max = max;
- };
- // eslint-disable-next-line valid-jsdoc
- /**
- * Find the closestPointRange across all series.
- *
- * @private
- * @function Highcharts.Axis#getClosest
- */
- Axis.prototype.getClosest = function () {
- var ret;
- if (this.categories) {
- ret = 1;
- }
- else {
- this.series.forEach(function (series) {
- var seriesClosest = series.closestPointRange,
- visible = series.visible ||
- !series.chart.options.chart.ignoreHiddenSeries;
- if (!series.noSharedTooltip &&
- defined(seriesClosest) &&
- visible) {
- ret = defined(ret) ?
- Math.min(ret, seriesClosest) :
- seriesClosest;
- }
- });
- }
- return ret;
- };
- /**
- * When a point name is given and no x, search for the name in the existing
- * categories, or if categories aren't provided, search names or create a
- * new category (#2522).
- * @private
- * @function Highcharts.Axis#nameToX
- *
- * @param {Highcharts.Point} point
- * The point to inspect.
- *
- * @return {number}
- * The X value that the point is given.
- */
- Axis.prototype.nameToX = function (point) {
- var explicitCategories = isArray(this.categories),
- names = explicitCategories ? this.categories : this.names,
- nameX = point.options.x,
- x;
- point.series.requireSorting = false;
- if (!defined(nameX)) {
- nameX = this.options.uniqueNames === false ?
- point.series.autoIncrement() :
- (explicitCategories ?
- names.indexOf(point.name) :
- pick(names.keys[point.name], -1));
- }
- if (nameX === -1) { // Not found in currenct categories
- if (!explicitCategories) {
- x = names.length;
- }
- }
- else {
- x = nameX;
- }
- // Write the last point's name to the names array
- if (typeof x !== 'undefined') {
- this.names[x] = point.name;
- // Backwards mapping is much faster than array searching (#7725)
- this.names.keys[point.name] = x;
- }
- return x;
- };
- /**
- * When changes have been done to series data, update the axis.names.
- *
- * @private
- * @function Highcharts.Axis#updateNames
- */
- Axis.prototype.updateNames = function () {
- var axis = this,
- names = this.names,
- i = names.length;
- if (i > 0) {
- Object.keys(names.keys).forEach(function (key) {
- delete (names.keys)[key];
- });
- names.length = 0;
- this.minRange = this.userMinRange; // Reset
- (this.series || []).forEach(function (series) {
- // Reset incrementer (#5928)
- series.xIncrement = null;
- // When adding a series, points are not yet generated
- if (!series.points || series.isDirtyData) {
- // When we're updating the series with data that is longer
- // than it was, and cropThreshold is passed, we need to make
- // sure that the axis.max is increased _before_ running the
- // premature processData. Otherwise this early iteration of
- // processData will crop the points to axis.max, and the
- // names array will be too short (#5857).
- axis.max = Math.max(axis.max, series.xData.length - 1);
- series.processData();
- series.generatePoints();
- }
- series.data.forEach(function (point, i) {
- var x;
- if (point &&
- point.options &&
- typeof point.name !== 'undefined' // #9562
- ) {
- x = axis.nameToX(point);
- if (typeof x !== 'undefined' && x !== point.x) {
- point.x = x;
- series.xData[i] = x;
- }
- }
- });
- });
- }
- };
- /**
- * Update translation information.
- *
- * @private
- * @function Highcharts.Axis#setAxisTranslation
- *
- * @param {boolean} [saveOld]
- * TO-DO: parameter description
- *
- * @fires Highcharts.Axis#event:afterSetAxisTranslation
- */
- Axis.prototype.setAxisTranslation = function (saveOld) {
- var axis = this,
- range = axis.max - axis.min,
- pointRange = axis.axisPointRange || 0,
- closestPointRange,
- minPointOffset = 0,
- pointRangePadding = 0,
- linkedParent = axis.linkedParent,
- ordinalCorrection,
- hasCategories = !!axis.categories,
- transA = axis.transA,
- isXAxis = axis.isXAxis;
- // Adjust translation for padding. Y axis with categories need to go
- // through the same (#1784).
- if (isXAxis || hasCategories || pointRange) {
- // Get the closest points
- closestPointRange = axis.getClosest();
- if (linkedParent) {
- minPointOffset = linkedParent.minPointOffset;
- pointRangePadding = linkedParent.pointRangePadding;
- }
- else {
- axis.series.forEach(function (series) {
- var seriesPointRange = hasCategories ?
- 1 :
- (isXAxis ?
- pick(series.options.pointRange,
- closestPointRange, 0) :
- (axis.axisPointRange || 0)), // #2806
- pointPlacement = series.options.pointPlacement;
- pointRange = Math.max(pointRange, seriesPointRange);
- if (!axis.single || hasCategories) {
- // TODO: series should internally set x- and y-
- // pointPlacement to simplify this logic.
- var isPointPlacementAxis = series.is('xrange') ? !isXAxis : isXAxis;
- // minPointOffset is the value padding to the left of
- // the axis in order to make room for points with a
- // pointRange, typically columns. When the
- // pointPlacement option is 'between' or 'on', this
- // padding does not apply.
- minPointOffset = Math.max(minPointOffset, isPointPlacementAxis && isString(pointPlacement) ?
- 0 :
- seriesPointRange / 2);
- // Determine the total padding needed to the length of
- // the axis to make room for the pointRange. If the
- // series' pointPlacement is 'on', no padding is added.
- pointRangePadding = Math.max(pointRangePadding, isPointPlacementAxis && pointPlacement === 'on' ?
- 0 :
- seriesPointRange);
- }
- });
- }
- // Record minPointOffset and pointRangePadding
- ordinalCorrection = axis.ordinal && axis.ordinal.slope && closestPointRange ?
- axis.ordinal.slope / closestPointRange :
- 1; // #988, #1853
- axis.minPointOffset = minPointOffset =
- minPointOffset * ordinalCorrection;
- axis.pointRangePadding =
- pointRangePadding = pointRangePadding * ordinalCorrection;
- // pointRange means the width reserved for each point, like in a
- // column chart
- axis.pointRange = Math.min(pointRange, axis.single && hasCategories ? 1 : range);
- // closestPointRange means the closest distance between points. In
- // columns it is mostly equal to pointRange, but in lines pointRange
- // is 0 while closestPointRange is some other value
- if (isXAxis) {
- axis.closestPointRange = closestPointRange;
- }
- }
- // Secondary values
- if (saveOld) {
- axis.oldTransA = transA;
- }
- axis.translationSlope = axis.transA = transA =
- axis.staticScale ||
- axis.len / ((range + pointRangePadding) || 1);
- // Translation addend
- axis.transB = axis.horiz ? axis.left : axis.bottom;
- axis.minPixelPadding = transA * minPointOffset;
- fireEvent(this, 'afterSetAxisTranslation');
- };
- /**
- * @private
- * @function Highcharts.Axis#minFromRange
- *
- * @return {number}
- */
- Axis.prototype.minFromRange = function () {
- var axis = this;
- return axis.max - axis.range;
- };
- /**
- * Set the tick positions to round values and optionally extend the extremes
- * to the nearest tick.
- *
- * @private
- * @function Highcharts.Axis#setTickInterval
- *
- * @param {boolean} secondPass
- * TO-DO: parameter description
- *
- * @fires Highcharts.Axis#event:foundExtremes
- */
- Axis.prototype.setTickInterval = function (secondPass) {
- var axis = this,
- chart = axis.chart,
- log = axis.logarithmic,
- options = axis.options,
- isXAxis = axis.isXAxis,
- isLinked = axis.isLinked,
- maxPadding = options.maxPadding,
- minPadding = options.minPadding,
- length,
- linkedParentExtremes,
- tickIntervalOption = options.tickInterval,
- minTickInterval,
- tickPixelIntervalOption = options.tickPixelInterval,
- categories = axis.categories,
- threshold = isNumber(axis.threshold) ? axis.threshold : null,
- softThreshold = axis.softThreshold,
- thresholdMin,
- thresholdMax,
- hardMin,
- hardMax;
- if (!axis.dateTime && !categories && !isLinked) {
- this.getTickAmount();
- }
- // Min or max set either by zooming/setExtremes or initial options
- hardMin = pick(axis.userMin, options.min);
- hardMax = pick(axis.userMax, options.max);
- // Linked axis gets the extremes from the parent axis
- if (isLinked) {
- axis.linkedParent = chart[axis.coll][options.linkedTo];
- linkedParentExtremes = axis.linkedParent.getExtremes();
- axis.min = pick(linkedParentExtremes.min, linkedParentExtremes.dataMin);
- axis.max = pick(linkedParentExtremes.max, linkedParentExtremes.dataMax);
- if (options.type !== axis.linkedParent.options.type) {
- // Can't link axes of different type
- error(11, 1, chart);
- }
- // Initial min and max from the extreme data values
- }
- else {
- // Adjust to hard threshold
- if (softThreshold && defined(threshold)) {
- if (axis.dataMin >= threshold) {
- thresholdMin = threshold;
- minPadding = 0;
- }
- else if (axis.dataMax <= threshold) {
- thresholdMax = threshold;
- maxPadding = 0;
- }
- }
- axis.min = pick(hardMin, thresholdMin, axis.dataMin);
- axis.max = pick(hardMax, thresholdMax, axis.dataMax);
- }
- if (log) {
- if (axis.positiveValuesOnly &&
- !secondPass &&
- Math.min(axis.min, pick(axis.dataMin, axis.min)) <= 0) { // #978
- // Can't plot negative values on log axis
- error(10, 1, chart);
- }
- // The correctFloat cures #934, float errors on full tens. But it
- // was too aggressive for #4360 because of conversion back to lin,
- // therefore use precision 15.
- axis.min = correctFloat(log.log2lin(axis.min), 16);
- axis.max = correctFloat(log.log2lin(axis.max), 16);
- }
- // handle zoomed range
- if (axis.range && defined(axis.max)) {
- // #618, #6773:
- axis.userMin = axis.min = hardMin =
- Math.max(axis.dataMin, axis.minFromRange());
- axis.userMax = hardMax = axis.max;
- axis.range = null; // don't use it when running setExtremes
- }
- // Hook for Highstock Scroller. Consider combining with beforePadding.
- fireEvent(axis, 'foundExtremes');
- // Hook for adjusting this.min and this.max. Used by bubble series.
- if (axis.beforePadding) {
- axis.beforePadding();
- }
- // adjust min and max for the minimum range
- axis.adjustForMinRange();
- // Pad the values to get clear of the chart's edges. To avoid
- // tickInterval taking the padding into account, we do this after
- // computing tick interval (#1337).
- if (!categories &&
- !axis.axisPointRange &&
- !(axis.stacking && axis.stacking.usePercentage) &&
- !isLinked &&
- defined(axis.min) &&
- defined(axis.max)) {
- length = axis.max - axis.min;
- if (length) {
- if (!defined(hardMin) && minPadding) {
- axis.min -= length * minPadding;
- }
- if (!defined(hardMax) && maxPadding) {
- axis.max += length * maxPadding;
- }
- }
- }
- // Handle options for floor, ceiling, softMin and softMax (#6359)
- if (!isNumber(axis.userMin)) {
- if (isNumber(options.softMin) && options.softMin < axis.min) {
- axis.min = hardMin = options.softMin; // #6894
- }
- if (isNumber(options.floor)) {
- axis.min = Math.max(axis.min, options.floor);
- }
- }
- if (!isNumber(axis.userMax)) {
- if (isNumber(options.softMax) && options.softMax > axis.max) {
- axis.max = hardMax = options.softMax; // #6894
- }
- if (isNumber(options.ceiling)) {
- axis.max = Math.min(axis.max, options.ceiling);
- }
- }
- // When the threshold is soft, adjust the extreme value only if the data
- // extreme and the padded extreme land on either side of the threshold.
- // For example, a series of [0, 1, 2, 3] would make the yAxis add a tick
- // for -1 because of the default minPadding and startOnTick options.
- // This is prevented by the softThreshold option.
- if (softThreshold && defined(axis.dataMin)) {
- threshold = threshold || 0;
- if (!defined(hardMin) &&
- axis.min < threshold &&
- axis.dataMin >= threshold) {
- axis.min = axis.options.minRange ?
- Math.min(threshold, axis.max -
- axis.minRange) :
- threshold;
- }
- else if (!defined(hardMax) &&
- axis.max > threshold &&
- axis.dataMax <= threshold) {
- axis.max = axis.options.minRange ?
- Math.max(threshold, axis.min +
- axis.minRange) :
- threshold;
- }
- }
- // get tickInterval
- if (axis.min === axis.max ||
- typeof axis.min === 'undefined' ||
- typeof axis.max === 'undefined') {
- axis.tickInterval = 1;
- }
- else if (isLinked &&
- !tickIntervalOption &&
- tickPixelIntervalOption ===
- axis.linkedParent.options.tickPixelInterval) {
- axis.tickInterval = tickIntervalOption =
- axis.linkedParent.tickInterval;
- }
- else {
- axis.tickInterval = pick(tickIntervalOption, this.tickAmount ?
- ((axis.max - axis.min) /
- Math.max(this.tickAmount - 1, 1)) :
- void 0,
- // For categoried axis, 1 is default, for linear axis use
- // tickPix
- categories ?
- 1 :
- // don't let it be more than the data range
- (axis.max - axis.min) *
- tickPixelIntervalOption /
- Math.max(axis.len, tickPixelIntervalOption));
- }
- // Now we're finished detecting min and max, crop and group series data.
- // This is in turn needed in order to find tick positions in ordinal
- // axes.
- if (isXAxis && !secondPass) {
- axis.series.forEach(function (series) {
- series.processData(axis.min !== axis.oldMin || axis.max !== axis.oldMax);
- });
- }
- // set the translation factor used in translate function
- axis.setAxisTranslation(true);
- // hook for ordinal axes and radial axes
- fireEvent(this, 'initialAxisTranslation');
- // In column-like charts, don't cramp in more ticks than there are
- // points (#1943, #4184)
- if (axis.pointRange && !tickIntervalOption) {
- axis.tickInterval = Math.max(axis.pointRange, axis.tickInterval);
- }
- // Before normalizing the tick interval, handle minimum tick interval.
- // This applies only if tickInterval is not defined.
- minTickInterval = pick(options.minTickInterval,
- // In datetime axes, don't go below the data interval, except when
- // there are scatter-like series involved (#13369).
- axis.dateTime &&
- !axis.series.some(function (s) { return s.noSharedTooltip; }) ?
- axis.closestPointRange : 0);
- if (!tickIntervalOption && axis.tickInterval < minTickInterval) {
- axis.tickInterval = minTickInterval;
- }
- // for linear axes, get magnitude and normalize the interval
- if (!axis.dateTime && !axis.logarithmic && !tickIntervalOption) {
- axis.tickInterval = normalizeTickInterval(axis.tickInterval, void 0, getMagnitude(axis.tickInterval), pick(options.allowDecimals,
- // If the tick interval is greather than 0.5, avoid
- // decimals, as linear axes are often used to render
- // discrete values. #3363. If a tick amount is set, allow
- // decimals by default, as it increases the chances for a
- // good fit.
- axis.tickInterval < 0.5 || this.tickAmount !== void 0), !!this.tickAmount);
- }
- // Prevent ticks from getting so close that we can't draw the labels
- if (!this.tickAmount) {
- axis.tickInterval = axis.unsquish();
- }
- this.setTickPositions();
- };
- /**
- * Now we have computed the normalized tickInterval, get the tick positions.
- *
- * @private
- * @function Highcharts.Axis#setTickPositions
- *
- * @fires Highcharts.Axis#event:afterSetTickPositions
- */
- Axis.prototype.setTickPositions = function () {
- var axis = this,
- options = this.options,
- tickPositions,
- tickPositionsOption = options.tickPositions,
- minorTickIntervalOption = this.getMinorTickInterval(),
- tickPositioner = options.tickPositioner,
- hasVerticalPanning = this.hasVerticalPanning(),
- isColorAxis = this.coll === 'colorAxis',
- startOnTick = (isColorAxis || !hasVerticalPanning) && options.startOnTick,
- endOnTick = (isColorAxis || !hasVerticalPanning) && options.endOnTick;
- // Set the tickmarkOffset
- this.tickmarkOffset = (this.categories &&
- options.tickmarkPlacement === 'between' &&
- this.tickInterval === 1) ? 0.5 : 0; // #3202
- // get minorTickInterval
- this.minorTickInterval =
- minorTickIntervalOption === 'auto' &&
- this.tickInterval ?
- this.tickInterval / 5 :
- minorTickIntervalOption;
- // When there is only one point, or all points have the same value on
- // this axis, then min and max are equal and tickPositions.length is 0
- // or 1. In this case, add some padding in order to center the point,
- // but leave it with one tick. #1337.
- this.single =
- this.min === this.max &&
- defined(this.min) &&
- !this.tickAmount &&
- (
- // Data is on integer (#6563)
- parseInt(this.min, 10) === this.min ||
- // Between integers and decimals are not allowed (#6274)
- options.allowDecimals !== false);
- /**
- * Contains the current positions that are laid out on the axis. The
- * positions are numbers in terms of axis values. In a category axis
- * they are integers, in a datetime axis they are also integers, but
- * designating milliseconds.
- *
- * This property is read only - for modifying the tick positions, use
- * the `tickPositioner` callback or [axis.tickPositions(
- * https://api.highcharts.com/highcharts/xAxis.tickPositions) option
- * instead.
- *
- * @name Highcharts.Axis#tickPositions
- * @type {Highcharts.AxisTickPositionsArray|undefined}
- */
- this.tickPositions =
- // Find the tick positions. Work on a copy (#1565)
- tickPositions =
- (tickPositionsOption && tickPositionsOption.slice());
- if (!tickPositions) {
- // Too many ticks (#6405). Create a friendly warning and provide two
- // ticks so at least we can show the data series.
- if ((!axis.ordinal || !axis.ordinal.positions) &&
- ((this.max - this.min) /
- this.tickInterval >
- Math.max(2 * this.len, 200))) {
- tickPositions = [this.min, this.max];
- error(19, false, this.chart);
- }
- else if (axis.dateTime) {
- tickPositions = axis.getTimeTicks(axis.dateTime.normalizeTimeTickInterval(this.tickInterval, options.units), this.min, this.max, options.startOfWeek, axis.ordinal && axis.ordinal.positions, this.closestPointRange, true);
- }
- else if (axis.logarithmic) {
- tickPositions = axis.logarithmic.getLogTickPositions(this.tickInterval, this.min, this.max);
- }
- else {
- tickPositions = this.getLinearTickPositions(this.tickInterval, this.min, this.max);
- }
- // Too dense ticks, keep only the first and last (#4477)
- if (tickPositions.length > this.len) {
- tickPositions = [tickPositions[0], tickPositions.pop()];
- // Reduce doubled value (#7339)
- if (tickPositions[0] === tickPositions[1]) {
- tickPositions.length = 1;
- }
- }
- this.tickPositions = tickPositions;
- // Run the tick positioner callback, that allows modifying auto tick
- // positions.
- if (tickPositioner) {
- tickPositioner = tickPositioner.apply(axis, [this.min, this.max]);
- if (tickPositioner) {
- this.tickPositions = tickPositions = tickPositioner;
- }
- }
- }
- // Reset min/max or remove extremes based on start/end on tick
- this.paddedTicks = tickPositions.slice(0); // Used for logarithmic minor
- this.trimTicks(tickPositions, startOnTick, endOnTick);
- if (!this.isLinked) {
- // Substract half a unit (#2619, #2846, #2515, #3390),
- // but not in case of multiple ticks (#6897)
- if (this.single &&
- tickPositions.length < 2 &&
- !this.categories &&
- !this.series.some(function (s) {
- return (s.is('heatmap') && s.options.pointPlacement === 'between');
- })) {
- this.min -= 0.5;
- this.max += 0.5;
- }
- if (!tickPositionsOption && !tickPositioner) {
- this.adjustTickAmount();
- }
- }
- fireEvent(this, 'afterSetTickPositions');
- };
- /**
- * Handle startOnTick and endOnTick by either adapting to padding min/max or
- * rounded min/max. Also handle single data points.
- *
- * @private
- * @function Highcharts.Axis#trimTicks
- *
- * @param {Array<number>} tickPositions
- * TO-DO: parameter description
- *
- * @param {boolean} [startOnTick]
- * TO-DO: parameter description
- *
- * @param {boolean} [endOnTick]
- * TO-DO: parameter description
- */
- Axis.prototype.trimTicks = function (tickPositions, startOnTick, endOnTick) {
- var roundedMin = tickPositions[0],
- roundedMax = tickPositions[tickPositions.length - 1],
- minPointOffset = (!this.isOrdinal && this.minPointOffset) || 0; // (#12716)
- fireEvent(this, 'trimTicks');
- if (!this.isLinked) {
- if (startOnTick && roundedMin !== -Infinity) { // #6502
- this.min = roundedMin;
- }
- else {
- while (this.min - minPointOffset > tickPositions[0]) {
- tickPositions.shift();
- }
- }
- if (endOnTick) {
- this.max = roundedMax;
- }
- else {
- while (this.max + minPointOffset <
- tickPositions[tickPositions.length - 1]) {
- tickPositions.pop();
- }
- }
- // If no tick are left, set one tick in the middle (#3195)
- if (tickPositions.length === 0 &&
- defined(roundedMin) &&
- !this.options.tickPositions) {
- tickPositions.push((roundedMax + roundedMin) / 2);
- }
- }
- };
- /**
- * Check if there are multiple axes in the same pane.
- *
- * @private
- * @function Highcharts.Axis#alignToOthers
- *
- * @return {boolean|undefined}
- * True if there are other axes.
- */
- Axis.prototype.alignToOthers = function () {
- var axis = this,
- others = // Whether there is another axis to pair with this one
- {},
- hasOther,
- options = axis.options;
- if (
- // Only if alignTicks is true
- this.chart.options.chart.alignTicks !== false &&
- options.alignTicks !== false &&
- // Disabled when startOnTick or endOnTick are false (#7604)
- options.startOnTick !== false &&
- options.endOnTick !== false &&
- // Don't try to align ticks on a log axis, they are not evenly
- // spaced (#6021)
- !axis.logarithmic) {
- this.chart[this.coll].forEach(function (axis) {
- var otherOptions = axis.options, horiz = axis.horiz, key = [
- horiz ? otherOptions.left : otherOptions.top,
- otherOptions.width,
- otherOptions.height,
- otherOptions.pane
- ].join(',');
- if (axis.series.length) { // #4442
- if (others[key]) {
- hasOther = true; // #4201
- }
- else {
- others[key] = 1;
- }
- }
- });
- }
- return hasOther;
- };
- /**
- * Find the max ticks of either the x and y axis collection, and record it
- * in `this.tickAmount`.
- *
- * @private
- * @function Highcharts.Axis#getTickAmount
- */
- Axis.prototype.getTickAmount = function () {
- var axis = this,
- options = this.options,
- tickAmount = options.tickAmount,
- tickPixelInterval = options.tickPixelInterval;
- if (!defined(options.tickInterval) &&
- !tickAmount && this.len < tickPixelInterval &&
- !this.isRadial &&
- !axis.logarithmic &&
- options.startOnTick &&
- options.endOnTick) {
- tickAmount = 2;
- }
- if (!tickAmount && this.alignToOthers()) {
- // Add 1 because 4 tick intervals require 5 ticks (including first
- // and last)
- tickAmount = Math.ceil(this.len / tickPixelInterval) + 1;
- }
- // For tick amounts of 2 and 3, compute five ticks and remove the
- // intermediate ones. This prevents the axis from adding ticks that are
- // too far away from the data extremes.
- if (tickAmount < 4) {
- this.finalTickAmt = tickAmount;
- tickAmount = 5;
- }
- this.tickAmount = tickAmount;
- };
- /**
- * When using multiple axes, adjust the number of ticks to match the highest
- * number of ticks in that group.
- *
- * @private
- * @function Highcharts.Axis#adjustTickAmount
- */
- Axis.prototype.adjustTickAmount = function () {
- var axis = this,
- axisOptions = axis.options,
- tickInterval = axis.tickInterval,
- tickPositions = axis.tickPositions,
- tickAmount = axis.tickAmount,
- finalTickAmt = axis.finalTickAmt,
- currentTickAmount = tickPositions && tickPositions.length,
- threshold = pick(axis.threshold,
- axis.softThreshold ? 0 : null),
- min,
- len,
- i;
- if (axis.hasData()) {
- if (currentTickAmount < tickAmount) {
- min = axis.min;
- while (tickPositions.length < tickAmount) {
- // Extend evenly for both sides unless we're on the
- // threshold (#3965)
- if (tickPositions.length % 2 ||
- min === threshold) {
- // to the end
- tickPositions.push(correctFloat(tickPositions[tickPositions.length - 1] +
- tickInterval));
- }
- else {
- // to the start
- tickPositions.unshift(correctFloat(tickPositions[0] - tickInterval));
- }
- }
- axis.transA *= (currentTickAmount - 1) / (tickAmount - 1);
- // Do not crop when ticks are not extremes (#9841)
- axis.min = axisOptions.startOnTick ?
- tickPositions[0] :
- Math.min(axis.min, tickPositions[0]);
- axis.max = axisOptions.endOnTick ?
- tickPositions[tickPositions.length - 1] :
- Math.max(axis.max, tickPositions[tickPositions.length - 1]);
- // We have too many ticks, run second pass to try to reduce ticks
- }
- else if (currentTickAmount > tickAmount) {
- axis.tickInterval *= 2;
- axis.setTickPositions();
- }
- // The finalTickAmt property is set in getTickAmount
- if (defined(finalTickAmt)) {
- i = len = tickPositions.length;
- while (i--) {
- if (
- // Remove every other tick
- (finalTickAmt === 3 && i % 2 === 1) ||
- // Remove all but first and last
- (finalTickAmt <= 2 && i > 0 && i < len - 1)) {
- tickPositions.splice(i, 1);
- }
- }
- axis.finalTickAmt = void 0;
- }
- }
- };
- /**
- * Set the scale based on data min and max, user set min and max or options.
- *
- * @private
- * @function Highcharts.Axis#setScale
- *
- * @fires Highcharts.Axis#event:afterSetScale
- */
- Axis.prototype.setScale = function () {
- var axis = this,
- isDirtyAxisLength,
- isDirtyData = false,
- isXAxisDirty = false;
- axis.series.forEach(function (series) {
- var _a;
- isDirtyData = isDirtyData || series.isDirtyData || series.isDirty;
- // When x axis is dirty, we need new data extremes for y as
- // well:
- isXAxisDirty = isXAxisDirty || ((_a = series.xAxis) === null || _a === void 0 ? void 0 : _a.isDirty) || false;
- });
- axis.oldMin = axis.min;
- axis.oldMax = axis.max;
- axis.oldAxisLength = axis.len;
- // set the new axisLength
- axis.setAxisSize();
- isDirtyAxisLength = axis.len !== axis.oldAxisLength;
- // do we really need to go through all this?
- if (isDirtyAxisLength ||
- isDirtyData ||
- isXAxisDirty ||
- axis.isLinked ||
- axis.forceRedraw ||
- axis.userMin !== axis.oldUserMin ||
- axis.userMax !== axis.oldUserMax ||
- axis.alignToOthers()) {
- if (axis.stacking) {
- axis.stacking.resetStacks();
- }
- axis.forceRedraw = false;
- // get data extremes if needed
- axis.getSeriesExtremes();
- // get fixed positions based on tickInterval
- axis.setTickInterval();
- // record old values to decide whether a rescale is necessary later
- // on (#540)
- axis.oldUserMin = axis.userMin;
- axis.oldUserMax = axis.userMax;
- // Mark as dirty if it is not already set to dirty and extremes have
- // changed. #595.
- if (!axis.isDirty) {
- axis.isDirty =
- isDirtyAxisLength ||
- axis.min !== axis.oldMin ||
- axis.max !== axis.oldMax;
- }
- }
- else if (axis.stacking) {
- axis.stacking.cleanStacks();
- }
- // Recalculate panning state object, when the data
- // has changed. It is required when vertical panning is enabled.
- if (isDirtyData && axis.panningState) {
- axis.panningState.isDirty = true;
- }
- fireEvent(this, 'afterSetScale');
- };
- /**
- * Set the minimum and maximum of the axes after render time. If the
- * `startOnTick` and `endOnTick` options are true, the minimum and maximum
- * values are rounded off to the nearest tick. To prevent this, these
- * options can be set to false before calling setExtremes. Also, setExtremes
- * will not allow a range lower than the `minRange` option, which by default
- * is the range of five points.
- *
- * @sample highcharts/members/axis-setextremes/
- * Set extremes from a button
- * @sample highcharts/members/axis-setextremes-datetime/
- * Set extremes on a datetime axis
- * @sample highcharts/members/axis-setextremes-off-ticks/
- * Set extremes off ticks
- * @sample stock/members/axis-setextremes/
- * Set extremes in Highstock
- * @sample maps/members/axis-setextremes/
- * Set extremes in Highmaps
- *
- * @function Highcharts.Axis#setExtremes
- *
- * @param {number} [newMin]
- * The new minimum value.
- *
- * @param {number} [newMax]
- * The new maximum value.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call to
- * {@link Highcharts.Chart#redraw}
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Enable or modify animations.
- *
- * @param {*} [eventArguments]
- * Arguments to be accessed in event handler.
- *
- * @fires Highcharts.Axis#event:setExtremes
- */
- Axis.prototype.setExtremes = function (newMin, newMax, redraw, animation, eventArguments) {
- var axis = this,
- chart = axis.chart;
- redraw = pick(redraw, true); // defaults to true
- axis.series.forEach(function (serie) {
- delete serie.kdTree;
- });
- // Extend the arguments with min and max
- eventArguments = extend(eventArguments, {
- min: newMin,
- max: newMax
- });
- // Fire the event
- fireEvent(axis, 'setExtremes', eventArguments, function () {
- axis.userMin = newMin;
- axis.userMax = newMax;
- axis.eventArgs = eventArguments;
- if (redraw) {
- chart.redraw(animation);
- }
- });
- };
- /**
- * Overridable method for zooming chart. Pulled out in a separate method to
- * allow overriding in stock charts.
- * @private
- * @function Highcharts.Axis#zoom
- *
- * @param {number} newMin
- * TO-DO: parameter description
- *
- * @param {number} newMax
- * TO-DO: parameter description
- *
- * @return {boolean}
- */
- Axis.prototype.zoom = function (newMin, newMax) {
- var axis = this,
- dataMin = this.dataMin,
- dataMax = this.dataMax,
- options = this.options,
- min = Math.min(dataMin,
- pick(options.min,
- dataMin)),
- max = Math.max(dataMax,
- pick(options.max,
- dataMax)),
- evt = {
- newMin: newMin,
- newMax: newMax
- };
- fireEvent(this, 'zoom', evt, function (e) {
- // Use e.newMin and e.newMax - event handlers may have altered them
- var newMin = e.newMin,
- newMax = e.newMax;
- if (newMin !== axis.min || newMax !== axis.max) { // #5790
- // Prevent pinch zooming out of range. Check for defined is for
- // #1946. #1734.
- if (!axis.allowZoomOutside) {
- // #6014, sometimes newMax will be smaller than min (or
- // newMin will be larger than max).
- if (defined(dataMin)) {
- if (newMin < min) {
- newMin = min;
- }
- if (newMin > max) {
- newMin = max;
- }
- }
- if (defined(dataMax)) {
- if (newMax < min) {
- newMax = min;
- }
- if (newMax > max) {
- newMax = max;
- }
- }
- }
- // In full view, displaying the reset zoom button is not
- // required
- axis.displayBtn = (typeof newMin !== 'undefined' ||
- typeof newMax !== 'undefined');
- // Do it
- axis.setExtremes(newMin, newMax, false, void 0, { trigger: 'zoom' });
- }
- e.zoomed = true;
- });
- return evt.zoomed;
- };
- /**
- * Update the axis metrics.
- *
- * @private
- * @function Highcharts.Axis#setAxisSize
- */
- Axis.prototype.setAxisSize = function () {
- var chart = this.chart,
- options = this.options,
- // [top, right, bottom, left]
- offsets = options.offsets || [0, 0, 0, 0],
- horiz = this.horiz,
- // Check for percentage based input values. Rounding fixes problems
- // with column overflow and plot line filtering (#4898, #4899)
- width = this.width = Math.round(relativeLength(pick(options.width,
- chart.plotWidth - offsets[3] + offsets[1]),
- chart.plotWidth)),
- height = this.height = Math.round(relativeLength(pick(options.height,
- chart.plotHeight - offsets[0] + offsets[2]),
- chart.plotHeight)),
- top = this.top = Math.round(relativeLength(pick(options.top,
- chart.plotTop + offsets[0]),
- chart.plotHeight,
- chart.plotTop)),
- left = this.left = Math.round(relativeLength(pick(options.left,
- chart.plotLeft + offsets[3]),
- chart.plotWidth,
- chart.plotLeft));
- // Expose basic values to use in Series object and navigator
- this.bottom = chart.chartHeight - height - top;
- this.right = chart.chartWidth - width - left;
- // Direction agnostic properties
- this.len = Math.max(horiz ? width : height, 0); // Math.max fixes #905
- this.pos = horiz ? left : top; // distance from SVG origin
- };
- /**
- * Get the current extremes for the axis.
- *
- * @sample highcharts/members/axis-getextremes/
- * Report extremes by click on a button
- * @sample maps/members/axis-getextremes/
- * Get extremes in Highmaps
- *
- * @function Highcharts.Axis#getExtremes
- *
- * @return {Highcharts.ExtremesObject}
- * An object containing extremes information.
- */
- Axis.prototype.getExtremes = function () {
- var axis = this;
- var log = axis.logarithmic;
- return {
- min: log ?
- correctFloat(log.lin2log(axis.min)) :
- axis.min,
- max: log ?
- correctFloat(log.lin2log(axis.max)) :
- axis.max,
- dataMin: axis.dataMin,
- dataMax: axis.dataMax,
- userMin: axis.userMin,
- userMax: axis.userMax
- };
- };
- /**
- * Get the zero plane either based on zero or on the min or max value.
- * Used in bar and area plots.
- *
- * @function Highcharts.Axis#getThreshold
- *
- * @param {number} threshold
- * The threshold in axis values.
- *
- * @return {number|undefined}
- * The translated threshold position in terms of pixels, and corrected to
- * stay within the axis bounds.
- */
- Axis.prototype.getThreshold = function (threshold) {
- var axis = this,
- log = axis.logarithmic,
- realMin = log ? log.lin2log(axis.min) : axis.min,
- realMax = log ? log.lin2log(axis.max) : axis.max;
- if (threshold === null || threshold === -Infinity) {
- threshold = realMin;
- }
- else if (threshold === Infinity) {
- threshold = realMax;
- }
- else if (realMin > threshold) {
- threshold = realMin;
- }
- else if (realMax < threshold) {
- threshold = realMax;
- }
- return axis.translate(threshold, 0, 1, 0, 1);
- };
- /**
- * Compute auto alignment for the axis label based on which side the axis is
- * on and the given rotation for the label.
- *
- * @private
- * @function Highcharts.Axis#autoLabelAlign
- *
- * @param {number} rotation
- * The rotation in degrees as set by either the `rotation` or `autoRotation`
- * options.
- *
- * @return {Highcharts.AlignValue}
- * Can be `"center"`, `"left"` or `"right"`.
- */
- Axis.prototype.autoLabelAlign = function (rotation) {
- var angle = (pick(rotation, 0) - (this.side * 90) + 720) % 360,
- evt = { align: 'center' };
- fireEvent(this, 'autoLabelAlign', evt, function (e) {
- if (angle > 15 && angle < 165) {
- e.align = 'right';
- }
- else if (angle > 195 && angle < 345) {
- e.align = 'left';
- }
- });
- return evt.align;
- };
- /**
- * Get the tick length and width for the axis based on axis options.
- * @private
- * @function Highcharts.Axis#tickSize
- *
- * @param {string} [prefix]
- * 'tick' or 'minorTick'
- *
- * @return {Array<number,number>|undefined}
- * An array of tickLength and tickWidth
- */
- Axis.prototype.tickSize = function (prefix) {
- var options = this.options, tickLength = options[prefix === 'tick' ? 'tickLength' : 'minorTickLength'], tickWidth = pick(options[prefix === 'tick' ? 'tickWidth' : 'minorTickWidth'],
- // Default to 1 on linear and datetime X axes
- prefix === 'tick' && this.isXAxis && !this.categories ? 1 : 0), e, tickSize;
- if (tickWidth && tickLength) {
- // Negate the length
- if (options[prefix + 'Position'] === 'inside') {
- tickLength = -tickLength;
- }
- tickSize = [tickLength, tickWidth];
- }
- e = { tickSize: tickSize };
- fireEvent(this, 'afterTickSize', e);
- return e.tickSize;
- };
- /**
- * Return the size of the labels.
- *
- * @private
- * @function Highcharts.Axis#labelMetrics
- *
- * @return {Highcharts.FontMetricsObject}
- */
- Axis.prototype.labelMetrics = function () {
- var index = this.tickPositions && this.tickPositions[0] || 0;
- return this.chart.renderer.fontMetrics(this.options.labels.style &&
- this.options.labels.style.fontSize, this.ticks[index] && this.ticks[index].label);
- };
- /**
- * Prevent the ticks from getting so close we can't draw the labels. On a
- * horizontal axis, this is handled by rotating the labels, removing ticks
- * and adding ellipsis. On a vertical axis remove ticks and add ellipsis.
- *
- * @private
- * @function Highcharts.Axis#unsquish
- *
- * @return {number}
- */
- Axis.prototype.unsquish = function () {
- var labelOptions = this.options.labels,
- horiz = this.horiz,
- tickInterval = this.tickInterval,
- newTickInterval = tickInterval,
- slotSize = this.len / (((this.categories ? 1 : 0) +
- this.max -
- this.min) /
- tickInterval),
- rotation,
- rotationOption = labelOptions.rotation,
- labelMetrics = this.labelMetrics(),
- step,
- bestScore = Number.MAX_VALUE,
- autoRotation,
- range = this.max - this.min,
- // Return the multiple of tickInterval that is needed to avoid
- // collision
- getStep = function (spaceNeeded) {
- var step = spaceNeeded / (slotSize || 1);
- step = step > 1 ? Math.ceil(step) : 1;
- // Guard for very small or negative angles (#9835)
- if (step * tickInterval > range &&
- spaceNeeded !== Infinity &&
- slotSize !== Infinity &&
- range) {
- step = Math.ceil(range / tickInterval);
- }
- return correctFloat(step * tickInterval);
- };
- if (horiz) {
- autoRotation = !labelOptions.staggerLines &&
- !labelOptions.step &&
- ( // #3971
- defined(rotationOption) ?
- [rotationOption] :
- slotSize < pick(labelOptions.autoRotationLimit, 80) && labelOptions.autoRotation);
- if (autoRotation) {
- // Loop over the given autoRotation options, and determine
- // which gives the best score. The best score is that with
- // the lowest number of steps and a rotation closest
- // to horizontal.
- autoRotation.forEach(function (rot) {
- var score;
- if (rot === rotationOption ||
- (rot && rot >= -90 && rot <= 90)) { // #3891
- step = getStep(Math.abs(labelMetrics.h / Math.sin(deg2rad * rot)));
- score = step + Math.abs(rot / 360);
- if (score < bestScore) {
- bestScore = score;
- rotation = rot;
- newTickInterval = step;
- }
- }
- });
- }
- }
- else if (!labelOptions.step) { // #4411
- newTickInterval = getStep(labelMetrics.h);
- }
- this.autoRotation = autoRotation;
- this.labelRotation = pick(rotation, rotationOption);
- return newTickInterval;
- };
- /**
- * Get the general slot width for labels/categories on this axis. This may
- * change between the pre-render (from Axis.getOffset) and the final tick
- * rendering and placement.
- *
- * @private
- * @function Highcharts.Axis#getSlotWidth
- *
- * @param {Highcharts.Tick} [tick] Optionally, calculate the slot width
- * basing on tick label. It is used in highcharts-3d module, where the slots
- * has different widths depending on perspective angles.
- *
- * @return {number}
- * The pixel width allocated to each axis label.
- */
- Axis.prototype.getSlotWidth = function (tick) {
- var _a;
- // #5086, #1580, #1931
- var chart = this.chart,
- horiz = this.horiz,
- labelOptions = this.options.labels,
- slotCount = Math.max(this.tickPositions.length - (this.categories ? 0 : 1), 1),
- marginLeft = chart.margin[3];
- // Used by grid axis
- if (tick && isNumber(tick.slotWidth)) { // #13221, can be 0
- return tick.slotWidth;
- }
- if (horiz &&
- labelOptions &&
- (labelOptions.step || 0) < 2) {
- if (labelOptions.rotation) { // #4415
- return 0;
- }
- return ((this.staggerLines || 1) * this.len) / slotCount;
- }
- if (!horiz) {
- // #7028
- var cssWidth = (_a = labelOptions === null || labelOptions === void 0 ? void 0 : labelOptions.style) === null || _a === void 0 ? void 0 : _a.width;
- if (cssWidth !== void 0) {
- return parseInt(cssWidth, 10);
- }
- if (marginLeft) {
- return marginLeft - chart.spacing[3];
- }
- }
- // Last resort, a fraction of the available size
- return chart.chartWidth * 0.33;
- };
- /**
- * Render the axis labels and determine whether ellipsis or rotation need to
- * be applied.
- *
- * @private
- * @function Highcharts.Axis#renderUnsquish
- */
- Axis.prototype.renderUnsquish = function () {
- var chart = this.chart,
- renderer = chart.renderer,
- tickPositions = this.tickPositions,
- ticks = this.ticks,
- labelOptions = this.options.labels,
- labelStyleOptions = (labelOptions && labelOptions.style || {}),
- horiz = this.horiz,
- slotWidth = this.getSlotWidth(),
- innerWidth = Math.max(1,
- Math.round(slotWidth - 2 * (labelOptions.padding || 5))),
- attr = {},
- labelMetrics = this.labelMetrics(),
- textOverflowOption = (labelOptions.style &&
- labelOptions.style.textOverflow),
- commonWidth,
- commonTextOverflow,
- maxLabelLength = 0,
- label,
- i,
- pos;
- // Set rotation option unless it is "auto", like in gauges
- if (!isString(labelOptions.rotation)) {
- // #4443:
- attr.rotation = labelOptions.rotation || 0;
- }
- // Get the longest label length
- tickPositions.forEach(function (tick) {
- tick = ticks[tick];
- // Replace label - sorting animation
- if (tick.movedLabel) {
- tick.replaceMovedLabel();
- }
- if (tick &&
- tick.label &&
- tick.label.textPxLength > maxLabelLength) {
- maxLabelLength = tick.label.textPxLength;
- }
- });
- this.maxLabelLength = maxLabelLength;
- // Handle auto rotation on horizontal axis
- if (this.autoRotation) {
- // Apply rotation only if the label is too wide for the slot, and
- // the label is wider than its height.
- if (maxLabelLength > innerWidth &&
- maxLabelLength > labelMetrics.h) {
- attr.rotation = this.labelRotation;
- }
- else {
- this.labelRotation = 0;
- }
- // Handle word-wrap or ellipsis on vertical axis
- }
- else if (slotWidth) {
- // For word-wrap or ellipsis
- commonWidth = innerWidth;
- if (!textOverflowOption) {
- commonTextOverflow = 'clip';
- // On vertical axis, only allow word wrap if there is room
- // for more lines.
- i = tickPositions.length;
- while (!horiz && i--) {
- pos = tickPositions[i];
- label = ticks[pos].label;
- if (label) {
- // Reset ellipsis in order to get the correct
- // bounding box (#4070)
- if (label.styles &&
- label.styles.textOverflow === 'ellipsis') {
- label.css({ textOverflow: 'clip' });
- // Set the correct width in order to read
- // the bounding box height (#4678, #5034)
- }
- else if (label.textPxLength > slotWidth) {
- label.css({ width: slotWidth + 'px' });
- }
- if (label.getBBox().height > (this.len / tickPositions.length -
- (labelMetrics.h - labelMetrics.f))) {
- label.specificTextOverflow = 'ellipsis';
- }
- }
- }
- }
- }
- // Add ellipsis if the label length is significantly longer than ideal
- if (attr.rotation) {
- commonWidth = (maxLabelLength > chart.chartHeight * 0.5 ?
- chart.chartHeight * 0.33 :
- maxLabelLength);
- if (!textOverflowOption) {
- commonTextOverflow = 'ellipsis';
- }
- }
- // Set the explicit or automatic label alignment
- this.labelAlign = labelOptions.align ||
- this.autoLabelAlign(this.labelRotation);
- if (this.labelAlign) {
- attr.align = this.labelAlign;
- }
- // Apply general and specific CSS
- tickPositions.forEach(function (pos) {
- var tick = ticks[pos],
- label = tick && tick.label,
- widthOption = labelStyleOptions.width,
- css = {};
- if (label) {
- // This needs to go before the CSS in old IE (#4502)
- label.attr(attr);
- if (tick.shortenLabel) {
- tick.shortenLabel();
- }
- else if (commonWidth &&
- !widthOption &&
- // Setting width in this case messes with the bounding box
- // (#7975)
- labelStyleOptions.whiteSpace !== 'nowrap' &&
- (
- // Speed optimizing, #7656
- commonWidth < label.textPxLength ||
- // Resetting CSS, #4928
- label.element.tagName === 'SPAN')) {
- css.width = commonWidth + 'px';
- if (!textOverflowOption) {
- css.textOverflow = (label.specificTextOverflow ||
- commonTextOverflow);
- }
- label.css(css);
- // Reset previously shortened label (#8210)
- }
- else if (label.styles &&
- label.styles.width &&
- !css.width &&
- !widthOption) {
- label.css({ width: null });
- }
- delete label.specificTextOverflow;
- tick.rotation = attr.rotation;
- }
- }, this);
- // Note: Why is this not part of getLabelPosition?
- this.tickRotCorr = renderer.rotCorr(labelMetrics.b, this.labelRotation || 0, this.side !== 0);
- };
- /**
- * Return true if the axis has associated data.
- *
- * @function Highcharts.Axis#hasData
- *
- * @return {boolean}
- * True if the axis has associated visible series and those series have
- * either valid data points or explicit `min` and `max` settings.
- */
- Axis.prototype.hasData = function () {
- return this.series.some(function (s) {
- return s.hasData();
- }) ||
- (this.options.showEmpty &&
- defined(this.min) &&
- defined(this.max));
- };
- /**
- * Adds the title defined in axis.options.title.
- *
- * @function Highcharts.Axis#addTitle
- *
- * @param {boolean} [display]
- * Whether or not to display the title.
- */
- Axis.prototype.addTitle = function (display) {
- var axis = this,
- renderer = axis.chart.renderer,
- horiz = axis.horiz,
- opposite = axis.opposite,
- options = axis.options,
- axisTitleOptions = options.title,
- textAlign,
- styledMode = axis.chart.styledMode;
- if (!axis.axisTitle) {
- textAlign = axisTitleOptions.textAlign;
- if (!textAlign) {
- textAlign = (horiz ? {
- low: 'left',
- middle: 'center',
- high: 'right'
- } : {
- low: opposite ? 'right' : 'left',
- middle: 'center',
- high: opposite ? 'left' : 'right'
- })[axisTitleOptions.align];
- }
- axis.axisTitle = renderer
- .text(axisTitleOptions.text, 0, 0, axisTitleOptions.useHTML)
- .attr({
- zIndex: 7,
- rotation: axisTitleOptions.rotation || 0,
- align: textAlign
- })
- .addClass('highcharts-axis-title');
- // #7814, don't mutate style option
- if (!styledMode) {
- axis.axisTitle.css(merge(axisTitleOptions.style));
- }
- axis.axisTitle.add(axis.axisGroup);
- axis.axisTitle.isNew = true;
- }
- // Max width defaults to the length of the axis
- if (!styledMode &&
- !axisTitleOptions.style.width &&
- !axis.isRadial) {
- axis.axisTitle.css({
- width: axis.len + 'px'
- });
- }
- // hide or show the title depending on whether showEmpty is set
- axis.axisTitle[display ? 'show' : 'hide'](display);
- };
- /**
- * Generates a tick for initial positioning.
- *
- * @private
- * @function Highcharts.Axis#generateTick
- *
- * @param {number} pos
- * The tick position in axis values.
- *
- * @param {number} [i]
- * The index of the tick in {@link Axis.tickPositions}.
- */
- Axis.prototype.generateTick = function (pos) {
- var axis = this;
- var ticks = axis.ticks;
- if (!ticks[pos]) {
- ticks[pos] = new Tick(axis, pos);
- }
- else {
- ticks[pos].addLabel(); // update labels depending on tick interval
- }
- };
- /**
- * Render the tick labels to a preliminary position to get their sizes
- *
- * @private
- * @function Highcharts.Axis#getOffset
- *
- * @fires Highcharts.Axis#event:afterGetOffset
- */
- Axis.prototype.getOffset = function () {
- var axis = this,
- chart = axis.chart,
- renderer = chart.renderer,
- options = axis.options,
- tickPositions = axis.tickPositions,
- ticks = axis.ticks,
- horiz = axis.horiz,
- side = axis.side,
- invertedSide = chart.inverted &&
- !axis.isZAxis ? [1, 0, 3, 2][side] : side,
- hasData,
- showAxis,
- titleOffset = 0,
- titleOffsetOption,
- titleMargin = 0,
- axisTitleOptions = options.title,
- labelOptions = options.labels,
- labelOffset = 0, // reset
- labelOffsetPadded,
- axisOffset = chart.axisOffset,
- clipOffset = chart.clipOffset,
- clip,
- directionFactor = [-1, 1, 1, -1][side],
- className = options.className,
- axisParent = axis.axisParent, // Used in color axis
- lineHeightCorrection;
- // For reuse in Axis.render
- hasData = axis.hasData();
- axis.showAxis = showAxis = hasData || pick(options.showEmpty, true);
- // Set/reset staggerLines
- axis.staggerLines = axis.horiz && labelOptions.staggerLines;
- // Create the axisGroup and gridGroup elements on first iteration
- if (!axis.axisGroup) {
- axis.gridGroup = renderer.g('grid')
- .attr({ zIndex: options.gridZIndex || 1 })
- .addClass('highcharts-' + this.coll.toLowerCase() + '-grid ' +
- (className || ''))
- .add(axisParent);
- axis.axisGroup = renderer.g('axis')
- .attr({ zIndex: options.zIndex || 2 })
- .addClass('highcharts-' + this.coll.toLowerCase() + ' ' +
- (className || ''))
- .add(axisParent);
- axis.labelGroup = renderer.g('axis-labels')
- .attr({ zIndex: labelOptions.zIndex || 7 })
- .addClass('highcharts-' + axis.coll.toLowerCase() + '-labels ' +
- (className || ''))
- .add(axisParent);
- }
- if (hasData || axis.isLinked) {
- // Generate ticks
- tickPositions.forEach(function (pos, i) {
- // i is not used here, but may be used in overrides
- axis.generateTick(pos, i);
- });
- axis.renderUnsquish();
- // Left side must be align: right and right side must
- // have align: left for labels
- axis.reserveSpaceDefault = (side === 0 ||
- side === 2 ||
- { 1: 'left', 3: 'right' }[side] === axis.labelAlign);
- if (pick(labelOptions.reserveSpace, axis.labelAlign === 'center' ? true : null, axis.reserveSpaceDefault)) {
- tickPositions.forEach(function (pos) {
- // get the highest offset
- labelOffset = Math.max(ticks[pos].getLabelSize(), labelOffset);
- });
- }
- if (axis.staggerLines) {
- labelOffset *= axis.staggerLines;
- }
- axis.labelOffset = labelOffset * (axis.opposite ? -1 : 1);
- }
- else { // doesn't have data
- objectEach(ticks, function (tick, n) {
- tick.destroy();
- delete ticks[n];
- });
- }
- if (axisTitleOptions &&
- axisTitleOptions.text &&
- axisTitleOptions.enabled !== false) {
- axis.addTitle(showAxis);
- if (showAxis && axisTitleOptions.reserveSpace !== false) {
- axis.titleOffset = titleOffset =
- axis.axisTitle.getBBox()[horiz ? 'height' : 'width'];
- titleOffsetOption = axisTitleOptions.offset;
- titleMargin = defined(titleOffsetOption) ?
- 0 :
- pick(axisTitleOptions.margin, horiz ? 5 : 10);
- }
- }
- // Render the axis line
- axis.renderLine();
- // handle automatic or user set offset
- axis.offset = directionFactor * pick(options.offset, axisOffset[side] ? axisOffset[side] + (options.margin || 0) : 0);
- axis.tickRotCorr = axis.tickRotCorr || { x: 0, y: 0 }; // polar
- if (side === 0) {
- lineHeightCorrection = -axis.labelMetrics().h;
- }
- else if (side === 2) {
- lineHeightCorrection = axis.tickRotCorr.y;
- }
- else {
- lineHeightCorrection = 0;
- }
- // Find the padded label offset
- labelOffsetPadded = Math.abs(labelOffset) + titleMargin;
- if (labelOffset) {
- labelOffsetPadded -= lineHeightCorrection;
- labelOffsetPadded += directionFactor * (horiz ?
- pick(labelOptions.y, axis.tickRotCorr.y + directionFactor * 8) :
- labelOptions.x);
- }
- axis.axisTitleMargin = pick(titleOffsetOption, labelOffsetPadded);
- if (axis.getMaxLabelDimensions) {
- axis.maxLabelDimensions = axis.getMaxLabelDimensions(ticks, tickPositions);
- }
- // Due to GridAxis.tickSize, tickSize should be calculated after ticks
- // has rendered.
- var tickSize = this.tickSize('tick');
- axisOffset[side] = Math.max(axisOffset[side], axis.axisTitleMargin + titleOffset +
- directionFactor * axis.offset, labelOffsetPadded, // #3027
- tickPositions && tickPositions.length && tickSize ?
- tickSize[0] + directionFactor * axis.offset :
- 0 // #4866
- );
- // Decide the clipping needed to keep the graph inside
- // the plot area and axis lines
- clip = options.offset ?
- 0 :
- // #4308, #4371:
- Math.floor(axis.axisLine.strokeWidth() / 2) * 2;
- clipOffset[invertedSide] =
- Math.max(clipOffset[invertedSide], clip);
- fireEvent(this, 'afterGetOffset');
- };
- /**
- * Internal function to get the path for the axis line. Extended for polar
- * charts.
- *
- * @function Highcharts.Axis#getLinePath
- *
- * @param {number} lineWidth
- * The line width in pixels.
- *
- * @return {Highcharts.SVGPathArray}
- * The SVG path definition in array form.
- */
- Axis.prototype.getLinePath = function (lineWidth) {
- var chart = this.chart,
- opposite = this.opposite,
- offset = this.offset,
- horiz = this.horiz,
- lineLeft = this.left + (opposite ? this.width : 0) + offset,
- lineTop = chart.chartHeight - this.bottom -
- (opposite ? this.height : 0) + offset;
- if (opposite) {
- lineWidth *= -1; // crispify the other way - #1480, #1687
- }
- return chart.renderer
- .crispLine([
- [
- 'M',
- horiz ?
- this.left :
- lineLeft,
- horiz ?
- lineTop :
- this.top
- ],
- [
- 'L',
- horiz ?
- chart.chartWidth - this.right :
- lineLeft,
- horiz ?
- lineTop :
- chart.chartHeight - this.bottom
- ]
- ], lineWidth);
- };
- /**
- * Render the axis line. Called internally when rendering and redrawing the
- * axis.
- *
- * @function Highcharts.Axis#renderLine
- */
- Axis.prototype.renderLine = function () {
- if (!this.axisLine) {
- this.axisLine = this.chart.renderer.path()
- .addClass('highcharts-axis-line')
- .add(this.axisGroup);
- if (!this.chart.styledMode) {
- this.axisLine.attr({
- stroke: this.options.lineColor,
- 'stroke-width': this.options.lineWidth,
- zIndex: 7
- });
- }
- }
- };
- /**
- * Position the axis title.
- *
- * @private
- * @function Highcharts.Axis#getTitlePosition
- *
- * @return {Highcharts.PositionObject}
- * X and Y positions for the title.
- */
- Axis.prototype.getTitlePosition = function () {
- // compute anchor points for each of the title align options
- var horiz = this.horiz,
- axisLeft = this.left,
- axisTop = this.top,
- axisLength = this.len,
- axisTitleOptions = this.options.title,
- margin = horiz ? axisLeft : axisTop,
- opposite = this.opposite,
- offset = this.offset,
- xOption = axisTitleOptions.x || 0,
- yOption = axisTitleOptions.y || 0,
- axisTitle = this.axisTitle,
- fontMetrics = this.chart.renderer.fontMetrics(axisTitleOptions.style &&
- axisTitleOptions.style.fontSize,
- axisTitle),
- // The part of a multiline text that is below the baseline of the
- // first line. Subtract 1 to preserve pixel-perfectness from the
- // old behaviour (v5.0.12), where only one line was allowed.
- textHeightOvershoot = Math.max(axisTitle.getBBox(null, 0).height - fontMetrics.h - 1, 0),
- // the position in the length direction of the axis
- alongAxis = {
- low: margin + (horiz ? 0 : axisLength),
- middle: margin + axisLength / 2,
- high: margin + (horiz ? axisLength : 0)
- }[axisTitleOptions.align],
- // the position in the perpendicular direction of the axis
- offAxis = (horiz ? axisTop + this.height : axisLeft) +
- (horiz ? 1 : -1) * // horizontal axis reverses the margin
- (opposite ? -1 : 1) * // so does opposite axes
- this.axisTitleMargin +
- [
- -textHeightOvershoot,
- textHeightOvershoot,
- fontMetrics.f,
- -textHeightOvershoot // left
- ][this.side],
- titlePosition = {
- x: horiz ?
- alongAxis + xOption :
- offAxis + (opposite ? this.width : 0) + offset + xOption,
- y: horiz ?
- offAxis + yOption - (opposite ? this.height : 0) + offset :
- alongAxis + yOption
- };
- fireEvent(this, 'afterGetTitlePosition', { titlePosition: titlePosition });
- return titlePosition;
- };
- /**
- * Render a minor tick into the given position. If a minor tick already
- * exists in this position, move it.
- *
- * @function Highcharts.Axis#renderMinorTick
- *
- * @param {number} pos
- * The position in axis values.
- */
- Axis.prototype.renderMinorTick = function (pos) {
- var axis = this;
- var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin);
- var minorTicks = axis.minorTicks;
- if (!minorTicks[pos]) {
- minorTicks[pos] = new Tick(axis, pos, 'minor');
- }
- // Render new ticks in old position
- if (slideInTicks && minorTicks[pos].isNew) {
- minorTicks[pos].render(null, true);
- }
- minorTicks[pos].render(null, false, 1);
- };
- /**
- * Render a major tick into the given position. If a tick already exists
- * in this position, move it.
- *
- * @function Highcharts.Axis#renderTick
- *
- * @param {number} pos
- * The position in axis values.
- *
- * @param {number} i
- * The tick index.
- */
- Axis.prototype.renderTick = function (pos, i) {
- var axis = this;
- var isLinked = axis.isLinked;
- var ticks = axis.ticks;
- var slideInTicks = axis.chart.hasRendered && isNumber(axis.oldMin);
- // Linked axes need an extra check to find out if
- if (!isLinked ||
- (pos >= axis.min && pos <= axis.max)) {
- if (!ticks[pos]) {
- ticks[pos] = new Tick(axis, pos);
- }
- // NOTE this seems like overkill. Could be handled in tick.render by
- // setting old position in attr, then set new position in animate.
- // render new ticks in old position
- if (slideInTicks && ticks[pos].isNew) {
- // Start with negative opacity so that it is visible from
- // halfway into the animation
- ticks[pos].render(i, true, -1);
- }
- ticks[pos].render(i);
- }
- };
- /**
- * Render the axis.
- *
- * @private
- * @function Highcharts.Axis#render
- *
- * @fires Highcharts.Axis#event:afterRender
- */
- Axis.prototype.render = function () {
- var axis = this,
- chart = axis.chart,
- log = axis.logarithmic,
- renderer = chart.renderer,
- options = axis.options,
- isLinked = axis.isLinked,
- tickPositions = axis.tickPositions,
- axisTitle = axis.axisTitle,
- ticks = axis.ticks,
- minorTicks = axis.minorTicks,
- alternateBands = axis.alternateBands,
- stackLabelOptions = options.stackLabels,
- alternateGridColor = options.alternateGridColor,
- tickmarkOffset = axis.tickmarkOffset,
- axisLine = axis.axisLine,
- showAxis = axis.showAxis,
- animation = animObject(renderer.globalAnimation),
- from,
- to;
- // Reset
- axis.labelEdge.length = 0;
- axis.overlap = false;
- // Mark all elements inActive before we go over and mark the active ones
- [ticks, minorTicks, alternateBands].forEach(function (coll) {
- objectEach(coll, function (tick) {
- tick.isActive = false;
- });
- });
- // If the series has data draw the ticks. Else only the line and title
- if (axis.hasData() || isLinked) {
- // minor ticks
- if (axis.minorTickInterval && !axis.categories) {
- axis.getMinorTickPositions().forEach(function (pos) {
- axis.renderMinorTick(pos);
- });
- }
- // Major ticks. Pull out the first item and render it last so that
- // we can get the position of the neighbour label. #808.
- if (tickPositions.length) { // #1300
- tickPositions.forEach(function (pos, i) {
- axis.renderTick(pos, i);
- });
- // In a categorized axis, the tick marks are displayed
- // between labels. So we need to add a tick mark and
- // grid line at the left edge of the X axis.
- if (tickmarkOffset && (axis.min === 0 || axis.single)) {
- if (!ticks[-1]) {
- ticks[-1] = new Tick(axis, -1, null, true);
- }
- ticks[-1].render(-1);
- }
- }
- // alternate grid color
- if (alternateGridColor) {
- tickPositions.forEach(function (pos, i) {
- to = typeof tickPositions[i + 1] !== 'undefined' ?
- tickPositions[i + 1] + tickmarkOffset :
- axis.max - tickmarkOffset;
- if (i % 2 === 0 &&
- pos < axis.max &&
- to <= axis.max + (chart.polar ?
- -tickmarkOffset :
- tickmarkOffset)) { // #2248, #4660
- if (!alternateBands[pos]) {
- // Should be imported from PlotLineOrBand.js, but
- // the dependency cycle with axis is a problem
- alternateBands[pos] = new H.PlotLineOrBand(axis);
- }
- from = pos + tickmarkOffset; // #949
- alternateBands[pos].options = {
- from: log ? log.lin2log(from) : from,
- to: log ? log.lin2log(to) : to,
- color: alternateGridColor,
- className: 'highcharts-alternate-grid'
- };
- alternateBands[pos].render();
- alternateBands[pos].isActive = true;
- }
- });
- }
- // custom plot lines and bands
- if (!axis._addedPlotLB) { // only first time
- (options.plotLines || [])
- .concat(options.plotBands || [])
- .forEach(function (plotLineOptions) {
- axis.addPlotBandOrLine(plotLineOptions);
- });
- axis._addedPlotLB = true;
- }
- } // end if hasData
- // Remove inactive ticks
- [ticks, minorTicks, alternateBands].forEach(function (coll) {
- var i,
- forDestruction = [],
- delay = animation.duration,
- destroyInactiveItems = function () {
- i = forDestruction.length;
- while (i--) {
- // When resizing rapidly, the same items
- // may be destroyed in different timeouts,
- // or the may be reactivated
- if (coll[forDestruction[i]] &&
- !coll[forDestruction[i]].isActive) {
- coll[forDestruction[i]].destroy();
- delete coll[forDestruction[i]];
- }
- }
- };
- objectEach(coll, function (tick, pos) {
- if (!tick.isActive) {
- // Render to zero opacity
- tick.render(pos, false, 0);
- tick.isActive = false;
- forDestruction.push(pos);
- }
- });
- // When the objects are finished fading out, destroy them
- syncTimeout(destroyInactiveItems, coll === alternateBands ||
- !chart.hasRendered ||
- !delay ?
- 0 :
- delay);
- });
- // Set the axis line path
- if (axisLine) {
- axisLine[axisLine.isPlaced ? 'animate' : 'attr']({
- d: this.getLinePath(axisLine.strokeWidth())
- });
- axisLine.isPlaced = true;
- // Show or hide the line depending on options.showEmpty
- axisLine[showAxis ? 'show' : 'hide'](showAxis);
- }
- if (axisTitle && showAxis) {
- var titleXy = axis.getTitlePosition();
- if (isNumber(titleXy.y)) {
- axisTitle[axisTitle.isNew ? 'attr' : 'animate'](titleXy);
- axisTitle.isNew = false;
- }
- else {
- axisTitle.attr('y', -9999);
- axisTitle.isNew = true;
- }
- }
- // Stacked totals:
- if (stackLabelOptions && stackLabelOptions.enabled && axis.stacking) {
- axis.stacking.renderStackTotals();
- }
- // End stacked totals
- axis.isDirty = false;
- fireEvent(this, 'afterRender');
- };
- /**
- * Redraw the axis to reflect changes in the data or axis extremes. Called
- * internally from Highcharts.Chart#redraw.
- *
- * @private
- * @function Highcharts.Axis#redraw
- */
- Axis.prototype.redraw = function () {
- if (this.visible) {
- // render the axis
- this.render();
- // move plot lines and bands
- this.plotLinesAndBands.forEach(function (plotLine) {
- plotLine.render();
- });
- }
- // mark associated series as dirty and ready for redraw
- this.series.forEach(function (series) {
- series.isDirty = true;
- });
- };
- /**
- * Returns an array of axis properties, that should be untouched during
- * reinitialization.
- *
- * @private
- * @function Highcharts.Axis#getKeepProps
- *
- * @return {Array<string>}
- */
- Axis.prototype.getKeepProps = function () {
- return (this.keepProps || Axis.keepProps);
- };
- /**
- * Destroys an Axis instance. See {@link Axis#remove} for the API endpoint
- * to fully remove the axis.
- *
- * @private
- * @function Highcharts.Axis#destroy
- *
- * @param {boolean} [keepEvents]
- * Whether to preserve events, used internally in Axis.update.
- */
- Axis.prototype.destroy = function (keepEvents) {
- var axis = this,
- plotLinesAndBands = axis.plotLinesAndBands,
- plotGroup,
- i;
- fireEvent(this, 'destroy', { keepEvents: keepEvents });
- // Remove the events
- if (!keepEvents) {
- removeEvent(axis);
- }
- // Destroy collections
- [axis.ticks, axis.minorTicks, axis.alternateBands].forEach(function (coll) {
- destroyObjectProperties(coll);
- });
- if (plotLinesAndBands) {
- i = plotLinesAndBands.length;
- while (i--) { // #1975
- plotLinesAndBands[i].destroy();
- }
- }
- // Destroy elements
- ['axisLine', 'axisTitle', 'axisGroup',
- 'gridGroup', 'labelGroup', 'cross', 'scrollbar'].forEach(function (prop) {
- if (axis[prop]) {
- axis[prop] = axis[prop].destroy();
- }
- });
- // Destroy each generated group for plotlines and plotbands
- for (plotGroup in axis.plotLinesAndBandsGroups) { // eslint-disable-line guard-for-in
- axis.plotLinesAndBandsGroups[plotGroup] =
- axis.plotLinesAndBandsGroups[plotGroup].destroy();
- }
- // Delete all properties and fall back to the prototype.
- objectEach(axis, function (val, key) {
- if (axis.getKeepProps().indexOf(key) === -1) {
- delete axis[key];
- }
- });
- };
- /**
- * Internal function to draw a crosshair.
- *
- * @function Highcharts.Axis#drawCrosshair
- *
- * @param {Highcharts.PointerEventObject} [e]
- * The event arguments from the modified pointer event, extended with
- * `chartX` and `chartY`
- *
- * @param {Highcharts.Point} [point]
- * The Point object if the crosshair snaps to points.
- *
- * @fires Highcharts.Axis#event:afterDrawCrosshair
- * @fires Highcharts.Axis#event:drawCrosshair
- */
- Axis.prototype.drawCrosshair = function (e, point) {
- var path,
- options = this.crosshair,
- snap = pick(options.snap,
- true),
- pos,
- categorized,
- graphic = this.cross,
- crossOptions,
- chart = this.chart;
- fireEvent(this, 'drawCrosshair', { e: e, point: point });
- // Use last available event when updating non-snapped crosshairs without
- // mouse interaction (#5287)
- if (!e) {
- e = this.cross && this.cross.e;
- }
- if (
- // Disabled in options
- !this.crosshair ||
- // Snap
- ((defined(point) || !snap) === false)) {
- this.hideCrosshair();
- }
- else {
- // Get the path
- if (!snap) {
- pos = e &&
- (this.horiz ?
- e.chartX - this.pos :
- this.len - e.chartY + this.pos);
- }
- else if (defined(point)) {
- // #3834
- pos = pick(this.coll !== 'colorAxis' ?
- point.crosshairPos : // 3D axis extension
- null, this.isXAxis ?
- point.plotX :
- this.len - point.plotY);
- }
- if (defined(pos)) {
- crossOptions = {
- // value, only used on radial
- value: point && (this.isXAxis ?
- point.x :
- pick(point.stackY, point.y)),
- translatedValue: pos
- };
- if (chart.polar) {
- // Additional information required for crosshairs in
- // polar chart
- extend(crossOptions, {
- isCrosshair: true,
- chartX: e && e.chartX,
- chartY: e && e.chartY,
- point: point
- });
- }
- path = this.getPlotLinePath(crossOptions) ||
- null; // #3189
- }
- if (!defined(path)) {
- this.hideCrosshair();
- return;
- }
- categorized = this.categories && !this.isRadial;
- // Draw the cross
- if (!graphic) {
- this.cross = graphic = chart.renderer
- .path()
- .addClass('highcharts-crosshair highcharts-crosshair-' +
- (categorized ? 'category ' : 'thin ') +
- options.className)
- .attr({
- zIndex: pick(options.zIndex, 2)
- })
- .add();
- // Presentational attributes
- if (!chart.styledMode) {
- graphic.attr({
- stroke: options.color ||
- (categorized ?
- Color
- .parse('#ccd6eb')
- .setOpacity(0.25)
- .get() :
- '#cccccc'),
- 'stroke-width': pick(options.width, 1)
- }).css({
- 'pointer-events': 'none'
- });
- if (options.dashStyle) {
- graphic.attr({
- dashstyle: options.dashStyle
- });
- }
- }
- }
- graphic.show().attr({
- d: path
- });
- if (categorized && !options.width) {
- graphic.attr({
- 'stroke-width': this.transA
- });
- }
- this.cross.e = e;
- }
- fireEvent(this, 'afterDrawCrosshair', { e: e, point: point });
- };
- /**
- * Hide the crosshair if visible.
- *
- * @function Highcharts.Axis#hideCrosshair
- */
- Axis.prototype.hideCrosshair = function () {
- if (this.cross) {
- this.cross.hide();
- }
- fireEvent(this, 'afterHideCrosshair');
- };
- /**
- * Check whether the chart has vertical panning ('y' or 'xy' type).
- *
- * @private
- * @function Highcharts.Axis#hasVerticalPanning
- * @return {boolean}
- *
- */
- Axis.prototype.hasVerticalPanning = function () {
- var _a,
- _b;
- return /y/.test(((_b = (_a = this.chart.options.chart) === null || _a === void 0 ? void 0 : _a.panning) === null || _b === void 0 ? void 0 : _b.type) || '');
- };
- /**
- * Check whether the given value is a positive valid axis value.
- *
- * @private
- * @function Highcharts.Axis#validatePositiveValue
- *
- * @param {unknown} value
- * The axis value
- * @return {boolean}
- *
- */
- Axis.prototype.validatePositiveValue = function (value) {
- return isNumber(value) && value > 0;
- };
- /* *
- *
- * Static Properties
- *
- * */
- /**
- * The X axis or category axis. Normally this is the horizontal axis,
- * though if the chart is inverted this is the vertical axis. In case of
- * multiple axes, the xAxis node is an array of configuration objects.
- *
- * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic
- * access to the axis.
- *
- * @productdesc {highmaps}
- * In Highmaps, the axis is hidden, but it is used behind the scenes to
- * control features like zooming and panning. Zooming is in effect the same
- * as setting the extremes of one of the exes.
- *
- * @type {*|Array<*>}
- * @optionparent xAxis
- *
- * @private
- */
- Axis.defaultOptions = {
- /**
- * When using multiple axis, the ticks of two or more opposite axes
- * will automatically be aligned by adding ticks to the axis or axes
- * with the least ticks, as if `tickAmount` were specified.
- *
- * This can be prevented by setting `alignTicks` to false. If the grid
- * lines look messy, it's a good idea to hide them for the secondary
- * axis by setting `gridLineWidth` to 0.
- *
- * If `startOnTick` or `endOnTick` in an Axis options are set to false,
- * then the `alignTicks ` will be disabled for the Axis.
- *
- * Disabled for logarithmic axes.
- *
- * @type {boolean}
- * @default true
- * @product highcharts highstock gantt
- * @apioption xAxis.alignTicks
- */
- /**
- * Whether to allow decimals in this axis' ticks. When counting
- * integers, like persons or hits on a web page, decimals should
- * be avoided in the labels.
- *
- * @see [minTickInterval](#xAxis.minTickInterval)
- *
- * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-true/
- * True by default
- * @sample {highcharts|highstock} highcharts/yaxis/allowdecimals-false/
- * False
- *
- * @type {boolean}
- * @default true
- * @since 2.0
- * @apioption xAxis.allowDecimals
- */
- /**
- * When using an alternate grid color, a band is painted across the
- * plot area between every other grid line.
- *
- * @sample {highcharts} highcharts/yaxis/alternategridcolor/
- * Alternate grid color on the Y axis
- * @sample {highstock} stock/xaxis/alternategridcolor/
- * Alternate grid color on the Y axis
- *
- * @type {Highcharts.ColorType}
- * @apioption xAxis.alternateGridColor
- */
- /**
- * An array defining breaks in the axis, the sections defined will be
- * left out and all the points shifted closer to each other.
- *
- * @productdesc {highcharts}
- * Requires that the broken-axis.js module is loaded.
- *
- * @sample {highcharts} highcharts/axisbreak/break-simple/
- * Simple break
- * @sample {highcharts|highstock} highcharts/axisbreak/break-visualized/
- * Advanced with callback
- * @sample {highstock} stock/demo/intraday-breaks/
- * Break on nights and weekends
- *
- * @type {Array<*>}
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.breaks
- */
- /**
- * A number indicating how much space should be left between the start
- * and the end of the break. The break size is given in axis units,
- * so for instance on a `datetime` axis, a break size of 3600000 would
- * indicate the equivalent of an hour.
- *
- * @type {number}
- * @default 0
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.breaks.breakSize
- */
- /**
- * The point where the break starts.
- *
- * @type {number}
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.breaks.from
- */
- /**
- * Defines an interval after which the break appears again. By default
- * the breaks do not repeat.
- *
- * @type {number}
- * @default 0
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.breaks.repeat
- */
- /**
- * The point where the break ends.
- *
- * @type {number}
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.breaks.to
- */
- /**
- * If categories are present for the xAxis, names are used instead of
- * numbers for that axis.
- *
- * Since Highcharts 3.0, categories can also
- * be extracted by giving each point a [name](#series.data) and setting
- * axis [type](#xAxis.type) to `category`. However, if you have multiple
- * series, best practice remains defining the `categories` array.
- *
- * Example: `categories: ['Apples', 'Bananas', 'Oranges']`
- *
- * @sample {highcharts} highcharts/demo/line-labels/
- * With
- * @sample {highcharts} highcharts/xaxis/categories/
- * Without
- *
- * @type {Array<string>}
- * @product highcharts gantt
- * @apioption xAxis.categories
- */
- /**
- * The highest allowed value for automatically computed axis extremes.
- *
- * @see [floor](#xAxis.floor)
- *
- * @sample {highcharts|highstock} highcharts/yaxis/floor-ceiling/
- * Floor and ceiling
- *
- * @type {number}
- * @since 4.0
- * @product highcharts highstock gantt
- * @apioption xAxis.ceiling
- */
- /**
- * A class name that opens for styling the axis by CSS, especially in
- * Highcharts styled mode. The class name is applied to group elements
- * for the grid, axis elements and labels.
- *
- * @sample {highcharts|highstock|highmaps} highcharts/css/axis/
- * Multiple axes with separate styling
- *
- * @type {string}
- * @since 5.0.0
- * @apioption xAxis.className
- */
- /**
- * Configure a crosshair that follows either the mouse pointer or the
- * hovered point.
- *
- * In styled mode, the crosshairs are styled in the
- * `.highcharts-crosshair`, `.highcharts-crosshair-thin` or
- * `.highcharts-xaxis-category` classes.
- *
- * @productdesc {highstock}
- * In Highstock, by default, the crosshair is enabled on the X axis and
- * disabled on the Y axis.
- *
- * @sample {highcharts} highcharts/xaxis/crosshair-both/
- * Crosshair on both axes
- * @sample {highstock} stock/xaxis/crosshairs-xy/
- * Crosshair on both axes
- * @sample {highmaps} highcharts/xaxis/crosshair-both/
- * Crosshair on both axes
- *
- * @declare Highcharts.AxisCrosshairOptions
- * @type {boolean|*}
- * @default false
- * @since 4.1
- * @apioption xAxis.crosshair
- */
- /**
- * A class name for the crosshair, especially as a hook for styling.
- *
- * @type {string}
- * @since 5.0.0
- * @apioption xAxis.crosshair.className
- */
- /**
- * The color of the crosshair. Defaults to `#cccccc` for numeric and
- * datetime axes, and `rgba(204,214,235,0.25)` for category axes, where
- * the crosshair by default highlights the whole category.
- *
- * @sample {highcharts|highstock|highmaps} highcharts/xaxis/crosshair-customized/
- * Customized crosshairs
- *
- * @type {Highcharts.ColorType}
- * @default #cccccc
- * @since 4.1
- * @apioption xAxis.crosshair.color
- */
- /**
- * The dash style for the crosshair. See
- * [plotOptions.series.dashStyle](#plotOptions.series.dashStyle)
- * for possible values.
- *
- * @sample {highcharts|highmaps} highcharts/xaxis/crosshair-dotted/
- * Dotted crosshair
- * @sample {highstock} stock/xaxis/crosshair-dashed/
- * Dashed X axis crosshair
- *
- * @type {Highcharts.DashStyleValue}
- * @default Solid
- * @since 4.1
- * @apioption xAxis.crosshair.dashStyle
- */
- /**
- * A label on the axis next to the crosshair.
- *
- * In styled mode, the label is styled with the
- * `.highcharts-crosshair-label` class.
- *
- * @sample {highstock} stock/xaxis/crosshair-label/
- * Crosshair labels
- * @sample {highstock} highcharts/css/crosshair-label/
- * Style mode
- *
- * @declare Highcharts.AxisCrosshairLabelOptions
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label
- */
- /**
- * Alignment of the label compared to the axis. Defaults to `"left"` for
- * right-side axes, `"right"` for left-side axes and `"center"` for
- * horizontal axes.
- *
- * @type {Highcharts.AlignValue}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.align
- */
- /**
- * The background color for the label. Defaults to the related series
- * color, or `#666666` if that is not available.
- *
- * @type {Highcharts.ColorType}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.backgroundColor
- */
- /**
- * The border color for the crosshair label
- *
- * @type {Highcharts.ColorType}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.borderColor
- */
- /**
- * The border corner radius of the crosshair label.
- *
- * @type {number}
- * @default 3
- * @since 2.1.10
- * @product highstock
- * @apioption xAxis.crosshair.label.borderRadius
- */
- /**
- * The border width for the crosshair label.
- *
- * @type {number}
- * @default 0
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.borderWidth
- */
- /**
- * Flag to enable crosshair's label.
- *
- * @sample {highstock} stock/xaxis/crosshairs-xy/
- * Enabled label for yAxis' crosshair
- *
- * @type {boolean}
- * @default false
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.enabled
- */
- /**
- * A format string for the crosshair label. Defaults to `{value}` for
- * numeric axes and `{value:%b %d, %Y}` for datetime axes.
- *
- * @type {string}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.format
- */
- /**
- * Formatter function for the label text.
- *
- * @type {Highcharts.XAxisCrosshairLabelFormatterCallbackFunction}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.formatter
- */
- /**
- * Padding inside the crosshair label.
- *
- * @type {number}
- * @default 8
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.padding
- */
- /**
- * The shape to use for the label box.
- *
- * @type {string}
- * @default callout
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.shape
- */
- /**
- * Text styles for the crosshair label.
- *
- * @type {Highcharts.CSSObject}
- * @default {"color": "white", "fontWeight": "normal", "fontSize": "11px", "textAlign": "center"}
- * @since 2.1
- * @product highstock
- * @apioption xAxis.crosshair.label.style
- */
- /**
- * Whether the crosshair should snap to the point or follow the pointer
- * independent of points.
- *
- * @sample {highcharts|highstock} highcharts/xaxis/crosshair-snap-false/
- * True by default
- * @sample {highmaps} maps/demo/latlon-advanced/
- * Snap is false
- *
- * @type {boolean}
- * @default true
- * @since 4.1
- * @apioption xAxis.crosshair.snap
- */
- /**
- * The pixel width of the crosshair. Defaults to 1 for numeric or
- * datetime axes, and for one category width for category axes.
- *
- * @sample {highcharts} highcharts/xaxis/crosshair-customized/
- * Customized crosshairs
- * @sample {highstock} highcharts/xaxis/crosshair-customized/
- * Customized crosshairs
- * @sample {highmaps} highcharts/xaxis/crosshair-customized/
- * Customized crosshairs
- *
- * @type {number}
- * @default 1
- * @since 4.1
- * @apioption xAxis.crosshair.width
- */
- /**
- * The Z index of the crosshair. Higher Z indices allow drawing the
- * crosshair on top of the series or behind the grid lines.
- *
- * @type {number}
- * @default 2
- * @since 4.1
- * @apioption xAxis.crosshair.zIndex
- */
- /**
- * Whether to zoom axis. If `chart.zoomType` is set, the option allows
- * to disable zooming on an individual axis.
- *
- * @sample {highcharts} highcharts/xaxis/zoomenabled/
- * Zoom enabled is false
- *
- *
- * @type {boolean}
- * @default enabled
- * @apioption xAxis.zoomEnabled
- */
- /**
- * For a datetime axis, the scale will automatically adjust to the
- * appropriate unit. This member gives the default string
- * representations used for each unit. For intermediate values,
- * different units may be used, for example the `day` unit can be used
- * on midnight and `hour` unit be used for intermediate values on the
- * same axis. For an overview of the replacement codes, see
- * [dateFormat](/class-reference/Highcharts#dateFormat).
- *
- * Defaults to:
- * ```js
- * {
- * millisecond: '%H:%M:%S.%L',
- * second: '%H:%M:%S',
- * minute: '%H:%M',
- * hour: '%H:%M',
- * day: '%e. %b',
- * week: '%e. %b',
- * month: '%b \'%y',
- * year: '%Y'
- * }
- * ```
- *
- * @sample {highcharts} highcharts/xaxis/datetimelabelformats/
- * Different day format on X axis
- * @sample {highstock} stock/xaxis/datetimelabelformats/
- * More information in x axis labels
- *
- * @declare Highcharts.AxisDateTimeLabelFormatsOptions
- * @product highcharts highstock
- */
- dateTimeLabelFormats: {
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- millisecond: {
- main: '%H:%M:%S.%L',
- range: false
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- second: {
- main: '%H:%M:%S',
- range: false
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- minute: {
- main: '%H:%M',
- range: false
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- hour: {
- main: '%H:%M',
- range: false
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- day: {
- main: '%e. %b'
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- week: {
- main: '%e. %b'
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- month: {
- main: '%b \'%y'
- },
- /**
- * @declare Highcharts.AxisDateTimeLabelFormatsOptionsObject
- * @type {string|*}
- */
- year: {
- main: '%Y'
- }
- },
- /**
- * Whether to force the axis to end on a tick. Use this option with
- * the `maxPadding` option to control the axis end.
- *
- * @productdesc {highstock}
- * In Highstock, `endOnTick` is always `false` when the navigator
- * is enabled, to prevent jumpy scrolling.
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * True by default
- * @sample {highcharts} highcharts/yaxis/endontick/
- * False
- * @sample {highstock} stock/demo/basic-line/
- * True by default
- * @sample {highstock} stock/xaxis/endontick/
- * False
- *
- * @since 1.2.0
- */
- endOnTick: false,
- /**
- * Event handlers for the axis.
- *
- * @type {*}
- * @apioption xAxis.events
- */
- /**
- * An event fired after the breaks have rendered.
- *
- * @see [breaks](#xAxis.breaks)
- *
- * @sample {highcharts} highcharts/axisbreak/break-event/
- * AfterBreak Event
- *
- * @type {Highcharts.AxisEventCallbackFunction}
- * @since 4.1.0
- * @product highcharts gantt
- * @apioption xAxis.events.afterBreaks
- */
- /**
- * As opposed to the `setExtremes` event, this event fires after the
- * final min and max values are computed and corrected for `minRange`.
- *
- * Fires when the minimum and maximum is set for the axis, either by
- * calling the `.setExtremes()` method or by selecting an area in the
- * chart. One parameter, `event`, is passed to the function, containing
- * common event information.
- *
- * The new user set minimum and maximum values can be found by
- * `event.min` and `event.max`. These reflect the axis minimum and
- * maximum in axis values. The actual data extremes are found in
- * `event.dataMin` and `event.dataMax`.
- *
- * @type {Highcharts.AxisSetExtremesEventCallbackFunction}
- * @since 2.3
- * @context Highcharts.Axis
- * @apioption xAxis.events.afterSetExtremes
- */
- /**
- * An event fired when a break from this axis occurs on a point.
- *
- * @see [breaks](#xAxis.breaks)
- *
- * @sample {highcharts} highcharts/axisbreak/break-visualized/
- * Visualization of a Break
- *
- * @type {Highcharts.AxisPointBreakEventCallbackFunction}
- * @since 4.1.0
- * @product highcharts gantt
- * @context Highcharts.Axis
- * @apioption xAxis.events.pointBreak
- */
- /**
- * An event fired when a point falls inside a break from this axis.
- *
- * @type {Highcharts.AxisPointBreakEventCallbackFunction}
- * @product highcharts highstock gantt
- * @context Highcharts.Axis
- * @apioption xAxis.events.pointInBreak
- */
- /**
- * Fires when the minimum and maximum is set for the axis, either by
- * calling the `.setExtremes()` method or by selecting an area in the
- * chart. One parameter, `event`, is passed to the function,
- * containing common event information.
- *
- * The new user set minimum and maximum values can be found by
- * `event.min` and `event.max`. These reflect the axis minimum and
- * maximum in data values. When an axis is zoomed all the way out from
- * the "Reset zoom" button, `event.min` and `event.max` are null, and
- * the new extremes are set based on `this.dataMin` and `this.dataMax`.
- *
- * @sample {highstock} stock/xaxis/events-setextremes/
- * Log new extremes on x axis
- *
- * @type {Highcharts.AxisSetExtremesEventCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Axis
- * @apioption xAxis.events.setExtremes
- */
- /**
- * The lowest allowed value for automatically computed axis extremes.
- *
- * @see [ceiling](#yAxis.ceiling)
- *
- * @sample {highcharts} highcharts/yaxis/floor-ceiling/
- * Floor and ceiling
- * @sample {highstock} stock/demo/lazy-loading/
- * Prevent negative stock price on Y axis
- *
- * @type {number}
- * @since 4.0
- * @product highcharts highstock gantt
- * @apioption xAxis.floor
- */
- /**
- * The dash or dot style of the grid lines. For possible values, see
- * [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/).
- *
- * @sample {highcharts} highcharts/yaxis/gridlinedashstyle/
- * Long dashes
- * @sample {highstock} stock/xaxis/gridlinedashstyle/
- * Long dashes
- *
- * @type {Highcharts.DashStyleValue}
- * @default Solid
- * @since 1.2
- * @apioption xAxis.gridLineDashStyle
- */
- /**
- * The Z index of the grid lines.
- *
- * @sample {highcharts|highstock} highcharts/xaxis/gridzindex/
- * A Z index of 4 renders the grid above the graph
- *
- * @type {number}
- * @default 1
- * @product highcharts highstock gantt
- * @apioption xAxis.gridZIndex
- */
- /**
- * An id for the axis. This can be used after render time to get
- * a pointer to the axis object through `chart.get()`.
- *
- * @sample {highcharts} highcharts/xaxis/id/
- * Get the object
- * @sample {highstock} stock/xaxis/id/
- * Get the object
- *
- * @type {string}
- * @since 1.2.0
- * @apioption xAxis.id
- */
- /**
- * The axis labels show the number or category for each tick.
- *
- * Since v8.0.0: Labels are animated in categorized x-axis with
- * updating data if `tickInterval` and `step` is set to 1.
- *
- * @productdesc {highmaps}
- * X and Y axis labels are by default disabled in Highmaps, but the
- * functionality is inherited from Highcharts and used on `colorAxis`,
- * and can be enabled on X and Y axes too.
- */
- labels: {
- /**
- * What part of the string the given position is anchored to.
- * If `left`, the left side of the string is at the axis position.
- * Can be one of `"left"`, `"center"` or `"right"`. Defaults to
- * an intelligent guess based on which side of the chart the axis
- * is on and the rotation of the label.
- *
- * @see [reserveSpace](#xAxis.labels.reserveSpace)
- *
- * @sample {highcharts} highcharts/xaxis/labels-align-left/
- * Left
- * @sample {highcharts} highcharts/xaxis/labels-align-right/
- * Right
- * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/
- * Left-aligned labels on a vertical category axis
- *
- * @type {Highcharts.AlignValue}
- * @apioption xAxis.labels.align
- */
- /**
- * For horizontal axes, the allowed degrees of label rotation
- * to prevent overlapping labels. If there is enough space,
- * labels are not rotated. As the chart gets narrower, it
- * will start rotating the labels -45 degrees, then remove
- * every second label and try again with rotations 0 and -45 etc.
- * Set it to `false` to disable rotation, which will
- * cause the labels to word-wrap if possible.
- *
- * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-default/
- * Default auto rotation of 0 or -45
- * @sample {highcharts|highstock} highcharts/xaxis/labels-autorotation-0-90/
- * Custom graded auto rotation
- *
- * @type {Array<number>|false}
- * @default [-45]
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.labels.autoRotation
- */
- /**
- * When each category width is more than this many pixels, we don't
- * apply auto rotation. Instead, we lay out the axis label with word
- * wrap. A lower limit makes sense when the label contains multiple
- * short words that don't extend the available horizontal space for
- * each label.
- *
- * @sample {highcharts} highcharts/xaxis/labels-autorotationlimit/
- * Lower limit
- *
- * @type {number}
- * @default 80
- * @since 4.1.5
- * @product highcharts gantt
- * @apioption xAxis.labels.autoRotationLimit
- */
- /**
- * Polar charts only. The label's pixel distance from the perimeter
- * of the plot area.
- *
- * @type {number}
- * @default 15
- * @product highcharts gantt
- * @apioption xAxis.labels.distance
- */
- /**
- * Enable or disable the axis labels.
- *
- * @sample {highcharts} highcharts/xaxis/labels-enabled/
- * X axis labels disabled
- * @sample {highstock} stock/xaxis/labels-enabled/
- * X axis labels disabled
- *
- * @default {highcharts|highstock|gantt} true
- * @default {highmaps} false
- */
- enabled: true,
- /**
- * A [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting)
- * for the axis label.
- *
- * @sample {highcharts|highstock} highcharts/yaxis/labels-format/
- * Add units to Y axis label
- *
- * @type {string}
- * @default {value}
- * @since 3.0
- * @apioption xAxis.labels.format
- */
- /**
- * Callback JavaScript function to format the label. The value
- * is given by `this.value`. Additional properties for `this` are
- * `axis`, `chart`, `isFirst` and `isLast`. The value of the default
- * label formatter can be retrieved by calling
- * `this.axis.defaultLabelFormatter.call(this)` within the function.
- *
- * Defaults to:
- * ```js
- * function() {
- * return this.value;
- * }
- * ```
- *
- * @sample {highcharts} highcharts/xaxis/labels-formatter-linked/
- * Linked category names
- * @sample {highcharts} highcharts/xaxis/labels-formatter-extended/
- * Modified numeric labels
- * @sample {highstock} stock/xaxis/labels-formatter/
- * Added units on Y axis
- *
- * @type {Highcharts.AxisLabelsFormatterCallbackFunction}
- * @apioption xAxis.labels.formatter
- */
- /**
- * The number of pixels to indent the labels per level in a treegrid
- * axis.
- *
- * @sample gantt/treegrid-axis/demo
- * Indentation 10px by default.
- * @sample gantt/treegrid-axis/indentation-0px
- * Indentation set to 0px.
- *
- * @product gantt
- */
- indentation: 10,
- /**
- * Horizontal axis only. When `staggerLines` is not set,
- * `maxStaggerLines` defines how many lines the axis is allowed to
- * add to automatically avoid overlapping X labels. Set to `1` to
- * disable overlap detection.
- *
- * @deprecated
- * @type {number}
- * @default 5
- * @since 1.3.3
- * @apioption xAxis.labels.maxStaggerLines
- */
- /**
- * How to handle overflowing labels on horizontal axis. If set to
- * `"allow"`, it will not be aligned at all. By default it
- * `"justify"` labels inside the chart area. If there is room to
- * move it, it will be aligned to the edge, else it will be removed.
- *
- * @type {string}
- * @default justify
- * @since 2.2.5
- * @validvalue ["allow", "justify"]
- * @apioption xAxis.labels.overflow
- */
- /**
- * The pixel padding for axis labels, to ensure white space between
- * them.
- *
- * @type {number}
- * @default 5
- * @product highcharts gantt
- * @apioption xAxis.labels.padding
- */
- /**
- * Whether to reserve space for the labels. By default, space is
- * reserved for the labels in these cases:
- *
- * * On all horizontal axes.
- * * On vertical axes if `label.align` is `right` on a left-side
- * axis or `left` on a right-side axis.
- * * On vertical axes if `label.align` is `center`.
- *
- * This can be turned off when for example the labels are rendered
- * inside the plot area instead of outside.
- *
- * @see [labels.align](#xAxis.labels.align)
- *
- * @sample {highcharts} highcharts/xaxis/labels-reservespace/
- * No reserved space, labels inside plot
- * @sample {highcharts} highcharts/xaxis/labels-reservespace-true/
- * Left-aligned labels on a vertical category axis
- *
- * @type {boolean}
- * @since 4.1.10
- * @product highcharts gantt
- * @apioption xAxis.labels.reserveSpace
- */
- /**
- * Rotation of the labels in degrees.
- *
- * @sample {highcharts} highcharts/xaxis/labels-rotation/
- * X axis labels rotated 90°
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.labels.rotation
- */
- /**
- * Horizontal axes only. The number of lines to spread the labels
- * over to make room or tighter labels.
- *
- * @sample {highcharts} highcharts/xaxis/labels-staggerlines/
- * Show labels over two lines
- * @sample {highstock} stock/xaxis/labels-staggerlines/
- * Show labels over two lines
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.labels.staggerLines
- */
- /**
- * To show only every _n_'th label on the axis, set the step to _n_.
- * Setting the step to 2 shows every other label.
- *
- * By default, the step is calculated automatically to avoid
- * overlap. To prevent this, set it to 1\. This usually only
- * happens on a category axis, and is often a sign that you have
- * chosen the wrong axis type.
- *
- * Read more at
- * [Axis docs](https://www.highcharts.com/docs/chart-concepts/axes)
- * => What axis should I use?
- *
- * @sample {highcharts} highcharts/xaxis/labels-step/
- * Showing only every other axis label on a categorized
- * x-axis
- * @sample {highcharts} highcharts/xaxis/labels-step-auto/
- * Auto steps on a category axis
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.labels.step
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the labels.
- *
- * @type {boolean}
- * @default false
- * @apioption xAxis.labels.useHTML
- */
- /**
- * The x position offset of all labels relative to the tick
- * positions on the axis.
- *
- * @sample {highcharts} highcharts/xaxis/labels-x/
- * Y axis labels placed on grid lines
- */
- x: 0,
- /**
- * The y position offset of all labels relative to the tick
- * positions on the axis. The default makes it adapt to the font
- * size of the bottom axis.
- *
- * @sample {highcharts} highcharts/xaxis/labels-x/
- * Y axis labels placed on grid lines
- *
- * @type {number}
- * @apioption xAxis.labels.y
- */
- /**
- * The Z index for the axis labels.
- *
- * @type {number}
- * @default 7
- * @apioption xAxis.labels.zIndex
- */
- /**
- * CSS styles for the label. Use `whiteSpace: 'nowrap'` to prevent
- * wrapping of category labels. Use `textOverflow: 'none'` to
- * prevent ellipsis (dots).
- *
- * In styled mode, the labels are styled with the
- * `.highcharts-axis-labels` class.
- *
- * @sample {highcharts} highcharts/xaxis/labels-style/
- * Red X axis labels
- *
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @internal */
- color: '#666666',
- /** @internal */
- cursor: 'default',
- /** @internal */
- fontSize: '11px'
- }
- },
- /**
- * The left position as the horizontal axis. If it's a number, it is
- * interpreted as pixel position relative to the chart.
- *
- * Since Highcharts v5.0.13: If it's a percentage string, it is
- * interpreted as percentages of the plot width, offset from plot area
- * left.
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption xAxis.left
- */
- /**
- * The top position as the vertical axis. If it's a number, it is
- * interpreted as pixel position relative to the chart.
- *
- * Since Highcharts 2: If it's a percentage string, it is interpreted
- * as percentages of the plot height, offset from plot area top.
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption xAxis.top
- */
- /**
- * Index of another axis that this axis is linked to. When an axis is
- * linked to a master axis, it will take the same extremes as
- * the master, but as assigned by min or max or by setExtremes.
- * It can be used to show additional info, or to ease reading the
- * chart by duplicating the scales.
- *
- * @sample {highcharts} highcharts/xaxis/linkedto/
- * Different string formats of the same date
- * @sample {highcharts} highcharts/yaxis/linkedto/
- * Y values on both sides
- *
- * @type {number}
- * @since 2.0.2
- * @product highcharts highstock gantt
- * @apioption xAxis.linkedTo
- */
- /**
- * The maximum value of the axis. If `null`, the max value is
- * automatically calculated.
- *
- * If the [endOnTick](#yAxis.endOnTick) option is true, the `max` value
- * might be rounded up.
- *
- * If a [tickAmount](#yAxis.tickAmount) is set, the axis may be extended
- * beyond the set max in order to reach the given number of ticks. The
- * same may happen in a chart with multiple axes, determined by [chart.
- * alignTicks](#chart), where a `tickAmount` is applied internally.
- *
- * @sample {highcharts} highcharts/yaxis/max-200/
- * Y axis max of 200
- * @sample {highcharts} highcharts/yaxis/max-logarithmic/
- * Y axis max on logarithmic axis
- * @sample {highstock} stock/xaxis/min-max/
- * Fixed min and max on X axis
- * @sample {highmaps} maps/axis/min-max/
- * Pre-zoomed to a specific area
- *
- * @type {number|null}
- * @apioption xAxis.max
- */
- /**
- * Padding of the max value relative to the length of the axis. A
- * padding of 0.05 will make a 100px axis 5px longer. This is useful
- * when you don't want the highest data value to appear on the edge
- * of the plot area. When the axis' `max` option is set or a max extreme
- * is set using `axis.setExtremes()`, the maxPadding will be ignored.
- *
- * @sample {highcharts} highcharts/yaxis/maxpadding/
- * Max padding of 0.25 on y axis
- * @sample {highstock} stock/xaxis/minpadding-maxpadding/
- * Greater min- and maxPadding
- * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/
- * Add some padding
- *
- * @default {highcharts} 0.01
- * @default {highstock|highmaps} 0
- * @since 1.2.0
- */
- maxPadding: 0.01,
- /**
- * Deprecated. Use `minRange` instead.
- *
- * @deprecated
- * @type {number}
- * @product highcharts highstock
- * @apioption xAxis.maxZoom
- */
- /**
- * The minimum value of the axis. If `null` the min value is
- * automatically calculated.
- *
- * If the [startOnTick](#yAxis.startOnTick) option is true (default),
- * the `min` value might be rounded down.
- *
- * The automatically calculated minimum value is also affected by
- * [floor](#yAxis.floor), [softMin](#yAxis.softMin),
- * [minPadding](#yAxis.minPadding), [minRange](#yAxis.minRange)
- * as well as [series.threshold](#plotOptions.series.threshold)
- * and [series.softThreshold](#plotOptions.series.softThreshold).
- *
- * @sample {highcharts} highcharts/yaxis/min-startontick-false/
- * -50 with startOnTick to false
- * @sample {highcharts} highcharts/yaxis/min-startontick-true/
- * -50 with startOnTick true by default
- * @sample {highstock} stock/xaxis/min-max/
- * Set min and max on X axis
- * @sample {highmaps} maps/axis/min-max/
- * Pre-zoomed to a specific area
- *
- * @type {number|null}
- * @apioption xAxis.min
- */
- /**
- * The dash or dot style of the minor grid lines. For possible values,
- * see [this demonstration](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/).
- *
- * @sample {highcharts} highcharts/yaxis/minorgridlinedashstyle/
- * Long dashes on minor grid lines
- * @sample {highstock} stock/xaxis/minorgridlinedashstyle/
- * Long dashes on minor grid lines
- *
- * @type {Highcharts.DashStyleValue}
- * @default Solid
- * @since 1.2
- * @apioption xAxis.minorGridLineDashStyle
- */
- /**
- * Specific tick interval in axis units for the minor ticks. On a linear
- * axis, if `"auto"`, the minor tick interval is calculated as a fifth
- * of the tickInterval. If `null` or `undefined`, minor ticks are not
- * shown.
- *
- * On logarithmic axes, the unit is the power of the value. For example,
- * setting the minorTickInterval to 1 puts one tick on each of 0.1, 1,
- * 10, 100 etc. Setting the minorTickInterval to 0.1 produces 9 ticks
- * between 1 and 10, 10 and 100 etc.
- *
- * If user settings dictate minor ticks to become too dense, they don't
- * make sense, and will be ignored to prevent performance problems.
- *
- * @sample {highcharts} highcharts/yaxis/minortickinterval-null/
- * Null by default
- * @sample {highcharts} highcharts/yaxis/minortickinterval-5/
- * 5 units
- * @sample {highcharts} highcharts/yaxis/minortickinterval-log-auto/
- * "auto"
- * @sample {highcharts} highcharts/yaxis/minortickinterval-log/
- * 0.1
- * @sample {highstock} stock/demo/basic-line/
- * Null by default
- * @sample {highstock} stock/xaxis/minortickinterval-auto/
- * "auto"
- *
- * @type {number|string|null}
- * @apioption xAxis.minorTickInterval
- */
- /**
- * The pixel length of the minor tick marks.
- *
- * @sample {highcharts} highcharts/yaxis/minorticklength/
- * 10px on Y axis
- * @sample {highstock} stock/xaxis/minorticks/
- * 10px on Y axis
- */
- minorTickLength: 2,
- /**
- * The position of the minor tick marks relative to the axis line.
- * Can be one of `inside` and `outside`.
- *
- * @sample {highcharts} highcharts/yaxis/minortickposition-outside/
- * Outside by default
- * @sample {highcharts} highcharts/yaxis/minortickposition-inside/
- * Inside
- * @sample {highstock} stock/xaxis/minorticks/
- * Inside
- *
- * @validvalue ["inside", "outside"]
- */
- minorTickPosition: 'outside',
- /**
- * Enable or disable minor ticks. Unless
- * [minorTickInterval](#xAxis.minorTickInterval) is set, the tick
- * interval is calculated as a fifth of the `tickInterval`.
- *
- * On a logarithmic axis, minor ticks are laid out based on a best
- * guess, attempting to enter approximately 5 minor ticks between
- * each major tick.
- *
- * Prior to v6.0.0, ticks were unabled in auto layout by setting
- * `minorTickInterval` to `"auto"`.
- *
- * @productdesc {highcharts}
- * On axes using [categories](#xAxis.categories), minor ticks are not
- * supported.
- *
- * @sample {highcharts} highcharts/yaxis/minorticks-true/
- * Enabled on linear Y axis
- *
- * @type {boolean}
- * @default false
- * @since 6.0.0
- * @apioption xAxis.minorTicks
- */
- /**
- * The pixel width of the minor tick mark.
- *
- * @sample {highcharts} highcharts/yaxis/minortickwidth/
- * 3px width
- * @sample {highstock} stock/xaxis/minorticks/
- * 1px width
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.minorTickWidth
- */
- /**
- * Padding of the min value relative to the length of the axis. A
- * padding of 0.05 will make a 100px axis 5px longer. This is useful
- * when you don't want the lowest data value to appear on the edge
- * of the plot area. When the axis' `min` option is set or a min extreme
- * is set using `axis.setExtremes()`, the minPadding will be ignored.
- *
- * @sample {highcharts} highcharts/yaxis/minpadding/
- * Min padding of 0.2
- * @sample {highstock} stock/xaxis/minpadding-maxpadding/
- * Greater min- and maxPadding
- * @sample {highmaps} maps/chart/plotbackgroundcolor-gradient/
- * Add some padding
- *
- * @default {highcharts} 0.01
- * @default {highstock|highmaps} 0
- * @since 1.2.0
- * @product highcharts highstock gantt
- */
- minPadding: 0.01,
- /**
- * The minimum range to display on this axis. The entire axis will not
- * be allowed to span over a smaller interval than this. For example,
- * for a datetime axis the main unit is milliseconds. If minRange is
- * set to 3600000, you can't zoom in more than to one hour.
- *
- * The default minRange for the x axis is five times the smallest
- * interval between any of the data points.
- *
- * On a logarithmic axis, the unit for the minimum range is the power.
- * So a minRange of 1 means that the axis can be zoomed to 10-100,
- * 100-1000, 1000-10000 etc.
- *
- * **Note**: The `minPadding`, `maxPadding`, `startOnTick` and
- * `endOnTick` settings also affect how the extremes of the axis
- * are computed.
- *
- * @sample {highcharts} highcharts/xaxis/minrange/
- * Minimum range of 5
- * @sample {highstock} stock/xaxis/minrange/
- * Max zoom of 6 months overrides user selections
- * @sample {highmaps} maps/axis/minrange/
- * Minimum range of 1000
- *
- * @type {number}
- * @apioption xAxis.minRange
- */
- /**
- * The minimum tick interval allowed in axis values. For example on
- * zooming in on an axis with daily data, this can be used to prevent
- * the axis from showing hours. Defaults to the closest distance between
- * two points on the axis.
- *
- * @type {number}
- * @since 2.3.0
- * @apioption xAxis.minTickInterval
- */
- /**
- * The distance in pixels from the plot area to the axis line.
- * A positive offset moves the axis with it's line, labels and ticks
- * away from the plot area. This is typically used when two or more
- * axes are displayed on the same side of the plot. With multiple
- * axes the offset is dynamically adjusted to avoid collision, this
- * can be overridden by setting offset explicitly.
- *
- * @sample {highcharts} highcharts/yaxis/offset/
- * Y axis offset of 70
- * @sample {highcharts} highcharts/yaxis/offset-centered/
- * Axes positioned in the center of the plot
- * @sample {highstock} stock/xaxis/offset/
- * Y axis offset by 70 px
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.offset
- */
- /**
- * Whether to display the axis on the opposite side of the normal. The
- * normal is on the left side for vertical axes and bottom for
- * horizontal, so the opposite sides will be right and top respectively.
- * This is typically used with dual or multiple axes.
- *
- * @sample {highcharts} highcharts/yaxis/opposite/
- * Secondary Y axis opposite
- * @sample {highstock} stock/xaxis/opposite/
- * Y axis on left side
- *
- * @type {boolean}
- * @default false
- * @apioption xAxis.opposite
- */
- /**
- * In an ordinal axis, the points are equally spaced in the chart
- * regardless of the actual time or x distance between them. This means
- * that missing data periods (e.g. nights or weekends for a stock chart)
- * will not take up space in the chart.
- * Having `ordinal: false` will show any gaps created by the `gapSize`
- * setting proportionate to their duration.
- *
- * In stock charts the X axis is ordinal by default, unless
- * the boost module is used and at least one of the series' data length
- * exceeds the [boostThreshold](#series.line.boostThreshold).
- *
- * @sample {highstock} stock/xaxis/ordinal-true/
- * True by default
- * @sample {highstock} stock/xaxis/ordinal-false/
- * False
- *
- * @type {boolean}
- * @default true
- * @since 1.1
- * @product highstock
- * @apioption xAxis.ordinal
- */
- /**
- * Additional range on the right side of the xAxis. Works similar to
- * `xAxis.maxPadding`, but value is set in milliseconds. Can be set for
- * both main `xAxis` and the navigator's `xAxis`.
- *
- * @sample {highstock} stock/xaxis/overscroll/
- * One minute overscroll with live data
- *
- * @type {number}
- * @default 0
- * @since 6.0.0
- * @product highstock
- * @apioption xAxis.overscroll
- */
- /**
- * Refers to the index in the [panes](#panes) array. Used for circular
- * gauges and polar charts. When the option is not set then first pane
- * will be used.
- *
- * @sample highcharts/demo/gauge-vu-meter
- * Two gauges with different center
- *
- * @type {number}
- * @product highcharts
- * @apioption xAxis.pane
- */
- /**
- * The zoomed range to display when only defining one or none of `min`
- * or `max`. For example, to show the latest month, a range of one month
- * can be set.
- *
- * @sample {highstock} stock/xaxis/range/
- * Setting a zoomed range when the rangeSelector is disabled
- *
- * @type {number}
- * @product highstock
- * @apioption xAxis.range
- */
- /**
- * Whether to reverse the axis so that the highest number is closest
- * to the origin. If the chart is inverted, the x axis is reversed by
- * default.
- *
- * @sample {highcharts} highcharts/yaxis/reversed/
- * Reversed Y axis
- * @sample {highstock} stock/xaxis/reversed/
- * Reversed Y axis
- *
- * @type {boolean}
- * @default false
- * @apioption xAxis.reversed
- */
- // reversed: false,
- /**
- * This option determines how stacks should be ordered within a group.
- * For example reversed xAxis also reverses stacks, so first series
- * comes last in a group. To keep order like for non-reversed xAxis
- * enable this option.
- *
- * @sample {highcharts} highcharts/xaxis/reversedstacks/
- * Reversed stacks comparison
- * @sample {highstock} highcharts/xaxis/reversedstacks/
- * Reversed stacks comparison
- *
- * @type {boolean}
- * @default false
- * @since 6.1.1
- * @product highcharts highstock
- * @apioption xAxis.reversedStacks
- */
- /**
- * An optional scrollbar to display on the X axis in response to
- * limiting the minimum and maximum of the axis values.
- *
- * In styled mode, all the presentational options for the scrollbar are
- * replaced by the classes `.highcharts-scrollbar-thumb`,
- * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`,
- * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`.
- *
- * @sample {highstock} stock/yaxis/heatmap-scrollbars/
- * Heatmap with both scrollbars
- *
- * @extends scrollbar
- * @since 4.2.6
- * @product highstock
- * @apioption xAxis.scrollbar
- */
- /**
- * Whether to show the axis line and title when the axis has no data.
- *
- * @sample {highcharts} highcharts/yaxis/showempty/
- * When clicking the legend to hide series, one axis preserves
- * line and title, the other doesn't
- * @sample {highstock} highcharts/yaxis/showempty/
- * When clicking the legend to hide series, one axis preserves
- * line and title, the other doesn't
- *
- * @since 1.1
- */
- showEmpty: true,
- /**
- * Whether to show the first tick label.
- *
- * @sample {highcharts} highcharts/xaxis/showfirstlabel-false/
- * Set to false on X axis
- * @sample {highstock} stock/xaxis/showfirstlabel/
- * Labels below plot lines on Y axis
- *
- * @type {boolean}
- * @default true
- * @apioption xAxis.showFirstLabel
- */
- /**
- * Whether to show the last tick label. Defaults to `true` on cartesian
- * charts, and `false` on polar charts.
- *
- * @sample {highcharts} highcharts/xaxis/showlastlabel-true/
- * Set to true on X axis
- * @sample {highstock} stock/xaxis/showfirstlabel/
- * Labels below plot lines on Y axis
- *
- * @type {boolean}
- * @default true
- * @product highcharts highstock gantt
- * @apioption xAxis.showLastLabel
- */
- /**
- * A soft maximum for the axis. If the series data maximum is less than
- * this, the axis will stay at this maximum, but if the series data
- * maximum is higher, the axis will flex to show all data.
- *
- * @sample highcharts/yaxis/softmin-softmax/
- * Soft min and max
- *
- * @type {number}
- * @since 5.0.1
- * @product highcharts highstock gantt
- * @apioption xAxis.softMax
- */
- /**
- * A soft minimum for the axis. If the series data minimum is greater
- * than this, the axis will stay at this minimum, but if the series
- * data minimum is lower, the axis will flex to show all data.
- *
- * @sample highcharts/yaxis/softmin-softmax/
- * Soft min and max
- *
- * @type {number}
- * @since 5.0.1
- * @product highcharts highstock gantt
- * @apioption xAxis.softMin
- */
- /**
- * For datetime axes, this decides where to put the tick between weeks.
- * 0 = Sunday, 1 = Monday.
- *
- * @sample {highcharts} highcharts/xaxis/startofweek-monday/
- * Monday by default
- * @sample {highcharts} highcharts/xaxis/startofweek-sunday/
- * Sunday
- * @sample {highstock} stock/xaxis/startofweek-1
- * Monday by default
- * @sample {highstock} stock/xaxis/startofweek-0
- * Sunday
- *
- * @product highcharts highstock gantt
- */
- startOfWeek: 1,
- /**
- * Whether to force the axis to start on a tick. Use this option with
- * the `minPadding` option to control the axis start.
- *
- * @productdesc {highstock}
- * In Highstock, `startOnTick` is always `false` when the navigator
- * is enabled, to prevent jumpy scrolling.
- *
- * @sample {highcharts} highcharts/xaxis/startontick-false/
- * False by default
- * @sample {highcharts} highcharts/xaxis/startontick-true/
- * True
- *
- * @since 1.2.0
- */
- startOnTick: false,
- /**
- * The amount of ticks to draw on the axis. This opens up for aligning
- * the ticks of multiple charts or panes within a chart. This option
- * overrides the `tickPixelInterval` option.
- *
- * This option only has an effect on linear axes. Datetime, logarithmic
- * or category axes are not affected.
- *
- * @sample {highcharts} highcharts/yaxis/tickamount/
- * 8 ticks on Y axis
- * @sample {highstock} highcharts/yaxis/tickamount/
- * 8 ticks on Y axis
- *
- * @type {number}
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @apioption xAxis.tickAmount
- */
- /**
- * The interval of the tick marks in axis units. When `undefined`, the
- * tick interval is computed to approximately follow the
- * [tickPixelInterval](#xAxis.tickPixelInterval) on linear and datetime
- * axes. On categorized axes, a `undefined` tickInterval will default to
- * 1, one category. Note that datetime axes are based on milliseconds,
- * so for example an interval of one day is expressed as
- * `24 * 3600 * 1000`.
- *
- * On logarithmic axes, the tickInterval is based on powers, so a
- * tickInterval of 1 means one tick on each of 0.1, 1, 10, 100 etc. A
- * tickInterval of 2 means a tick of 0.1, 10, 1000 etc. A tickInterval
- * of 0.2 puts a tick on 0.1, 0.2, 0.4, 0.6, 0.8, 1, 2, 4, 6, 8, 10, 20,
- * 40 etc.
- *
- *
- * If the tickInterval is too dense for labels to be drawn, Highcharts
- * may remove ticks.
- *
- * If the chart has multiple axes, the [alignTicks](#chart.alignTicks)
- * option may interfere with the `tickInterval` setting.
- *
- * @see [tickPixelInterval](#xAxis.tickPixelInterval)
- * @see [tickPositions](#xAxis.tickPositions)
- * @see [tickPositioner](#xAxis.tickPositioner)
- *
- * @sample {highcharts} highcharts/xaxis/tickinterval-5/
- * Tick interval of 5 on a linear axis
- * @sample {highstock} stock/xaxis/tickinterval/
- * Tick interval of 0.01 on Y axis
- *
- * @type {number}
- * @apioption xAxis.tickInterval
- */
- /**
- * The pixel length of the main tick marks.
- *
- * @sample {highcharts} highcharts/xaxis/ticklength/
- * 20 px tick length on the X axis
- * @sample {highstock} stock/xaxis/ticks/
- * Formatted ticks on X axis
- */
- tickLength: 10,
- /**
- * If tickInterval is `null` this option sets the approximate pixel
- * interval of the tick marks. Not applicable to categorized axis.
- *
- * The tick interval is also influenced by the [minTickInterval](
- * #xAxis.minTickInterval) option, that, by default prevents ticks from
- * being denser than the data points.
- *
- * @see [tickInterval](#xAxis.tickInterval)
- * @see [tickPositioner](#xAxis.tickPositioner)
- * @see [tickPositions](#xAxis.tickPositions)
- *
- * @sample {highcharts} highcharts/xaxis/tickpixelinterval-50/
- * 50 px on X axis
- * @sample {highstock} stock/xaxis/tickpixelinterval/
- * 200 px on X axis
- */
- tickPixelInterval: 100,
- /**
- * For categorized axes only. If `on` the tick mark is placed in the
- * center of the category, if `between` the tick mark is placed between
- * categories. The default is `between` if the `tickInterval` is 1, else
- * `on`.
- *
- * @sample {highcharts} highcharts/xaxis/tickmarkplacement-between/
- * "between" by default
- * @sample {highcharts} highcharts/xaxis/tickmarkplacement-on/
- * "on"
- *
- * @product highcharts gantt
- * @validvalue ["on", "between"]
- */
- tickmarkPlacement: 'between',
- /**
- * The position of the major tick marks relative to the axis line.
- * Can be one of `inside` and `outside`.
- *
- * @sample {highcharts} highcharts/xaxis/tickposition-outside/
- * "outside" by default
- * @sample {highcharts} highcharts/xaxis/tickposition-inside/
- * "inside"
- * @sample {highstock} stock/xaxis/ticks/
- * Formatted ticks on X axis
- *
- * @validvalue ["inside", "outside"]
- */
- tickPosition: 'outside',
- /**
- * A callback function returning array defining where the ticks are
- * laid out on the axis. This overrides the default behaviour of
- * [tickPixelInterval](#xAxis.tickPixelInterval) and [tickInterval](
- * #xAxis.tickInterval). The automatic tick positions are accessible
- * through `this.tickPositions` and can be modified by the callback.
- *
- * @see [tickPositions](#xAxis.tickPositions)
- *
- * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/
- * Demo of tickPositions and tickPositioner
- * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/
- * Demo of tickPositions and tickPositioner
- *
- * @type {Highcharts.AxisTickPositionerCallbackFunction}
- * @apioption xAxis.tickPositioner
- */
- /**
- * An array defining where the ticks are laid out on the axis. This
- * overrides the default behaviour of [tickPixelInterval](
- * #xAxis.tickPixelInterval) and [tickInterval](#xAxis.tickInterval).
- *
- * @see [tickPositioner](#xAxis.tickPositioner)
- *
- * @sample {highcharts} highcharts/xaxis/tickpositions-tickpositioner/
- * Demo of tickPositions and tickPositioner
- * @sample {highstock} highcharts/xaxis/tickpositions-tickpositioner/
- * Demo of tickPositions and tickPositioner
- *
- * @type {Array<number>}
- * @apioption xAxis.tickPositions
- */
- /**
- * The pixel width of the major tick marks. Defaults to 0 on category
- * axes, otherwise 1.
- *
- * In styled mode, the stroke width is given in the `.highcharts-tick`
- * class, but in order for the element to be generated on category axes,
- * the option must be explicitly set to 1.
- *
- * @sample {highcharts} highcharts/xaxis/tickwidth/
- * 10 px width
- * @sample {highcharts} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/ticks/
- * Formatted ticks on X axis
- * @sample {highstock} highcharts/css/axis-grid/
- * Styled mode
- *
- * @type {undefined|number}
- * @default {highstock} 1
- * @default {highmaps} 0
- * @apioption xAxis.tickWidth
- */
- /**
- * The axis title, showing next to the axis line.
- *
- * @productdesc {highmaps}
- * In Highmaps, the axis is hidden by default, but adding an axis title
- * is still possible. X axis and Y axis titles will appear at the bottom
- * and left by default.
- */
- title: {
- /**
- * Deprecated. Set the `text` to `null` to disable the title.
- *
- * @deprecated
- * @type {boolean}
- * @product highcharts
- * @apioption xAxis.title.enabled
- */
- /**
- * The pixel distance between the axis labels or line and the title.
- * Defaults to 0 for horizontal axes, 10 for vertical
- *
- * @sample {highcharts} highcharts/xaxis/title-margin/
- * Y axis title margin of 60
- *
- * @type {number}
- * @apioption xAxis.title.margin
- */
- /**
- * The distance of the axis title from the axis line. By default,
- * this distance is computed from the offset width of the labels,
- * the labels' distance from the axis and the title's margin.
- * However when the offset option is set, it overrides all this.
- *
- * @sample {highcharts} highcharts/yaxis/title-offset/
- * Place the axis title on top of the axis
- * @sample {highstock} highcharts/yaxis/title-offset/
- * Place the axis title on top of the Y axis
- *
- * @type {number}
- * @since 2.2.0
- * @apioption xAxis.title.offset
- */
- /**
- * Whether to reserve space for the title when laying out the axis.
- *
- * @type {boolean}
- * @default true
- * @since 5.0.11
- * @product highcharts highstock gantt
- * @apioption xAxis.title.reserveSpace
- */
- /**
- * The rotation of the text in degrees. 0 is horizontal, 270 is
- * vertical reading from bottom to top.
- *
- * @sample {highcharts} highcharts/yaxis/title-offset/
- * Horizontal
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.title.rotation
- */
- /**
- * The actual text of the axis title. It can contain basic HTML tags
- * like `b`, `i` and `span` with style.
- *
- * @sample {highcharts} highcharts/xaxis/title-text/
- * Custom HTML
- * @sample {highstock} stock/xaxis/title-text/
- * Titles for both axes
- *
- * @type {string|null}
- * @apioption xAxis.title.text
- */
- /**
- * Alignment of the text, can be `"left"`, `"right"` or `"center"`.
- * Default alignment depends on the
- * [title.align](xAxis.title.align):
- *
- * Horizontal axes:
- * - for `align` = `"low"`, `textAlign` is set to `left`
- * - for `align` = `"middle"`, `textAlign` is set to `center`
- * - for `align` = `"high"`, `textAlign` is set to `right`
- *
- * Vertical axes:
- * - for `align` = `"low"` and `opposite` = `true`, `textAlign` is
- * set to `right`
- * - for `align` = `"low"` and `opposite` = `false`, `textAlign` is
- * set to `left`
- * - for `align` = `"middle"`, `textAlign` is set to `center`
- * - for `align` = `"high"` and `opposite` = `true` `textAlign` is
- * set to `left`
- * - for `align` = `"high"` and `opposite` = `false` `textAlign` is
- * set to `right`
- *
- * @type {Highcharts.AlignValue}
- * @apioption xAxis.title.textAlign
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the axis title.
- *
- * @type {boolean}
- * @default false
- * @product highcharts highstock gantt
- * @apioption xAxis.title.useHTML
- */
- /**
- * Horizontal pixel offset of the title position.
- *
- * @type {number}
- * @default 0
- * @since 4.1.6
- * @product highcharts highstock gantt
- * @apioption xAxis.title.x
- */
- /**
- * Vertical pixel offset of the title position.
- *
- * @type {number}
- * @product highcharts highstock gantt
- * @apioption xAxis.title.y
- */
- /**
- * Alignment of the title relative to the axis values. Possible
- * values are "low", "middle" or "high".
- *
- * @sample {highcharts} highcharts/xaxis/title-align-low/
- * "low"
- * @sample {highcharts} highcharts/xaxis/title-align-center/
- * "middle" by default
- * @sample {highcharts} highcharts/xaxis/title-align-high/
- * "high"
- * @sample {highcharts} highcharts/yaxis/title-offset/
- * Place the Y axis title on top of the axis
- * @sample {highstock} stock/xaxis/title-align/
- * Aligned to "high" value
- *
- * @type {Highcharts.AxisTitleAlignValue}
- */
- align: 'middle',
- /**
- * CSS styles for the title. If the title text is longer than the
- * axis length, it will wrap to multiple lines by default. This can
- * be customized by setting `textOverflow: 'ellipsis'`, by
- * setting a specific `width` or by setting `whiteSpace: 'nowrap'`.
- *
- * In styled mode, the stroke width is given in the
- * `.highcharts-axis-title` class.
- *
- * @sample {highcharts} highcharts/xaxis/title-style/
- * Red
- * @sample {highcharts} highcharts/css/axis/
- * Styled mode
- *
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @internal */
- color: '#666666'
- }
- },
- /**
- * The type of axis. Can be one of `linear`, `logarithmic`, `datetime`
- * or `category`. In a datetime axis, the numbers are given in
- * milliseconds, and tick marks are placed on appropriate values like
- * full hours or days. In a category axis, the
- * [point names](#series.line.data.name) of the chart's series are used
- * for categories, if not a [categories](#xAxis.categories) array is
- * defined.
- *
- * @sample {highcharts} highcharts/xaxis/type-linear/
- * Linear
- * @sample {highcharts} highcharts/yaxis/type-log/
- * Logarithmic
- * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/
- * Logarithmic with minor grid lines
- * @sample {highcharts} highcharts/xaxis/type-log-both/
- * Logarithmic on two axes
- * @sample {highcharts} highcharts/yaxis/type-log-negative/
- * Logarithmic with extension to emulate negative values
- *
- * @type {Highcharts.AxisTypeValue}
- * @product highcharts gantt
- */
- type: 'linear',
- /**
- * If there are multiple axes on the same side of the chart, the pixel
- * margin between the axes. Defaults to 0 on vertical axes, 15 on
- * horizontal axes.
- *
- * @type {number}
- * @since 7.0.3
- * @apioption xAxis.margin
- */
- /**
- * Applies only when the axis `type` is `category`. When `uniqueNames`
- * is true, points are placed on the X axis according to their names.
- * If the same point name is repeated in the same or another series,
- * the point is placed on the same X position as other points of the
- * same name. When `uniqueNames` is false, the points are laid out in
- * increasing X positions regardless of their names, and the X axis
- * category will take the name of the last point in each position.
- *
- * @sample {highcharts} highcharts/xaxis/uniquenames-true/
- * True by default
- * @sample {highcharts} highcharts/xaxis/uniquenames-false/
- * False
- *
- * @type {boolean}
- * @default true
- * @since 4.2.7
- * @product highcharts gantt
- * @apioption xAxis.uniqueNames
- */
- /**
- * Datetime axis only. An array determining what time intervals the
- * ticks are allowed to fall on. Each array item is an array where the
- * first value is the time unit and the second value another array of
- * allowed multiples.
- *
- * Defaults to:
- * ```js
- * units: [[
- * 'millisecond', // unit name
- * [1, 2, 5, 10, 20, 25, 50, 100, 200, 500] // allowed multiples
- * ], [
- * 'second',
- * [1, 2, 5, 10, 15, 30]
- * ], [
- * 'minute',
- * [1, 2, 5, 10, 15, 30]
- * ], [
- * 'hour',
- * [1, 2, 3, 4, 6, 8, 12]
- * ], [
- * 'day',
- * [1]
- * ], [
- * 'week',
- * [1]
- * ], [
- * 'month',
- * [1, 3, 6]
- * ], [
- * 'year',
- * null
- * ]]
- * ```
- *
- * @type {Array<Array<string,(Array<number>|null)>>}
- * @product highcharts highstock gantt
- * @apioption xAxis.units
- */
- /**
- * Whether axis, including axis title, line, ticks and labels, should
- * be visible.
- *
- * @type {boolean}
- * @default true
- * @since 4.1.9
- * @product highcharts highstock gantt
- * @apioption xAxis.visible
- */
- /**
- * Color of the minor, secondary grid lines.
- *
- * In styled mode, the stroke width is given in the
- * `.highcharts-minor-grid-line` class.
- *
- * @sample {highcharts} highcharts/yaxis/minorgridlinecolor/
- * Bright grey lines from Y axis
- * @sample {highcharts|highstock} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/minorgridlinecolor/
- * Bright grey lines from Y axis
- *
- * @type {Highcharts.ColorType}
- * @default #f2f2f2
- */
- minorGridLineColor: '#f2f2f2',
- /**
- * Width of the minor, secondary grid lines.
- *
- * In styled mode, the stroke width is given in the
- * `.highcharts-grid-line` class.
- *
- * @sample {highcharts} highcharts/yaxis/minorgridlinewidth/
- * 2px lines from Y axis
- * @sample {highcharts|highstock} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/minorgridlinewidth/
- * 2px lines from Y axis
- */
- minorGridLineWidth: 1,
- /**
- * Color for the minor tick marks.
- *
- * @sample {highcharts} highcharts/yaxis/minortickcolor/
- * Black tick marks on Y axis
- * @sample {highstock} stock/xaxis/minorticks/
- * Black tick marks on Y axis
- *
- * @type {Highcharts.ColorType}
- * @default #999999
- */
- minorTickColor: '#999999',
- /**
- * The color of the line marking the axis itself.
- *
- * In styled mode, the line stroke is given in the
- * `.highcharts-axis-line` or `.highcharts-xaxis-line` class.
- *
- * @productdesc {highmaps}
- * In Highmaps, the axis line is hidden by default, because the axis is
- * not visible by default.
- *
- * @sample {highcharts} highcharts/yaxis/linecolor/
- * A red line on Y axis
- * @sample {highcharts|highstock} highcharts/css/axis/
- * Axes in styled mode
- * @sample {highstock} stock/xaxis/linecolor/
- * A red line on X axis
- *
- * @type {Highcharts.ColorType}
- * @default #ccd6eb
- */
- lineColor: '#ccd6eb',
- /**
- * The width of the line marking the axis itself.
- *
- * In styled mode, the stroke width is given in the
- * `.highcharts-axis-line` or `.highcharts-xaxis-line` class.
- *
- * @sample {highcharts} highcharts/yaxis/linecolor/
- * A 1px line on Y axis
- * @sample {highcharts|highstock} highcharts/css/axis/
- * Axes in styled mode
- * @sample {highstock} stock/xaxis/linewidth/
- * A 2px line on X axis
- *
- * @default {highcharts|highstock} 1
- * @default {highmaps} 0
- */
- lineWidth: 1,
- /**
- * Color of the grid lines extending the ticks across the plot area.
- *
- * In styled mode, the stroke is given in the `.highcharts-grid-line`
- * class.
- *
- * @productdesc {highmaps}
- * In Highmaps, the grid lines are hidden by default.
- *
- * @sample {highcharts} highcharts/yaxis/gridlinecolor/
- * Green lines
- * @sample {highcharts|highstock} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/gridlinecolor/
- * Green lines
- *
- * @type {Highcharts.ColorType}
- * @default #e6e6e6
- */
- gridLineColor: '#e6e6e6',
- // gridLineDashStyle: 'solid',
- /**
- * The width of the grid lines extending the ticks across the plot area.
- *
- * In styled mode, the stroke width is given in the
- * `.highcharts-grid-line` class.
- *
- * @sample {highcharts} highcharts/yaxis/gridlinewidth/
- * 2px lines
- * @sample {highcharts|highstock} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/gridlinewidth/
- * 2px lines
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.gridLineWidth
- */
- // gridLineWidth: 0,
- /**
- * The height as the vertical axis. If it's a number, it is
- * interpreted as pixels.
- *
- * Since Highcharts 2: If it's a percentage string, it is interpreted
- * as percentages of the total plot height.
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption xAxis.height
- */
- /**
- * The width as the horizontal axis. If it's a number, it is interpreted
- * as pixels.
- *
- * Since Highcharts v5.0.13: If it's a percentage string, it is
- * interpreted as percentages of the total plot width.
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption xAxis.width
- */
- /**
- * Color for the main tick marks.
- *
- * In styled mode, the stroke is given in the `.highcharts-tick`
- * class.
- *
- * @sample {highcharts} highcharts/xaxis/tickcolor/
- * Red ticks on X axis
- * @sample {highcharts|highstock} highcharts/css/axis-grid/
- * Styled mode
- * @sample {highstock} stock/xaxis/ticks/
- * Formatted ticks on X axis
- *
- * @type {Highcharts.ColorType}
- * @default #ccd6eb
- */
- tickColor: '#ccd6eb'
- // tickWidth: 1
- };
- /**
- * The Y axis or value axis. Normally this is the vertical axis,
- * though if the chart is inverted this is the horizontal axis.
- * In case of multiple axes, the yAxis node is an array of
- * configuration objects.
- *
- * See [the Axis object](/class-reference/Highcharts.Axis) for programmatic
- * access to the axis.
- *
- * @type {*|Array<*>}
- * @extends xAxis
- * @excluding currentDateIndicator,ordinal,overscroll
- * @optionparent yAxis
- *
- * @private
- */
- Axis.defaultYAxisOptions = {
- /**
- * The type of axis. Can be one of `linear`, `logarithmic`, `datetime`,
- * `category` or `treegrid`. Defaults to `treegrid` for Gantt charts,
- * `linear` for other chart types.
- *
- * In a datetime axis, the numbers are given in milliseconds, and tick
- * marks are placed on appropriate values, like full hours or days. In a
- * category or treegrid axis, the [point names](#series.line.data.name)
- * of the chart's series are used for categories, if a
- * [categories](#xAxis.categories) array is not defined.
- *
- * @sample {highcharts} highcharts/yaxis/type-log-minorgrid/
- * Logarithmic with minor grid lines
- * @sample {highcharts} highcharts/yaxis/type-log-negative/
- * Logarithmic with extension to emulate negative values
- * @sample {gantt} gantt/treegrid-axis/demo
- * Treegrid axis
- *
- * @type {Highcharts.AxisTypeValue}
- * @default {highcharts} linear
- * @default {gantt} treegrid
- * @product highcharts gantt
- * @apioption yAxis.type
- */
- /**
- * The height of the Y axis. If it's a number, it is interpreted as
- * pixels.
- *
- * Since Highcharts 2: If it's a percentage string, it is interpreted as
- * percentages of the total plot height.
- *
- * @see [yAxis.top](#yAxis.top)
- *
- * @sample {highstock} stock/demo/candlestick-and-volume/
- * Percentage height panes
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption yAxis.height
- */
- /**
- * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color
- * to represent the maximum value of the Y axis.
- *
- * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/
- * Min and max colors
- *
- * @type {Highcharts.ColorType}
- * @default #003399
- * @since 4.0
- * @product highcharts
- * @apioption yAxis.maxColor
- */
- /**
- * Solid gauge only. Unless [stops](#yAxis.stops) are set, the color
- * to represent the minimum value of the Y axis.
- *
- * @sample {highcharts} highcharts/yaxis/mincolor-maxcolor/
- * Min and max color
- *
- * @type {Highcharts.ColorType}
- * @default #e6ebf5
- * @since 4.0
- * @product highcharts
- * @apioption yAxis.minColor
- */
- /**
- * Whether to reverse the axis so that the highest number is closest
- * to the origin.
- *
- * @sample {highcharts} highcharts/yaxis/reversed/
- * Reversed Y axis
- * @sample {highstock} stock/xaxis/reversed/
- * Reversed Y axis
- *
- * @type {boolean}
- * @default {highcharts} false
- * @default {highstock} false
- * @default {highmaps} true
- * @default {gantt} true
- * @apioption yAxis.reversed
- */
- /**
- * If `true`, the first series in a stack will be drawn on top in a
- * positive, non-reversed Y axis. If `false`, the first series is in
- * the base of the stack.
- *
- * @sample {highcharts} highcharts/yaxis/reversedstacks-false/
- * Non-reversed stacks
- * @sample {highstock} highcharts/yaxis/reversedstacks-false/
- * Non-reversed stacks
- *
- * @type {boolean}
- * @default true
- * @since 3.0.10
- * @product highcharts highstock
- * @apioption yAxis.reversedStacks
- */
- /**
- * Solid gauge series only. Color stops for the solid gauge. Use this
- * in cases where a linear gradient between a `minColor` and `maxColor`
- * is not sufficient. The stops is an array of tuples, where the first
- * item is a float between 0 and 1 assigning the relative position in
- * the gradient, and the second item is the color.
- *
- * For solid gauges, the Y axis also inherits the concept of
- * [data classes](https://api.highcharts.com/highmaps#colorAxis.dataClasses)
- * from the Highmaps color axis.
- *
- * @see [minColor](#yAxis.minColor)
- * @see [maxColor](#yAxis.maxColor)
- *
- * @sample {highcharts} highcharts/demo/gauge-solid/
- * True by default
- *
- * @type {Array<Array<number,Highcharts.ColorType>>}
- * @since 4.0
- * @product highcharts
- * @apioption yAxis.stops
- */
- /**
- * The pixel width of the major tick marks.
- *
- * @sample {highcharts} highcharts/xaxis/tickwidth/ 10 px width
- * @sample {highstock} stock/xaxis/ticks/ Formatted ticks on X axis
- *
- * @type {number}
- * @default 0
- * @product highcharts highstock gantt
- * @apioption yAxis.tickWidth
- */
- /**
- * Whether to force the axis to end on a tick. Use this option with
- * the `maxPadding` option to control the axis end.
- *
- * This option is always disabled, when panning type is
- * either `y` or `xy`.
- *
- * @see [type](#chart.panning.type)
- *
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * True by default
- * @sample {highcharts} highcharts/yaxis/endontick/
- * False
- * @sample {highstock} stock/demo/basic-line/
- * True by default
- * @sample {highstock} stock/xaxis/endontick/
- * False for Y axis
- *
- * @since 1.2.0
- */
- endOnTick: true,
- /**
- * Padding of the max value relative to the length of the axis. A
- * padding of 0.05 will make a 100px axis 5px longer. This is useful
- * when you don't want the highest data value to appear on the edge
- * of the plot area. When the axis' `max` option is set or a max extreme
- * is set using `axis.setExtremes()`, the maxPadding will be ignored.
- *
- * Also the `softThreshold` option takes precedence over `maxPadding`,
- * so if the data is tangent to the threshold, `maxPadding` may not
- * apply unless `softThreshold` is set to false.
- *
- * @sample {highcharts} highcharts/yaxis/maxpadding-02/
- * Max padding of 0.2
- * @sample {highstock} stock/xaxis/minpadding-maxpadding/
- * Greater min- and maxPadding
- *
- * @since 1.2.0
- * @product highcharts highstock gantt
- */
- maxPadding: 0.05,
- /**
- * Padding of the min value relative to the length of the axis. A
- * padding of 0.05 will make a 100px axis 5px longer. This is useful
- * when you don't want the lowest data value to appear on the edge
- * of the plot area. When the axis' `min` option is set or a max extreme
- * is set using `axis.setExtremes()`, the maxPadding will be ignored.
- *
- * Also the `softThreshold` option takes precedence over `minPadding`,
- * so if the data is tangent to the threshold, `minPadding` may not
- * apply unless `softThreshold` is set to false.
- *
- * @sample {highcharts} highcharts/yaxis/minpadding/
- * Min padding of 0.2
- * @sample {highstock} stock/xaxis/minpadding-maxpadding/
- * Greater min- and maxPadding
- *
- * @since 1.2.0
- * @product highcharts highstock gantt
- */
- minPadding: 0.05,
- /**
- * @productdesc {highstock}
- * In Highstock 1.x, the Y axis was placed on the left side by default.
- *
- * @sample {highcharts} highcharts/yaxis/opposite/
- * Secondary Y axis opposite
- * @sample {highstock} stock/xaxis/opposite/
- * Y axis on left side
- *
- * @type {boolean}
- * @default {highstock} true
- * @default {highcharts} false
- * @product highstock highcharts gantt
- * @apioption yAxis.opposite
- */
- /**
- * @see [tickInterval](#xAxis.tickInterval)
- * @see [tickPositioner](#xAxis.tickPositioner)
- * @see [tickPositions](#xAxis.tickPositions)
- */
- tickPixelInterval: 72,
- showLastLabel: true,
- /**
- * @extends xAxis.labels
- */
- labels: {
- /**
- * Angular gauges and solid gauges only.
- * The label's pixel distance from the perimeter of the plot area.
- *
- * Since v7.1.2: If it's a percentage string, it is interpreted the
- * same as [series.radius](#plotOptions.gauge.radius), so label can be
- * aligned under the gauge's shape.
- *
- * @sample {highcharts} highcharts/yaxis/labels-distance/
- * Labels centered under the arc
- *
- * @type {number|string}
- * @default -25
- * @product highcharts
- * @apioption yAxis.labels.distance
- */
- /**
- * The y position offset of all labels relative to the tick
- * positions on the axis. For polar and radial axis consider the use
- * of the [distance](#yAxis.labels.distance) option.
- *
- * @sample {highcharts} highcharts/xaxis/labels-x/
- * Y axis labels placed on grid lines
- *
- * @type {number}
- * @default {highcharts} 3
- * @default {highstock} -2
- * @default {highmaps} 3
- * @apioption yAxis.labels.y
- */
- /**
- * What part of the string the given position is anchored to. Can
- * be one of `"left"`, `"center"` or `"right"`. The exact position
- * also depends on the `labels.x` setting.
- *
- * Angular gauges and solid gauges defaults to `"center"`.
- * Solid gauges with two labels have additional option `"auto"`
- * for automatic horizontal and vertical alignment.
- *
- * @see [yAxis.labels.distance](#yAxis.labels.distance)
- *
- * @sample {highcharts} highcharts/yaxis/labels-align-left/
- * Left
- * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/
- * Solid gauge labels auto aligned
- *
- * @type {Highcharts.AlignValue}
- * @default {highcharts|highmaps} right
- * @default {highstock} left
- * @apioption yAxis.labels.align
- */
- /**
- * The x position offset of all labels relative to the tick
- * positions on the axis. Defaults to -15 for left axis, 15 for
- * right axis.
- *
- * @sample {highcharts} highcharts/xaxis/labels-x/
- * Y axis labels placed on grid lines
- */
- x: -8
- },
- /**
- * @productdesc {highmaps}
- * In Highmaps, the axis line is hidden by default, because the axis is
- * not visible by default.
- *
- * @type {Highcharts.ColorType}
- * @apioption yAxis.lineColor
- */
- /**
- * @sample {highcharts} highcharts/yaxis/max-200/
- * Y axis max of 200
- * @sample {highcharts} highcharts/yaxis/max-logarithmic/
- * Y axis max on logarithmic axis
- * @sample {highstock} stock/yaxis/min-max/
- * Fixed min and max on Y axis
- * @sample {highmaps} maps/axis/min-max/
- * Pre-zoomed to a specific area
- *
- * @apioption yAxis.max
- */
- /**
- * @sample {highcharts} highcharts/yaxis/min-startontick-false/
- * -50 with startOnTick to false
- * @sample {highcharts} highcharts/yaxis/min-startontick-true/
- * -50 with startOnTick true by default
- * @sample {highstock} stock/yaxis/min-max/
- * Fixed min and max on Y axis
- * @sample {highmaps} maps/axis/min-max/
- * Pre-zoomed to a specific area
- *
- * @apioption yAxis.min
- */
- /**
- * An optional scrollbar to display on the Y axis in response to
- * limiting the minimum an maximum of the axis values.
- *
- * In styled mode, all the presentational options for the scrollbar
- * are replaced by the classes `.highcharts-scrollbar-thumb`,
- * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`,
- * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`.
- *
- * @sample {highstock} stock/yaxis/scrollbar/
- * Scrollbar on the Y axis
- *
- * @extends scrollbar
- * @since 4.2.6
- * @product highstock
- * @excluding height
- * @apioption yAxis.scrollbar
- */
- /**
- * Enable the scrollbar on the Y axis.
- *
- * @sample {highstock} stock/yaxis/scrollbar/
- * Enabled on Y axis
- *
- * @type {boolean}
- * @default false
- * @since 4.2.6
- * @product highstock
- * @apioption yAxis.scrollbar.enabled
- */
- /**
- * Pixel margin between the scrollbar and the axis elements.
- *
- * @type {number}
- * @default 10
- * @since 4.2.6
- * @product highstock
- * @apioption yAxis.scrollbar.margin
- */
- /**
- * Whether to show the scrollbar when it is fully zoomed out at max
- * range. Setting it to `false` on the Y axis makes the scrollbar stay
- * hidden until the user zooms in, like common in browsers.
- *
- * @type {boolean}
- * @default true
- * @since 4.2.6
- * @product highstock
- * @apioption yAxis.scrollbar.showFull
- */
- /**
- * The width of a vertical scrollbar or height of a horizontal
- * scrollbar. Defaults to 20 on touch devices.
- *
- * @type {number}
- * @default 14
- * @since 4.2.6
- * @product highstock
- * @apioption yAxis.scrollbar.size
- */
- /**
- * Z index of the scrollbar elements.
- *
- * @type {number}
- * @default 3
- * @since 4.2.6
- * @product highstock
- * @apioption yAxis.scrollbar.zIndex
- */
- /**
- * A soft maximum for the axis. If the series data maximum is less
- * than this, the axis will stay at this maximum, but if the series
- * data maximum is higher, the axis will flex to show all data.
- *
- * **Note**: The [series.softThreshold](
- * #plotOptions.series.softThreshold) option takes precedence over this
- * option.
- *
- * @sample highcharts/yaxis/softmin-softmax/
- * Soft min and max
- *
- * @type {number}
- * @since 5.0.1
- * @product highcharts highstock gantt
- * @apioption yAxis.softMax
- */
- /**
- * A soft minimum for the axis. If the series data minimum is greater
- * than this, the axis will stay at this minimum, but if the series
- * data minimum is lower, the axis will flex to show all data.
- *
- * **Note**: The [series.softThreshold](
- * #plotOptions.series.softThreshold) option takes precedence over this
- * option.
- *
- * @sample highcharts/yaxis/softmin-softmax/
- * Soft min and max
- *
- * @type {number}
- * @since 5.0.1
- * @product highcharts highstock gantt
- * @apioption yAxis.softMin
- */
- /**
- * Defines the horizontal alignment of the stack total label. Can be one
- * of `"left"`, `"center"` or `"right"`. The default value is calculated
- * at runtime and depends on orientation and whether the stack is
- * positive or negative.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-align-left/
- * Aligned to the left
- * @sample {highcharts} highcharts/yaxis/stacklabels-align-center/
- * Aligned in center
- * @sample {highcharts} highcharts/yaxis/stacklabels-align-right/
- * Aligned to the right
- *
- * @type {Highcharts.AlignValue}
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.align
- */
- /**
- * A format string for the data label. Available variables are the same
- * as for `formatter`.
- *
- * @type {string}
- * @default {total}
- * @since 3.0.2
- * @product highcharts highstock
- * @apioption yAxis.stackLabels.format
- */
- /**
- * Rotation of the labels in degrees.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-rotation/
- * Labels rotated 45°
- *
- * @type {number}
- * @default 0
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.rotation
- */
- /**
- * The text alignment for the label. While `align` determines where the
- * texts anchor point is placed with regards to the stack, `textAlign`
- * determines how the text is aligned against its anchor point. Possible
- * values are `"left"`, `"center"` and `"right"`. The default value is
- * calculated at runtime and depends on orientation and whether the
- * stack is positive or negative.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-textalign-left/
- * Label in center position but text-aligned left
- *
- * @type {Highcharts.AlignValue}
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.textAlign
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the labels.
- *
- * @type {boolean}
- * @default false
- * @since 3.0
- * @product highcharts highstock
- * @apioption yAxis.stackLabels.useHTML
- */
- /**
- * Defines the vertical alignment of the stack total label. Can be one
- * of `"top"`, `"middle"` or `"bottom"`. The default value is calculated
- * at runtime and depends on orientation and whether the stack is
- * positive or negative.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-top/
- * Vertically aligned top
- * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-middle/
- * Vertically aligned middle
- * @sample {highcharts} highcharts/yaxis/stacklabels-verticalalign-bottom/
- * Vertically aligned bottom
- *
- * @type {Highcharts.VerticalAlignValue}
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.verticalAlign
- */
- /**
- * The x position offset of the label relative to the left of the
- * stacked bar. The default value is calculated at runtime and depends
- * on orientation and whether the stack is positive or negative.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-x/
- * Stack total labels with x offset
- *
- * @type {number}
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.x
- */
- /**
- * The y position offset of the label relative to the tick position
- * on the axis. The default value is calculated at runtime and depends
- * on orientation and whether the stack is positive or negative.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-y/
- * Stack total labels with y offset
- *
- * @type {number}
- * @since 2.1.5
- * @product highcharts
- * @apioption yAxis.stackLabels.y
- */
- /**
- * Whether to force the axis to start on a tick. Use this option with
- * the `maxPadding` option to control the axis start.
- *
- * This option is always disabled, when panning type is
- * either `y` or `xy`.
- *
- * @see [type](#chart.panning.type)
- *
- * @sample {highcharts} highcharts/xaxis/startontick-false/
- * False by default
- * @sample {highcharts} highcharts/xaxis/startontick-true/
- * True
- * @sample {highstock} stock/xaxis/endontick/
- * False for Y axis
- *
- * @since 1.2.0
- * @product highcharts highstock gantt
- */
- startOnTick: true,
- title: {
- /**
- * The pixel distance between the axis labels and the title.
- * Positive values are outside the axis line, negative are inside.
- *
- * @sample {highcharts} highcharts/xaxis/title-margin/
- * Y axis title margin of 60
- *
- * @type {number}
- * @default 40
- * @apioption yAxis.title.margin
- */
- /**
- * The rotation of the text in degrees. 0 is horizontal, 270 is
- * vertical reading from bottom to top.
- *
- * @sample {highcharts} highcharts/yaxis/title-offset/
- * Horizontal
- */
- rotation: 270,
- /**
- * The actual text of the axis title. Horizontal texts can contain
- * HTML, but rotated texts are painted using vector techniques and
- * must be clean text. The Y axis title is disabled by setting the
- * `text` option to `undefined`.
- *
- * @sample {highcharts} highcharts/xaxis/title-text/
- * Custom HTML
- *
- * @type {string|null}
- * @default {highcharts} Values
- * @default {highstock} undefined
- * @product highcharts highstock gantt
- */
- text: 'Values'
- },
- /**
- * The top position of the Y axis. If it's a number, it is interpreted
- * as pixel position relative to the chart.
- *
- * Since Highcharts 2: If it's a percentage string, it is interpreted as
- * percentages of the plot height, offset from plot area top.
- *
- * @see [yAxis.height](#yAxis.height)
- *
- * @sample {highstock} stock/demo/candlestick-and-volume/
- * Percentage height panes
- *
- * @type {number|string}
- * @product highcharts highstock
- * @apioption yAxis.top
- */
- /**
- * The stack labels show the total value for each bar in a stacked
- * column or bar chart. The label will be placed on top of positive
- * columns and below negative columns. In case of an inverted column
- * chart or a bar chart the label is placed to the right of positive
- * bars and to the left of negative bars.
- *
- * @product highcharts
- */
- stackLabels: {
- /**
- * Enable or disable the initial animation when a series is
- * displayed for the `stackLabels`. The animation can also be set as
- * a configuration object. Please note that this option only
- * applies to the initial animation.
- * For other animations, see [chart.animation](#chart.animation)
- * and the animation parameter under the API methods.
- * The following properties are supported:
- *
- * - `defer`: The animation delay time in milliseconds.
- *
- * @sample {highcharts} highcharts/plotoptions/animation-defer/
- * Animation defer settings
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @since 8.2.0
- * @apioption yAxis.stackLabels.animation
- */
- animation: {},
- /**
- * The animation delay time in milliseconds.
- * Set to `0` renders stackLabel immediately.
- * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer).
- *
- * @type {number}
- * @since 8.2.0
- * @apioption yAxis.stackLabels.animation.defer
- */
- /**
- * Allow the stack labels to overlap.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-allowoverlap-false/
- * Default false
- *
- * @since 5.0.13
- * @product highcharts
- */
- allowOverlap: false,
- /**
- * The background color or gradient for the stack label.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-box/
- * Stack labels box options
- * @type {Highcharts.ColorType}
- * @since 8.1.0
- * @apioption yAxis.stackLabels.backgroundColor
- */
- /**
- * The border color for the stack label. Defaults to `undefined`.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-box/
- * Stack labels box options
- * @type {Highcharts.ColorType}
- * @since 8.1.0
- * @apioption yAxis.stackLabels.borderColor
- */
- /**
- * The border radius in pixels for the stack label.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-box/
- * Stack labels box options
- * @type {number}
- * @default 0
- * @since 8.1.0
- * @apioption yAxis.stackLabels.borderRadius
- */
- /**
- * The border width in pixels for the stack label.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-box/
- * Stack labels box options
- * @type {number}
- * @default 0
- * @since 8.1.0
- * @apioption yAxis.stackLabels.borderWidth
- */
- /**
- * Enable or disable the stack total labels.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-enabled/
- * Enabled stack total labels
- * @sample {highcharts} highcharts/yaxis/stacklabels-enabled-waterfall/
- * Enabled stack labels in waterfall chart
- *
- * @since 2.1.5
- * @product highcharts
- */
- enabled: false,
- /**
- * Whether to hide stack labels that are outside the plot area.
- * By default, the stack label is moved
- * inside the plot area according to the
- * [overflow](/highcharts/#yAxis/stackLabels/overflow)
- * option.
- *
- * @type {boolean}
- * @since 7.1.3
- */
- crop: true,
- /**
- * How to handle stack total labels that flow outside the plot area.
- * The default is set to `"justify"`,
- * which aligns them inside the plot area.
- * For columns and bars, this means it will be moved inside the bar.
- * To display stack labels outside the plot area,
- * set `crop` to `false` and `overflow` to `"allow"`.
- *
- * @sample highcharts/yaxis/stacklabels-overflow/
- * Stack labels flows outside the plot area.
- *
- * @type {Highcharts.DataLabelsOverflowValue}
- * @since 7.1.3
- */
- overflow: 'justify',
- /* eslint-disable valid-jsdoc */
- /**
- * Callback JavaScript function to format the label. The value is
- * given by `this.total`.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-formatter/
- * Added units to stack total value
- *
- * @type {Highcharts.FormatterCallbackFunction<Highcharts.StackItemObject>}
- * @since 2.1.5
- * @product highcharts
- */
- formatter: function () {
- var numberFormatter = this.axis.chart.numberFormatter;
- /* eslint-enable valid-jsdoc */
- return numberFormatter(this.total, -1);
- },
- /**
- * CSS styles for the label.
- *
- * In styled mode, the styles are set in the
- * `.highcharts-stack-label` class.
- *
- * @sample {highcharts} highcharts/yaxis/stacklabels-style/
- * Red stack total labels
- *
- * @type {Highcharts.CSSObject}
- * @since 2.1.5
- * @product highcharts
- */
- style: {
- /** @internal */
- color: '#000000',
- /** @internal */
- fontSize: '11px',
- /** @internal */
- fontWeight: 'bold',
- /** @internal */
- textOutline: '1px contrast'
- }
- },
- gridLineWidth: 1,
- lineWidth: 0
- // tickWidth: 0
- };
- /**
- * The Z axis or depth axis for 3D plots.
- *
- * See the [Axis class](/class-reference/Highcharts.Axis) for programmatic
- * access to the axis.
- *
- * @sample {highcharts} highcharts/3d/scatter-zaxis-categories/
- * Z-Axis with Categories
- * @sample {highcharts} highcharts/3d/scatter-zaxis-grid/
- * Z-Axis with styling
- *
- * @type {*|Array<*>}
- * @extends xAxis
- * @since 5.0.0
- * @product highcharts
- * @excluding breaks, crosshair, height, left, lineColor, lineWidth,
- * nameToX, showEmpty, top, width
- * @apioption zAxis
- *
- * @private
- */
- // This variable extends the defaultOptions for left axes.
- Axis.defaultLeftAxisOptions = {
- labels: {
- x: -15
- },
- title: {
- rotation: 270
- }
- };
- // This variable extends the defaultOptions for right axes.
- Axis.defaultRightAxisOptions = {
- labels: {
- x: 15
- },
- title: {
- rotation: 90
- }
- };
- // This variable extends the defaultOptions for bottom axes.
- Axis.defaultBottomAxisOptions = {
- labels: {
- autoRotation: [-45],
- x: 0
- // overflow: undefined,
- // staggerLines: null
- },
- margin: 15,
- title: {
- rotation: 0
- }
- };
- // This variable extends the defaultOptions for top axes.
- Axis.defaultTopAxisOptions = {
- labels: {
- autoRotation: [-45],
- x: 0
- // overflow: undefined
- // staggerLines: null
- },
- margin: 15,
- title: {
- rotation: 0
- }
- };
- // Properties to survive after destroy, needed for Axis.update (#4317,
- // #5773, #5881).
- Axis.keepProps = ['extKey', 'hcEvents', 'names', 'series', 'userMax', 'userMin'];
- return Axis;
- }());
- H.Axis = Axis;
- return H.Axis;
- });
- _registerModule(_modules, 'Core/Axis/DateTimeAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Utilities.js']], function (Axis, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- getMagnitude = U.getMagnitude,
- normalizeTickInterval = U.normalizeTickInterval,
- timeUnits = U.timeUnits;
- /* eslint-disable valid-jsdoc */
- var DateTimeAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function DateTimeAxisAdditions(axis) {
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Get a normalized tick interval for dates. Returns a configuration object
- * with unit range (interval), count and name. Used to prepare data for
- * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as
- * `getTimeTicks` now runs of segments in stock charts, the normalizing
- * logic was extracted in order to prevent it for running over again for
- * each segment having the same interval. #662, #697.
- * @private
- */
- /**
- * Get a normalized tick interval for dates. Returns a configuration object
- * with unit range (interval), count and name. Used to prepare data for
- * `getTimeTicks`. Previously this logic was part of getTimeTicks, but as
- * `getTimeTicks` now runs of segments in stock charts, the normalizing
- * logic was extracted in order to prevent it for running over again for
- * each segment having the same interval. #662, #697.
- * @private
- */
- DateTimeAxisAdditions.prototype.normalizeTimeTickInterval = function (tickInterval, unitsOption) {
- var units = unitsOption || [[
- 'millisecond',
- [1, 2, 5, 10, 20, 25, 50, 100, 200, 500] // allowed multiples
- ],
- [
- 'second',
- [1, 2, 5, 10, 15, 30]
- ],
- [
- 'minute',
- [1, 2, 5, 10, 15, 30]
- ],
- [
- 'hour',
- [1, 2, 3, 4, 6, 8, 12]
- ],
- [
- 'day',
- [1, 2]
- ],
- [
- 'week',
- [1, 2]
- ],
- [
- 'month',
- [1, 2, 3, 4, 6]
- ],
- [
- 'year',
- null
- ]],
- unit = units[units.length - 1], // default unit is years
- interval = timeUnits[unit[0]],
- multiples = unit[1],
- count,
- i;
- // loop through the units to find the one that best fits the
- // tickInterval
- for (i = 0; i < units.length; i++) {
- unit = units[i];
- interval = timeUnits[unit[0]];
- multiples = unit[1];
- if (units[i + 1]) {
- // lessThan is in the middle between the highest multiple and
- // the next unit.
- var lessThan = (interval *
- multiples[multiples.length - 1] +
- timeUnits[units[i + 1][0]]) / 2;
- // break and keep the current unit
- if (tickInterval <= lessThan) {
- break;
- }
- }
- }
- // prevent 2.5 years intervals, though 25, 250 etc. are allowed
- if (interval === timeUnits.year && tickInterval < 5 * interval) {
- multiples = [1, 2, 5];
- }
- // get the count
- count = normalizeTickInterval(tickInterval / interval, multiples, unit[0] === 'year' ? // #1913, #2360
- Math.max(getMagnitude(tickInterval / interval), 1) :
- 1);
- return {
- unitRange: interval,
- count: count,
- unitName: unit[0]
- };
- };
- return DateTimeAxisAdditions;
- }());
- /**
- * Date and time support for axes.
- *
- * @private
- * @class
- */
- var DateTimeAxis = /** @class */ (function () {
- function DateTimeAxis() {
- }
- /* *
- *
- * Static Functions
- *
- * */
- /**
- * Extends axis class with date and time support.
- * @private
- */
- DateTimeAxis.compose = function (AxisClass) {
- AxisClass.keepProps.push('dateTime');
- var axisProto = AxisClass.prototype;
- /**
- * Set the tick positions to a time unit that makes sense, for example
- * on the first of each month or on every Monday. Return an array with
- * the time positions. Used in datetime axes as well as for grouping
- * data on a datetime axis.
- *
- * @private
- * @function Highcharts.Axis#getTimeTicks
- *
- * @param {Highcharts.TimeNormalizeObject} normalizedInterval
- * The interval in axis values (ms) and thecount.
- *
- * @param {number} min
- * The minimum in axis values.
- *
- * @param {number} max
- * The maximum in axis values.
- *
- * @param {number} startOfWeek
- *
- * @return {Highcharts.AxisTickPositionsArray}
- */
- axisProto.getTimeTicks = function () {
- return this.chart.time.getTimeTicks.apply(this.chart.time, arguments);
- };
- /* eslint-disable no-invalid-this */
- addEvent(AxisClass, 'init', function (e) {
- var axis = this;
- var options = e.userOptions;
- if (options.type !== 'datetime') {
- axis.dateTime = void 0;
- return;
- }
- if (!axis.dateTime) {
- axis.dateTime = new DateTimeAxisAdditions(axis);
- }
- });
- /* eslint-enable no-invalid-this */
- };
- /* *
- *
- * Static Properties
- *
- * */
- DateTimeAxis.AdditionsClass = DateTimeAxisAdditions;
- return DateTimeAxis;
- }());
- DateTimeAxis.compose(Axis);
- return DateTimeAxis;
- });
- _registerModule(_modules, 'Core/Axis/LogarithmicAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Utilities.js']], function (Axis, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- getMagnitude = U.getMagnitude,
- normalizeTickInterval = U.normalizeTickInterval,
- pick = U.pick;
- /* eslint-disable valid-jsdoc */
- /**
- * Provides logarithmic support for axes.
- *
- * @private
- * @class
- */
- var LogarithmicAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function LogarithmicAxisAdditions(axis) {
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Set the tick positions of a logarithmic axis.
- */
- LogarithmicAxisAdditions.prototype.getLogTickPositions = function (interval, min, max, minor) {
- var log = this;
- var axis = log.axis;
- var axisLength = axis.len;
- var options = axis.options;
- // Since we use this method for both major and minor ticks,
- // use a local variable and return the result
- var positions = [];
- // Reset
- if (!minor) {
- log.minorAutoInterval = void 0;
- }
- // First case: All ticks fall on whole logarithms: 1, 10, 100 etc.
- if (interval >= 0.5) {
- interval = Math.round(interval);
- positions = axis.getLinearTickPositions(interval, min, max);
- // Second case: We need intermediary ticks. For example
- // 1, 2, 4, 6, 8, 10, 20, 40 etc.
- }
- else if (interval >= 0.08) {
- var roundedMin = Math.floor(min),
- intermediate,
- i,
- j,
- len,
- pos,
- lastPos,
- break2;
- if (interval > 0.3) {
- intermediate = [1, 2, 4];
- // 0.2 equals five minor ticks per 1, 10, 100 etc
- }
- else if (interval > 0.15) {
- intermediate = [1, 2, 4, 6, 8];
- }
- else { // 0.1 equals ten minor ticks per 1, 10, 100 etc
- intermediate = [1, 2, 3, 4, 5, 6, 7, 8, 9];
- }
- for (i = roundedMin; i < max + 1 && !break2; i++) {
- len = intermediate.length;
- for (j = 0; j < len && !break2; j++) {
- pos = log.log2lin(log.lin2log(i) * intermediate[j]);
- // #1670, lastPos is #3113
- if (pos > min &&
- (!minor || lastPos <= max) &&
- typeof lastPos !== 'undefined') {
- positions.push(lastPos);
- }
- if (lastPos > max) {
- break2 = true;
- }
- lastPos = pos;
- }
- }
- // Third case: We are so deep in between whole logarithmic values that
- // we might as well handle the tick positions like a linear axis. For
- // example 1.01, 1.02, 1.03, 1.04.
- }
- else {
- var realMin = log.lin2log(min),
- realMax = log.lin2log(max),
- tickIntervalOption = minor ?
- axis.getMinorTickInterval() :
- options.tickInterval,
- filteredTickIntervalOption = tickIntervalOption === 'auto' ?
- null :
- tickIntervalOption,
- tickPixelIntervalOption = options.tickPixelInterval / (minor ? 5 : 1),
- totalPixelLength = minor ?
- axisLength / axis.tickPositions.length :
- axisLength;
- interval = pick(filteredTickIntervalOption, log.minorAutoInterval, (realMax - realMin) *
- tickPixelIntervalOption / (totalPixelLength || 1));
- interval = normalizeTickInterval(interval, void 0, getMagnitude(interval));
- positions = axis.getLinearTickPositions(interval, realMin, realMax).map(log.log2lin);
- if (!minor) {
- log.minorAutoInterval = interval / 5;
- }
- }
- // Set the axis-level tickInterval variable
- if (!minor) {
- axis.tickInterval = interval;
- }
- return positions;
- };
- LogarithmicAxisAdditions.prototype.lin2log = function (num) {
- return Math.pow(10, num);
- };
- LogarithmicAxisAdditions.prototype.log2lin = function (num) {
- return Math.log(num) / Math.LN10;
- };
- return LogarithmicAxisAdditions;
- }());
- var LogarithmicAxis = /** @class */ (function () {
- function LogarithmicAxis() {
- }
- /**
- * Provides logarithmic support for axes.
- *
- * @private
- */
- LogarithmicAxis.compose = function (AxisClass) {
- AxisClass.keepProps.push('logarithmic');
- // HC <= 8 backwards compatibility, allow wrapping
- // Axis.prototype.lin2log and log2lin
- // @todo Remove this in next major
- var axisProto = AxisClass.prototype;
- var logAxisProto = LogarithmicAxisAdditions.prototype;
- axisProto.log2lin = logAxisProto.log2lin;
- axisProto.lin2log = logAxisProto.lin2log;
- /* eslint-disable no-invalid-this */
- addEvent(AxisClass, 'init', function (e) {
- var axis = this;
- var options = e.userOptions;
- var logarithmic = axis.logarithmic;
- if (options.type !== 'logarithmic') {
- axis.logarithmic = void 0;
- }
- else {
- if (!logarithmic) {
- logarithmic = axis.logarithmic = new LogarithmicAxisAdditions(axis);
- }
- // HC <= 8 backwards compatibility, allow wrapping
- // Axis.prototype.lin2log and log2lin
- // @todo Remove this in next major
- if (axis.log2lin !== logarithmic.log2lin) {
- logarithmic.log2lin = axis.log2lin.bind(axis);
- }
- if (axis.lin2log !== logarithmic.lin2log) {
- logarithmic.lin2log = axis.lin2log.bind(axis);
- }
- }
- });
- addEvent(AxisClass, 'afterInit', function () {
- var axis = this;
- var log = axis.logarithmic;
- // extend logarithmic axis
- if (log) {
- axis.lin2val = function (num) {
- return log.lin2log(num);
- };
- axis.val2lin = function (num) {
- return log.log2lin(num);
- };
- }
- });
- };
- return LogarithmicAxis;
- }());
- LogarithmicAxis.compose(Axis); // @todo move to factory functions
- return LogarithmicAxis;
- });
- _registerModule(_modules, 'Core/Axis/PlotLineOrBand.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Axis, H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Options for plot bands on axes.
- *
- * @typedef {Highcharts.XAxisPlotBandsOptions|Highcharts.YAxisPlotBandsOptions|Highcharts.ZAxisPlotBandsOptions} Highcharts.AxisPlotBandsOptions
- */
- /**
- * Options for plot band labels on axes.
- *
- * @typedef {Highcharts.XAxisPlotBandsLabelOptions|Highcharts.YAxisPlotBandsLabelOptions|Highcharts.ZAxisPlotBandsLabelOptions} Highcharts.AxisPlotBandsLabelOptions
- */
- /**
- * Options for plot lines on axes.
- *
- * @typedef {Highcharts.XAxisPlotLinesOptions|Highcharts.YAxisPlotLinesOptions|Highcharts.ZAxisPlotLinesOptions} Highcharts.AxisPlotLinesOptions
- */
- /**
- * Options for plot line labels on axes.
- *
- * @typedef {Highcharts.XAxisPlotLinesLabelOptions|Highcharts.YAxisPlotLinesLabelOptions|Highcharts.ZAxisPlotLinesLabelOptions} Highcharts.AxisPlotLinesLabelOptions
- */
- var arrayMax = U.arrayMax,
- arrayMin = U.arrayMin,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- erase = U.erase,
- extend = U.extend,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The object wrapper for plot lines and plot bands
- *
- * @class
- * @name Highcharts.PlotLineOrBand
- *
- * @param {Highcharts.Axis} axis
- *
- * @param {Highcharts.AxisPlotLinesOptions|Highcharts.AxisPlotBandsOptions} [options]
- */
- var PlotLineOrBand = /** @class */ (function () {
- function PlotLineOrBand(axis, options) {
- this.axis = axis;
- if (options) {
- this.options = options;
- this.id = options.id;
- }
- }
- /**
- * Render the plot line or plot band. If it is already existing,
- * move it.
- *
- * @private
- * @function Highcharts.PlotLineOrBand#render
- * @return {Highcharts.PlotLineOrBand|undefined}
- */
- PlotLineOrBand.prototype.render = function () {
- H.fireEvent(this, 'render');
- var plotLine = this,
- axis = plotLine.axis,
- horiz = axis.horiz,
- log = axis.logarithmic,
- options = plotLine.options,
- optionsLabel = options.label,
- label = plotLine.label,
- to = options.to,
- from = options.from,
- value = options.value,
- isBand = defined(from) && defined(to),
- isLine = defined(value),
- svgElem = plotLine.svgElem,
- isNew = !svgElem,
- path = [],
- color = options.color,
- zIndex = pick(options.zIndex, 0),
- events = options.events,
- attribs = {
- 'class': 'highcharts-plot-' + (isBand ? 'band ' : 'line ') +
- (options.className || '')
- },
- groupAttribs = {},
- renderer = axis.chart.renderer,
- groupName = isBand ? 'bands' : 'lines',
- group;
- // logarithmic conversion
- if (log) {
- from = log.log2lin(from);
- to = log.log2lin(to);
- value = log.log2lin(value);
- }
- // Set the presentational attributes
- if (!axis.chart.styledMode) {
- if (isLine) {
- attribs.stroke = color || '#999999';
- attribs['stroke-width'] = pick(options.width, 1);
- if (options.dashStyle) {
- attribs.dashstyle =
- options.dashStyle;
- }
- }
- else if (isBand) { // plot band
- attribs.fill = color || '#e6ebf5';
- if (options.borderWidth) {
- attribs.stroke = options.borderColor;
- attribs['stroke-width'] = options.borderWidth;
- }
- }
- }
- // Grouping and zIndex
- groupAttribs.zIndex = zIndex;
- groupName += '-' + zIndex;
- group = axis.plotLinesAndBandsGroups[groupName];
- if (!group) {
- axis.plotLinesAndBandsGroups[groupName] = group =
- renderer.g('plot-' + groupName)
- .attr(groupAttribs).add();
- }
- // Create the path
- if (isNew) {
- /**
- * SVG element of the plot line or band.
- *
- * @name Highcharts.PlotLineOrBand#svgElement
- * @type {Highcharts.SVGElement}
- */
- plotLine.svgElem = svgElem = renderer
- .path()
- .attr(attribs)
- .add(group);
- }
- // Set the path or return
- if (isLine) {
- path = axis.getPlotLinePath({
- value: value,
- lineWidth: svgElem.strokeWidth(),
- acrossPanes: options.acrossPanes
- });
- }
- else if (isBand) { // plot band
- path = axis.getPlotBandPath(from, to, options);
- }
- else {
- return;
- }
- // common for lines and bands
- // Add events only if they were not added before.
- if (!plotLine.eventsAdded && events) {
- objectEach(events, function (event, eventType) {
- svgElem.on(eventType, function (e) {
- events[eventType].apply(plotLine, [e]);
- });
- });
- plotLine.eventsAdded = true;
- }
- if ((isNew || !svgElem.d) && path && path.length) {
- svgElem.attr({ d: path });
- }
- else if (svgElem) {
- if (path) {
- svgElem.show(true);
- svgElem.animate({ d: path });
- }
- else if (svgElem.d) {
- svgElem.hide();
- if (label) {
- plotLine.label = label = label.destroy();
- }
- }
- }
- // the plot band/line label
- if (optionsLabel &&
- (defined(optionsLabel.text) || defined(optionsLabel.formatter)) &&
- path &&
- path.length &&
- axis.width > 0 &&
- axis.height > 0 &&
- !path.isFlat) {
- // apply defaults
- optionsLabel = merge({
- align: horiz && isBand && 'center',
- x: horiz ? !isBand && 4 : 10,
- verticalAlign: !horiz && isBand && 'middle',
- y: horiz ? isBand ? 16 : 10 : isBand ? 6 : -4,
- rotation: horiz && !isBand && 90
- }, optionsLabel);
- this.renderLabel(optionsLabel, path, isBand, zIndex);
- }
- else if (label) { // move out of sight
- label.hide();
- }
- // chainable
- return plotLine;
- };
- /**
- * Render and align label for plot line or band.
- *
- * @private
- * @function Highcharts.PlotLineOrBand#renderLabel
- * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel
- * @param {Highcharts.SVGPathArray} path
- * @param {boolean} [isBand]
- * @param {number} [zIndex]
- * @return {void}
- */
- PlotLineOrBand.prototype.renderLabel = function (optionsLabel, path, isBand, zIndex) {
- var plotLine = this,
- label = plotLine.label,
- renderer = plotLine.axis.chart.renderer,
- attribs,
- xBounds,
- yBounds,
- x,
- y,
- labelText;
- // add the SVG element
- if (!label) {
- attribs = {
- align: optionsLabel.textAlign || optionsLabel.align,
- rotation: optionsLabel.rotation,
- 'class': 'highcharts-plot-' + (isBand ? 'band' : 'line') +
- '-label ' + (optionsLabel.className || '')
- };
- attribs.zIndex = zIndex;
- labelText = this.getLabelText(optionsLabel);
- /**
- * SVG element of the label.
- *
- * @name Highcharts.PlotLineOrBand#label
- * @type {Highcharts.SVGElement}
- */
- plotLine.label = label = renderer
- .text(labelText, 0, 0, optionsLabel.useHTML)
- .attr(attribs)
- .add();
- if (!this.axis.chart.styledMode) {
- label.css(optionsLabel.style);
- }
- }
- // get the bounding box and align the label
- // #3000 changed to better handle choice between plotband or plotline
- xBounds = path.xBounds ||
- [path[0][1], path[1][1], (isBand ? path[2][1] : path[0][1])];
- yBounds = path.yBounds ||
- [path[0][2], path[1][2], (isBand ? path[2][2] : path[0][2])];
- x = arrayMin(xBounds);
- y = arrayMin(yBounds);
- label.align(optionsLabel, false, {
- x: x,
- y: y,
- width: arrayMax(xBounds) - x,
- height: arrayMax(yBounds) - y
- });
- label.show(true);
- };
- /**
- * Get label's text content.
- *
- * @private
- * @function Highcharts.PlotLineOrBand#getLabelText
- * @param {Highcharts.AxisPlotLinesLabelOptions|Highcharts.AxisPlotBandsLabelOptions} optionsLabel
- * @return {string}
- */
- PlotLineOrBand.prototype.getLabelText = function (optionsLabel) {
- return defined(optionsLabel.formatter) ?
- optionsLabel.formatter
- .call(this) :
- optionsLabel.text;
- };
- /**
- * Remove the plot line or band.
- *
- * @function Highcharts.PlotLineOrBand#destroy
- * @return {void}
- */
- PlotLineOrBand.prototype.destroy = function () {
- // remove it from the lookup
- erase(this.axis.plotLinesAndBands, this);
- delete this.axis;
- destroyObjectProperties(this);
- };
- return PlotLineOrBand;
- }());
- /* eslint-enable no-invalid-this, valid-jsdoc */
- // Object with members for extending the Axis prototype
- extend(Axis.prototype, /** @lends Highcharts.Axis.prototype */ {
- /**
- * An array of colored bands stretching across the plot area marking an
- * interval on the axis.
- *
- * In styled mode, the plot bands are styled by the `.highcharts-plot-band`
- * class in addition to the `className` option.
- *
- * @productdesc {highcharts}
- * In a gauge, a plot band on the Y axis (value axis) will stretch along the
- * perimeter of the gauge.
- *
- * @type {Array<*>}
- * @product highcharts highstock gantt
- * @apioption xAxis.plotBands
- */
- /**
- * Flag to decide if plotBand should be rendered across all panes.
- *
- * @since 7.1.2
- * @product highstock
- * @type {boolean}
- * @default true
- * @apioption xAxis.plotBands.acrossPanes
- */
- /**
- * Border color for the plot band. Also requires `borderWidth` to be set.
- *
- * @type {Highcharts.ColorString}
- * @apioption xAxis.plotBands.borderColor
- */
- /**
- * Border width for the plot band. Also requires `borderColor` to be set.
- *
- * @type {number}
- * @default 0
- * @apioption xAxis.plotBands.borderWidth
- */
- /**
- * A custom class name, in addition to the default `highcharts-plot-band`,
- * to apply to each individual band.
- *
- * @type {string}
- * @since 5.0.0
- * @apioption xAxis.plotBands.className
- */
- /**
- * The color of the plot band.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-color/
- * Color band
- * @sample {highstock} stock/xaxis/plotbands/
- * Plot band on Y axis
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #e6ebf5
- * @apioption xAxis.plotBands.color
- */
- /**
- * An object defining mouse events for the plot band. Supported properties
- * are `click`, `mouseover`, `mouseout`, `mousemove`.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-events/
- * Mouse events demonstrated
- *
- * @since 1.2
- * @apioption xAxis.plotBands.events
- */
- /**
- * Click event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotBands.events.click
- */
- /**
- * Mouse move event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotBands.events.mousemove
- */
- /**
- * Mouse out event on the corner of a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotBands.events.mouseout
- */
- /**
- * Mouse over event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotBands.events.mouseover
- */
- /**
- * The start position of the plot band in axis units.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-color/
- * Datetime axis
- * @sample {highcharts} highcharts/xaxis/plotbands-from/
- * Categorized axis
- * @sample {highstock} stock/xaxis/plotbands/
- * Plot band on Y axis
- *
- * @type {number}
- * @apioption xAxis.plotBands.from
- */
- /**
- * An id used for identifying the plot band in Axis.removePlotBand.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-id/
- * Remove plot band by id
- * @sample {highstock} highcharts/xaxis/plotbands-id/
- * Remove plot band by id
- *
- * @type {string}
- * @apioption xAxis.plotBands.id
- */
- /**
- * The end position of the plot band in axis units.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-color/
- * Datetime axis
- * @sample {highcharts} highcharts/xaxis/plotbands-from/
- * Categorized axis
- * @sample {highstock} stock/xaxis/plotbands/
- * Plot band on Y axis
- *
- * @type {number}
- * @apioption xAxis.plotBands.to
- */
- /**
- * The z index of the plot band within the chart, relative to other
- * elements. Using the same z index as another element may give
- * unpredictable results, as the last rendered element will be on top.
- * Values from 0 to 20 make sense.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-color/
- * Behind plot lines by default
- * @sample {highcharts} highcharts/xaxis/plotbands-zindex/
- * Above plot lines
- * @sample {highcharts} highcharts/xaxis/plotbands-zindex-above-series/
- * Above plot lines and series
- *
- * @type {number}
- * @since 1.2
- * @apioption xAxis.plotBands.zIndex
- */
- /**
- * Text labels for the plot bands
- *
- * @product highcharts highstock gantt
- * @apioption xAxis.plotBands.label
- */
- /**
- * Horizontal alignment of the label. Can be one of "left", "center" or
- * "right".
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-align/
- * Aligned to the right
- * @sample {highstock} stock/xaxis/plotbands-label/
- * Plot band with labels
- *
- * @type {Highcharts.AlignValue}
- * @default center
- * @since 2.1
- * @apioption xAxis.plotBands.label.align
- */
- /**
- * Rotation of the text label in degrees .
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/
- * Vertical text
- *
- * @type {number}
- * @default 0
- * @since 2.1
- * @apioption xAxis.plotBands.label.rotation
- */
- /**
- * CSS styles for the text label.
- *
- * In styled mode, the labels are styled by the
- * `.highcharts-plot-band-label` class.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-style/
- * Blue and bold label
- *
- * @type {Highcharts.CSSObject}
- * @since 2.1
- * @apioption xAxis.plotBands.label.style
- */
- /**
- * The string text itself. A subset of HTML is supported.
- *
- * @type {string}
- * @since 2.1
- * @apioption xAxis.plotBands.label.text
- */
- /**
- * The text alignment for the label. While `align` determines where the
- * texts anchor point is placed within the plot band, `textAlign` determines
- * how the text is aligned against its anchor point. Possible values are
- * "left", "center" and "right". Defaults to the same as the `align` option.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-rotation/
- * Vertical text in center position but text-aligned left
- *
- * @type {Highcharts.AlignValue}
- * @since 2.1
- * @apioption xAxis.plotBands.label.textAlign
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the labels.
- *
- * @type {boolean}
- * @default false
- * @since 3.0.3
- * @apioption xAxis.plotBands.label.useHTML
- */
- /**
- * Vertical alignment of the label relative to the plot band. Can be one of
- * "top", "middle" or "bottom".
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-verticalalign/
- * Vertically centered label
- * @sample {highstock} stock/xaxis/plotbands-label/
- * Plot band with labels
- *
- * @type {Highcharts.VerticalAlignValue}
- * @default top
- * @since 2.1
- * @apioption xAxis.plotBands.label.verticalAlign
- */
- /**
- * Horizontal position relative the alignment. Default varies by
- * orientation.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-align/
- * Aligned 10px from the right edge
- * @sample {highstock} stock/xaxis/plotbands-label/
- * Plot band with labels
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.plotBands.label.x
- */
- /**
- * Vertical position of the text baseline relative to the alignment. Default
- * varies by orientation.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-label-y/
- * Label on x axis
- * @sample {highstock} stock/xaxis/plotbands-label/
- * Plot band with labels
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.plotBands.label.y
- */
- /**
- * An array of lines stretching across the plot area, marking a specific
- * value on one of the axes.
- *
- * In styled mode, the plot lines are styled by the
- * `.highcharts-plot-line` class in addition to the `className` option.
- *
- * @type {Array<*>}
- * @product highcharts highstock gantt
- * @sample {highcharts} highcharts/xaxis/plotlines-color/
- * Basic plot line
- * @sample {highcharts} highcharts/series-solidgauge/labels-auto-aligned/
- * Solid gauge plot line
- * @apioption xAxis.plotLines
- */
- /**
- * Flag to decide if plotLine should be rendered across all panes.
- *
- * @sample {highstock} stock/xaxis/plotlines-acrosspanes/
- * Plot lines on different panes
- *
- * @since 7.1.2
- * @product highstock
- * @type {boolean}
- * @default true
- * @apioption xAxis.plotLines.acrossPanes
- */
- /**
- * A custom class name, in addition to the default `highcharts-plot-line`,
- * to apply to each individual line.
- *
- * @type {string}
- * @since 5.0.0
- * @apioption xAxis.plotLines.className
- */
- /**
- * The color of the line.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-color/
- * A red line from X axis
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {Highcharts.ColorString}
- * @default #999999
- * @apioption xAxis.plotLines.color
- */
- /**
- * The dashing or dot style for the plot line. For possible values see
- * [this overview](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/plotoptions/series-dashstyle-all/).
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-dashstyle/
- * Dash and dot pattern
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {Highcharts.DashStyleValue}
- * @default Solid
- * @since 1.2
- * @apioption xAxis.plotLines.dashStyle
- */
- /**
- * An object defining mouse events for the plot line. Supported
- * properties are `click`, `mouseover`, `mouseout`, `mousemove`.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-events/
- * Mouse events demonstrated
- *
- * @since 1.2
- * @apioption xAxis.plotLines.events
- */
- /**
- * Click event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotLines.events.click
- */
- /**
- * Mouse move event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotLines.events.mousemove
- */
- /**
- * Mouse out event on the corner of a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotLines.events.mouseout
- */
- /**
- * Mouse over event on a plot band.
- *
- * @type {Highcharts.EventCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotLines.events.mouseover
- */
- /**
- * An id used for identifying the plot line in Axis.removePlotLine.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-id/
- * Remove plot line by id
- *
- * @type {string}
- * @apioption xAxis.plotLines.id
- */
- /**
- * The position of the line in axis units.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-color/
- * Between two categories on X axis
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {number}
- * @apioption xAxis.plotLines.value
- */
- /**
- * The width or thickness of the plot line.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-color/
- * 2px wide line from X axis
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {number}
- * @default 2
- * @apioption xAxis.plotLines.width
- */
- /**
- * The z index of the plot line within the chart.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-zindex-behind/
- * Behind plot lines by default
- * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above/
- * Above plot lines
- * @sample {highcharts} highcharts/xaxis/plotlines-zindex-above-all/
- * Above plot lines and series
- *
- * @type {number}
- * @since 1.2
- * @apioption xAxis.plotLines.zIndex
- */
- /**
- * Text labels for the plot bands
- *
- * @apioption xAxis.plotLines.label
- */
- /**
- * Horizontal alignment of the label. Can be one of "left", "center" or
- * "right".
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/
- * Aligned to the right
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {Highcharts.AlignValue}
- * @default left
- * @since 2.1
- * @apioption xAxis.plotLines.label.align
- */
- /**
- * Callback JavaScript function to format the label. Useful properties like
- * the value of plot line or the range of plot band (`from` & `to`
- * properties) can be found in `this.options` object.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-plotbands-label-formatter
- * Label formatters for plot line and plot band.
- * @type {Highcharts.FormatterCallbackFunction<Highcharts.PlotLineOrBand>}
- * @apioption xAxis.plotLines.label.formatter
- */
- /**
- * Rotation of the text label in degrees. Defaults to 0 for horizontal plot
- * lines and 90 for vertical lines.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/
- * Slanted text
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.plotLines.label.rotation
- */
- /**
- * CSS styles for the text label.
- *
- * In styled mode, the labels are styled by the
- * `.highcharts-plot-line-label` class.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-style/
- * Blue and bold label
- *
- * @type {Highcharts.CSSObject}
- * @since 2.1
- * @apioption xAxis.plotLines.label.style
- */
- /**
- * The text itself. A subset of HTML is supported.
- *
- * @type {string}
- * @since 2.1
- * @apioption xAxis.plotLines.label.text
- */
- /**
- * The text alignment for the label. While `align` determines where the
- * texts anchor point is placed within the plot band, `textAlign` determines
- * how the text is aligned against its anchor point. Possible values are
- * "left", "center" and "right". Defaults to the same as the `align` option.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-textalign/
- * Text label in bottom position
- *
- * @type {Highcharts.AlignValue}
- * @since 2.1
- * @apioption xAxis.plotLines.label.textAlign
- */
- /**
- * Whether to [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the labels.
- *
- * @type {boolean}
- * @default false
- * @since 3.0.3
- * @apioption xAxis.plotLines.label.useHTML
- */
- /**
- * Vertical alignment of the label relative to the plot line. Can be
- * one of "top", "middle" or "bottom".
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-verticalalign-middle/
- * Vertically centered label
- *
- * @type {Highcharts.VerticalAlignValue}
- * @default {highcharts} top
- * @default {highstock} top
- * @since 2.1
- * @apioption xAxis.plotLines.label.verticalAlign
- */
- /**
- * Horizontal position relative the alignment. Default varies by
- * orientation.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-align-right/
- * Aligned 10px from the right edge
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.plotLines.label.x
- */
- /**
- * Vertical position of the text baseline relative to the alignment. Default
- * varies by orientation.
- *
- * @sample {highcharts} highcharts/xaxis/plotlines-label-y/
- * Label below the plot line
- * @sample {highstock} stock/xaxis/plotlines/
- * Plot line on Y axis
- *
- * @type {number}
- * @since 2.1
- * @apioption xAxis.plotLines.label.y
- */
- /**
- *
- * @type {Array<*>}
- * @extends xAxis.plotBands
- * @apioption yAxis.plotBands
- */
- /**
- * In a gauge chart, this option determines the inner radius of the
- * plot band that stretches along the perimeter. It can be given as
- * a percentage string, like `"100%"`, or as a pixel number, like `100`.
- * By default, the inner radius is controlled by the [thickness](
- * #yAxis.plotBands.thickness) option.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-gauge
- * Gauge plot band
- *
- * @type {number|string}
- * @since 2.3
- * @product highcharts
- * @apioption yAxis.plotBands.innerRadius
- */
- /**
- * In a gauge chart, this option determines the outer radius of the
- * plot band that stretches along the perimeter. It can be given as
- * a percentage string, like `"100%"`, or as a pixel number, like `100`.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-gauge
- * Gauge plot band
- *
- * @type {number|string}
- * @default 100%
- * @since 2.3
- * @product highcharts
- * @apioption yAxis.plotBands.outerRadius
- */
- /**
- * In a gauge chart, this option sets the width of the plot band
- * stretching along the perimeter. It can be given as a percentage
- * string, like `"10%"`, or as a pixel number, like `10`. The default
- * value 10 is the same as the default [tickLength](#yAxis.tickLength),
- * thus making the plot band act as a background for the tick markers.
- *
- * @sample {highcharts} highcharts/xaxis/plotbands-gauge
- * Gauge plot band
- *
- * @type {number|string}
- * @default 10
- * @since 2.3
- * @product highcharts
- * @apioption yAxis.plotBands.thickness
- */
- /**
- * @type {Array<*>}
- * @extends xAxis.plotLines
- * @apioption yAxis.plotLines
- */
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Internal function to create the SVG path definition for a plot band.
- *
- * @function Highcharts.Axis#getPlotBandPath
- *
- * @param {number} from
- * The axis value to start from.
- *
- * @param {number} to
- * The axis value to end on.
- *
- * @return {Highcharts.SVGPathArray}
- * The SVG path definition in array form.
- */
- getPlotBandPath: function (from, to) {
- var toPath = this.getPlotLinePath({
- value: to,
- force: true,
- acrossPanes: this.options.acrossPanes
- }),
- path = this.getPlotLinePath({
- value: from,
- force: true,
- acrossPanes: this.options.acrossPanes
- }),
- result = [],
- i,
- // #4964 check if chart is inverted or plotband is on yAxis
- horiz = this.horiz,
- plus = 1,
- isFlat,
- outside = (from < this.min && to < this.min) ||
- (from > this.max && to > this.max);
- if (path && toPath) {
- // Flat paths don't need labels (#3836)
- if (outside) {
- isFlat = path.toString() === toPath.toString();
- plus = 0;
- }
- // Go over each subpath - for panes in Highstock
- for (i = 0; i < path.length; i += 2) {
- var pathStart = path[i],
- pathEnd = path[i + 1],
- toPathStart = toPath[i],
- toPathEnd = toPath[i + 1];
- // Type checking all affected path segments. Consider something
- // smarter.
- if ((pathStart[0] === 'M' || pathStart[0] === 'L') &&
- (pathEnd[0] === 'M' || pathEnd[0] === 'L') &&
- (toPathStart[0] === 'M' || toPathStart[0] === 'L') &&
- (toPathEnd[0] === 'M' || toPathEnd[0] === 'L')) {
- // Add 1 pixel when coordinates are the same
- if (horiz && toPathStart[1] === pathStart[1]) {
- toPathStart[1] += plus;
- toPathEnd[1] += plus;
- }
- else if (!horiz && toPathStart[2] === pathStart[2]) {
- toPathStart[2] += plus;
- toPathEnd[2] += plus;
- }
- result.push(['M', pathStart[1], pathStart[2]], ['L', pathEnd[1], pathEnd[2]], ['L', toPathEnd[1], toPathEnd[2]], ['L', toPathStart[1], toPathStart[2]], ['Z']);
- }
- result.isFlat = isFlat;
- }
- }
- else { // outside the axis area
- path = null;
- }
- return result;
- },
- /**
- * Add a plot band after render time.
- *
- * @sample highcharts/members/axis-addplotband/
- * Toggle the plot band from a button
- *
- * @function Highcharts.Axis#addPlotBand
- *
- * @param {Highcharts.AxisPlotBandsOptions} options
- * A configuration object for the plot band, as defined in
- * [xAxis.plotBands](https://api.highcharts.com/highcharts/xAxis.plotBands).
- *
- * @return {Highcharts.PlotLineOrBand|undefined}
- * The added plot band.
- */
- addPlotBand: function (options) {
- return this.addPlotBandOrLine(options, 'plotBands');
- },
- /**
- * Add a plot line after render time.
- *
- * @sample highcharts/members/axis-addplotline/
- * Toggle the plot line from a button
- *
- * @function Highcharts.Axis#addPlotLine
- *
- * @param {Highcharts.AxisPlotLinesOptions} options
- * A configuration object for the plot line, as defined in
- * [xAxis.plotLines](https://api.highcharts.com/highcharts/xAxis.plotLines).
- *
- * @return {Highcharts.PlotLineOrBand|undefined}
- * The added plot line.
- */
- addPlotLine: function (options) {
- return this.addPlotBandOrLine(options, 'plotLines');
- },
- /**
- * Add a plot band or plot line after render time. Called from addPlotBand
- * and addPlotLine internally.
- *
- * @private
- * @function Highcharts.Axis#addPlotBandOrLine
- *
- * @param {Highcharts.AxisPlotBandsOptions|Highcharts.AxisPlotLinesOptions} options
- * The plotBand or plotLine configuration object.
- *
- * @param {"plotBands"|"plotLines"} [coll]
- *
- * @return {Highcharts.PlotLineOrBand|undefined}
- */
- addPlotBandOrLine: function (options, coll) {
- var obj = new PlotLineOrBand(this,
- options).render(),
- userOptions = this.userOptions;
- if (obj) { // #2189
- // Add it to the user options for exporting and Axis.update
- if (coll) {
- // Workaround Microsoft/TypeScript issue #32693
- var updatedOptions = (userOptions[coll] || []);
- updatedOptions.push(options);
- userOptions[coll] = updatedOptions;
- }
- this.plotLinesAndBands.push(obj);
- this._addedPlotLB = true;
- }
- return obj;
- },
- /**
- * Remove a plot band or plot line from the chart by id. Called internally
- * from `removePlotBand` and `removePlotLine`.
- *
- * @private
- * @function Highcharts.Axis#removePlotBandOrLine
- * @param {string} id
- * @return {void}
- */
- removePlotBandOrLine: function (id) {
- var plotLinesAndBands = this.plotLinesAndBands,
- options = this.options,
- userOptions = this.userOptions,
- i = plotLinesAndBands.length;
- while (i--) {
- if (plotLinesAndBands[i].id === id) {
- plotLinesAndBands[i].destroy();
- }
- }
- ([
- options.plotLines || [],
- userOptions.plotLines || [],
- options.plotBands || [],
- userOptions.plotBands || []
- ]).forEach(function (arr) {
- i = arr.length;
- while (i--) {
- if ((arr[i] || {}).id === id) {
- erase(arr, arr[i]);
- }
- }
- });
- },
- /**
- * Remove a plot band by its id.
- *
- * @sample highcharts/members/axis-removeplotband/
- * Remove plot band by id
- * @sample highcharts/members/axis-addplotband/
- * Toggle the plot band from a button
- *
- * @function Highcharts.Axis#removePlotBand
- *
- * @param {string} id
- * The plot band's `id` as given in the original configuration
- * object or in the `addPlotBand` option.
- *
- * @return {void}
- */
- removePlotBand: function (id) {
- this.removePlotBandOrLine(id);
- },
- /**
- * Remove a plot line by its id.
- *
- * @sample highcharts/xaxis/plotlines-id/
- * Remove plot line by id
- * @sample highcharts/members/axis-addplotline/
- * Toggle the plot line from a button
- *
- * @function Highcharts.Axis#removePlotLine
- *
- * @param {string} id
- * The plot line's `id` as given in the original configuration
- * object or in the `addPlotLine` option.
- */
- removePlotLine: function (id) {
- this.removePlotBandOrLine(id);
- }
- });
- H.PlotLineOrBand = PlotLineOrBand;
- return H.PlotLineOrBand;
- });
- _registerModule(_modules, 'Core/Tooltip.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var doc = H.doc;
- var clamp = U.clamp,
- css = U.css,
- defined = U.defined,
- discardElement = U.discardElement,
- extend = U.extend,
- fireEvent = U.fireEvent,
- format = U.format,
- isNumber = U.isNumber,
- isString = U.isString,
- merge = U.merge,
- pick = U.pick,
- splat = U.splat,
- syncTimeout = U.syncTimeout,
- timeUnits = U.timeUnits;
- /**
- * Callback function to format the text of the tooltip from scratch.
- *
- * In case of single or shared tooltips, a string should be be returned. In case
- * of splitted tooltips, it should return an array where the first item is the
- * header, and subsequent items are mapped to the points. Return `false` to
- * disable tooltip for a specific point on series.
- *
- * @callback Highcharts.TooltipFormatterCallbackFunction
- *
- * @param {Highcharts.TooltipFormatterContextObject} this
- * Context to format
- *
- * @param {Highcharts.Tooltip} tooltip
- * The tooltip instance
- *
- * @return {false|string|Array<(string|null|undefined)>|null|undefined}
- * Formatted text or false
- */
- /**
- * @interface Highcharts.TooltipFormatterContextObject
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#color
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#colorIndex
- * @type {number|undefined}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#key
- * @type {number}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#percentage
- * @type {number|undefined}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#point
- * @type {Highcharts.Point}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#points
- * @type {Array<Highcharts.TooltipFormatterContextObject>|undefined}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#series
- * @type {Highcharts.Series}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#total
- * @type {number|undefined}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#x
- * @type {number}
- */ /**
- * @name Highcharts.TooltipFormatterContextObject#y
- * @type {number}
- */
- /**
- * A callback function to place the tooltip in a specific position.
- *
- * @callback Highcharts.TooltipPositionerCallbackFunction
- *
- * @param {Highcharts.Tooltip} this
- * Tooltip context of the callback.
- *
- * @param {number} labelWidth
- * Width of the tooltip.
- *
- * @param {number} labelHeight
- * Height of the tooltip.
- *
- * @param {Highcharts.Point|Highcharts.TooltipPositionerPointObject} point
- * Point information for positioning a tooltip.
- *
- * @return {Highcharts.PositionObject}
- * New position for the tooltip.
- */
- /**
- * Point information for positioning a tooltip.
- *
- * @interface Highcharts.TooltipPositionerPointObject
- */ /**
- * If `tooltip.split` option is enabled and positioner is called for each of the
- * boxes separately, this property indicates the call on the xAxis header, which
- * is not a point itself.
- * @name Highcharts.TooltipPositionerPointObject#isHeader
- * @type {boolean}
- */ /**
- * The reference point relative to the plot area. Add chart.plotLeft to get the
- * full coordinates.
- * @name Highcharts.TooltipPositionerPointObject#plotX
- * @type {number}
- */ /**
- * The reference point relative to the plot area. Add chart.plotTop to get the
- * full coordinates.
- * @name Highcharts.TooltipPositionerPointObject#plotY
- * @type {number}
- */
- /**
- * @typedef {"callout"|"circle"|"square"} Highcharts.TooltipShapeValue
- */
- ''; // separates doclets above from variables below
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Tooltip of a chart.
- *
- * @class
- * @name Highcharts.Tooltip
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.TooltipOptions} options
- * Tooltip options.
- */
- var Tooltip = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Tooltip(chart, options) {
- this.container = void 0;
- this.crosshairs = [];
- this.distance = 0;
- this.isHidden = true;
- this.isSticky = false;
- this.now = {};
- this.options = {};
- this.outside = false;
- this.chart = chart;
- this.init(chart, options);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * In styled mode, apply the default filter for the tooltip drop-shadow. It
- * needs to have an id specific to the chart, otherwise there will be issues
- * when one tooltip adopts the filter of a different chart, specifically one
- * where the container is hidden.
- *
- * @private
- * @function Highcharts.Tooltip#applyFilter
- */
- Tooltip.prototype.applyFilter = function () {
- var chart = this.chart;
- chart.renderer.definition({
- tagName: 'filter',
- id: 'drop-shadow-' + chart.index,
- opacity: 0.5,
- children: [{
- tagName: 'feGaussianBlur',
- 'in': 'SourceAlpha',
- stdDeviation: 1
- }, {
- tagName: 'feOffset',
- dx: 1,
- dy: 1
- }, {
- tagName: 'feComponentTransfer',
- children: [{
- tagName: 'feFuncA',
- type: 'linear',
- slope: 0.3
- }]
- }, {
- tagName: 'feMerge',
- children: [{
- tagName: 'feMergeNode'
- }, {
- tagName: 'feMergeNode',
- 'in': 'SourceGraphic'
- }]
- }]
- });
- chart.renderer.definition({
- tagName: 'style',
- textContent: '.highcharts-tooltip-' + chart.index + '{' +
- 'filter:url(#drop-shadow-' + chart.index + ')' +
- '}'
- });
- };
- /**
- * Build the body (lines) of the tooltip by iterating over the items and
- * returning one entry for each item, abstracting this functionality allows
- * to easily overwrite and extend it.
- *
- * @private
- * @function Highcharts.Tooltip#bodyFormatter
- * @param {Array<(Highcharts.Point|Highcharts.Series)>} items
- * @return {Array<string>}
- */
- Tooltip.prototype.bodyFormatter = function (items) {
- return items.map(function (item) {
- var tooltipOptions = item.series.tooltipOptions;
- return (tooltipOptions[(item.point.formatPrefix || 'point') + 'Formatter'] ||
- item.point.tooltipFormatter).call(item.point, tooltipOptions[(item.point.formatPrefix || 'point') + 'Format'] || '');
- });
- };
- /**
- * Destroy the single tooltips in a split tooltip.
- * If the tooltip is active then it is not destroyed, unless forced to.
- *
- * @private
- * @function Highcharts.Tooltip#cleanSplit
- *
- * @param {boolean} [force]
- * Force destroy all tooltips.
- */
- Tooltip.prototype.cleanSplit = function (force) {
- this.chart.series.forEach(function (series) {
- var tt = series && series.tt;
- if (tt) {
- if (!tt.isActive || force) {
- series.tt = tt.destroy();
- }
- else {
- tt.isActive = false;
- }
- }
- });
- };
- /**
- * In case no user defined formatter is given, this will be used. Note that
- * the context here is an object holding point, series, x, y etc.
- *
- * @function Highcharts.Tooltip#defaultFormatter
- *
- * @param {Highcharts.Tooltip} tooltip
- *
- * @return {Array<string>}
- */
- Tooltip.prototype.defaultFormatter = function (tooltip) {
- var items = this.points || splat(this),
- s;
- // Build the header
- s = [tooltip.tooltipFooterHeaderFormatter(items[0])];
- // build the values
- s = s.concat(tooltip.bodyFormatter(items));
- // footer
- s.push(tooltip.tooltipFooterHeaderFormatter(items[0], true));
- return s;
- };
- /**
- * Removes and destroys the tooltip and its elements.
- *
- * @function Highcharts.Tooltip#destroy
- */
- Tooltip.prototype.destroy = function () {
- // Destroy and clear local variables
- if (this.label) {
- this.label = this.label.destroy();
- }
- if (this.split && this.tt) {
- this.cleanSplit(this.chart, true);
- this.tt = this.tt.destroy();
- }
- if (this.renderer) {
- this.renderer = this.renderer.destroy();
- discardElement(this.container);
- }
- U.clearTimeout(this.hideTimer);
- U.clearTimeout(this.tooltipTimeout);
- };
- /**
- * Extendable method to get the anchor position of the tooltip
- * from a point or set of points
- *
- * @private
- * @function Highcharts.Tooltip#getAnchor
- *
- * @param {Highcharts.Point|Array<Highcharts.Point>} points
- *
- * @param {Highcharts.PointerEventObject} [mouseEvent]
- *
- * @return {Array<number>}
- */
- Tooltip.prototype.getAnchor = function (points, mouseEvent) {
- var ret,
- chart = this.chart,
- pointer = chart.pointer,
- inverted = chart.inverted,
- plotTop = chart.plotTop,
- plotLeft = chart.plotLeft,
- plotX = 0,
- plotY = 0,
- yAxis,
- xAxis;
- points = splat(points);
- // When tooltip follows mouse, relate the position to the mouse
- if (this.followPointer && mouseEvent) {
- if (typeof mouseEvent.chartX === 'undefined') {
- mouseEvent = pointer.normalize(mouseEvent);
- }
- ret = [
- mouseEvent.chartX - plotLeft,
- mouseEvent.chartY - plotTop
- ];
- // Some series types use a specificly calculated tooltip position for
- // each point
- }
- else if (points[0].tooltipPos) {
- ret = points[0].tooltipPos;
- // When shared, use the average position
- }
- else {
- points.forEach(function (point) {
- yAxis = point.series.yAxis;
- xAxis = point.series.xAxis;
- plotX += point.plotX +
- (!inverted && xAxis ? xAxis.left - plotLeft : 0);
- plotY += (point.plotLow ?
- (point.plotLow + point.plotHigh) / 2 :
- point.plotY) + (!inverted && yAxis ? yAxis.top - plotTop : 0); // #1151
- });
- plotX /= points.length;
- plotY /= points.length;
- ret = [
- inverted ? chart.plotWidth - plotY : plotX,
- this.shared && !inverted && points.length > 1 && mouseEvent ?
- // place shared tooltip next to the mouse (#424)
- mouseEvent.chartY - plotTop :
- inverted ? chart.plotHeight - plotX : plotY
- ];
- }
- return ret.map(Math.round);
- };
- /**
- * Get the optimal date format for a point, based on a range.
- *
- * @private
- * @function Highcharts.Tooltip#getDateFormat
- *
- * @param {number} range
- * The time range
- *
- * @param {number} date
- * The date of the point in question
- *
- * @param {number} startOfWeek
- * An integer representing the first day of the week, where 0 is
- * Sunday.
- *
- * @param {Highcharts.Dictionary<string>} dateTimeLabelFormats
- * A map of time units to formats.
- *
- * @return {string}
- * The optimal date format for a point.
- */
- Tooltip.prototype.getDateFormat = function (range, date, startOfWeek, dateTimeLabelFormats) {
- var time = this.chart.time, dateStr = time.dateFormat('%m-%d %H:%M:%S.%L', date), format, n, blank = '01-01 00:00:00.000', strpos = {
- millisecond: 15,
- second: 12,
- minute: 9,
- hour: 6,
- day: 3
- }, lastN = 'millisecond'; // for sub-millisecond data, #4223
- for (n in timeUnits) { // eslint-disable-line guard-for-in
- // If the range is exactly one week and we're looking at a
- // Sunday/Monday, go for the week format
- if (range === timeUnits.week &&
- +time.dateFormat('%w', date) === startOfWeek &&
- dateStr.substr(6) === blank.substr(6)) {
- n = 'week';
- break;
- }
- // The first format that is too great for the range
- if (timeUnits[n] > range) {
- n = lastN;
- break;
- }
- // If the point is placed every day at 23:59, we need to show
- // the minutes as well. #2637.
- if (strpos[n] &&
- dateStr.substr(strpos[n]) !== blank.substr(strpos[n])) {
- break;
- }
- // Weeks are outside the hierarchy, only apply them on
- // Mondays/Sundays like in the first condition
- if (n !== 'week') {
- lastN = n;
- }
- }
- if (n) {
- format = time.resolveDTLFormat(dateTimeLabelFormats[n]).main;
- }
- return format;
- };
- /**
- * Creates the Tooltip label element if it does not exist, then returns it.
- *
- * @function Highcharts.Tooltip#getLabel
- * @return {Highcharts.SVGElement}
- */
- Tooltip.prototype.getLabel = function () {
- var _a,
- _b;
- var tooltip = this,
- renderer = this.chart.renderer,
- styledMode = this.chart.styledMode,
- options = this.options,
- className = ('tooltip' + (defined(options.className) ?
- ' ' + options.className :
- '')),
- pointerEvents = (((_a = options.style) === null || _a === void 0 ? void 0 : _a.pointerEvents) ||
- (!this.followPointer && options.stickOnContact ? 'auto' : 'none')),
- container,
- set,
- onMouseEnter = function () {
- tooltip.inContact = true;
- }, onMouseLeave = function () {
- var series = tooltip.chart.hoverSeries;
- tooltip.inContact = false;
- if (series &&
- series.onMouseOut) {
- series.onMouseOut();
- }
- };
- if (!this.label) {
- if (this.outside) {
- /**
- * Reference to the tooltip's container, when
- * [Highcharts.Tooltip#outside] is set to true, otherwise
- * it's undefined.
- *
- * @name Highcharts.Tooltip#container
- * @type {Highcharts.HTMLDOMElement|undefined}
- */
- this.container = container = H.doc.createElement('div');
- container.className = 'highcharts-tooltip-container';
- css(container, {
- position: 'absolute',
- top: '1px',
- pointerEvents: pointerEvents,
- zIndex: 3
- });
- H.doc.body.appendChild(container);
- /**
- * Reference to the tooltip's renderer, when
- * [Highcharts.Tooltip#outside] is set to true, otherwise
- * it's undefined.
- *
- * @name Highcharts.Tooltip#renderer
- * @type {Highcharts.SVGRenderer|undefined}
- */
- this.renderer = renderer = new H.Renderer(container, 0, 0, (_b = this.chart.options.chart) === null || _b === void 0 ? void 0 : _b.style, void 0, void 0, renderer.styledMode);
- }
- // Create the label
- if (this.split) {
- this.label = renderer.g(className);
- }
- else {
- this.label = renderer
- .label('', 0, 0, options.shape || 'callout', null, null, options.useHTML, null, className)
- .attr({
- padding: options.padding,
- r: options.borderRadius
- });
- if (!styledMode) {
- this.label
- .attr({
- fill: options.backgroundColor,
- 'stroke-width': options.borderWidth
- })
- // #2301, #2657
- .css(options.style)
- .css({ pointerEvents: pointerEvents })
- .shadow(options.shadow);
- }
- }
- if (styledMode) {
- // Apply the drop-shadow filter
- this.applyFilter();
- this.label.addClass('highcharts-tooltip-' + this.chart.index);
- }
- // Split tooltip use updateTooltipContainer to position the tooltip
- // container.
- if (tooltip.outside && !tooltip.split) {
- var label_1 = this.label;
- var xSetter_1 = label_1.xSetter,
- ySetter_1 = label_1.ySetter;
- label_1.xSetter = function (value) {
- xSetter_1.call(label_1, tooltip.distance);
- container.style.left = value + 'px';
- };
- label_1.ySetter = function (value) {
- ySetter_1.call(label_1, tooltip.distance);
- container.style.top = value + 'px';
- };
- }
- this.label
- .on('mouseenter', onMouseEnter)
- .on('mouseleave', onMouseLeave)
- .attr({ zIndex: 8 })
- .add();
- }
- return this.label;
- };
- /**
- * Place the tooltip in a chart without spilling over
- * and not covering the point it self.
- *
- * @private
- * @function Highcharts.Tooltip#getPosition
- *
- * @param {number} boxWidth
- *
- * @param {number} boxHeight
- *
- * @param {Highcharts.Point} point
- *
- * @return {Highcharts.PositionObject}
- */
- Tooltip.prototype.getPosition = function (boxWidth, boxHeight, point) {
- var chart = this.chart,
- distance = this.distance,
- ret = {},
- // Don't use h if chart isn't inverted (#7242) ???
- h = (chart.inverted && point.h) || 0, // #4117 ???
- swapped,
- outside = this.outside,
- outerWidth = outside ?
- // substract distance to prevent scrollbars
- doc.documentElement.clientWidth - 2 * distance :
- chart.chartWidth,
- outerHeight = outside ?
- Math.max(doc.body.scrollHeight,
- doc.documentElement.scrollHeight,
- doc.body.offsetHeight,
- doc.documentElement.offsetHeight,
- doc.documentElement.clientHeight) :
- chart.chartHeight,
- chartPosition = chart.pointer.getChartPosition(),
- containerScaling = chart.containerScaling,
- scaleX = function (val) { return ( // eslint-disable-line no-confusing-arrow
- containerScaling ? val * containerScaling.scaleX : val); },
- scaleY = function (val) { return ( // eslint-disable-line no-confusing-arrow
- containerScaling ? val * containerScaling.scaleY : val); },
- // Build parameter arrays for firstDimension()/secondDimension()
- buildDimensionArray = function (dim) {
- var isX = dim === 'x';
- return [
- dim,
- isX ? outerWidth : outerHeight,
- isX ? boxWidth : boxHeight
- ].concat(outside ? [
- // If we are using tooltip.outside, we need to scale the
- // position to match scaling of the container in case there
- // is a transform/zoom on the container. #11329
- isX ? scaleX(boxWidth) : scaleY(boxHeight),
- isX ? chartPosition.left - distance +
- scaleX(point.plotX + chart.plotLeft) :
- chartPosition.top - distance +
- scaleY(point.plotY + chart.plotTop),
- 0,
- isX ? outerWidth : outerHeight
- ] : [
- // Not outside, no scaling is needed
- isX ? boxWidth : boxHeight,
- isX ? point.plotX + chart.plotLeft :
- point.plotY + chart.plotTop,
- isX ? chart.plotLeft : chart.plotTop,
- isX ? chart.plotLeft + chart.plotWidth :
- chart.plotTop + chart.plotHeight
- ]);
- }, first = buildDimensionArray('y'), second = buildDimensionArray('x'),
- // The far side is right or bottom
- preferFarSide = !this.followPointer && pick(point.ttBelow, !chart.inverted === !!point.negative), // #4984
- /*
- * Handle the preferred dimension. When the preferred dimension is
- * tooltip on top or bottom of the point, it will look for space
- * there.
- *
- * @private
- */
- firstDimension = function (dim, outerSize, innerSize, scaledInnerSize, // #11329
- point, min, max) {
- var scaledDist = dim === 'y' ?
- scaleY(distance) : scaleX(distance),
- scaleDiff = (innerSize - scaledInnerSize) / 2,
- roomLeft = scaledInnerSize < point - distance,
- roomRight = point + distance + scaledInnerSize < outerSize,
- alignedLeft = point - scaledDist - innerSize + scaleDiff,
- alignedRight = point + scaledDist - scaleDiff;
- if (preferFarSide && roomRight) {
- ret[dim] = alignedRight;
- }
- else if (!preferFarSide && roomLeft) {
- ret[dim] = alignedLeft;
- }
- else if (roomLeft) {
- ret[dim] = Math.min(max - scaledInnerSize, alignedLeft - h < 0 ? alignedLeft : alignedLeft - h);
- }
- else if (roomRight) {
- ret[dim] = Math.max(min, alignedRight + h + innerSize > outerSize ?
- alignedRight :
- alignedRight + h);
- }
- else {
- return false;
- }
- },
- /*
- * Handle the secondary dimension. If the preferred dimension is
- * tooltip on top or bottom of the point, the second dimension is to
- * align the tooltip above the point, trying to align center but
- * allowing left or right align within the chart box.
- *
- * @private
- */
- secondDimension = function (dim, outerSize, innerSize, scaledInnerSize, // #11329
- point) {
- var retVal;
- // Too close to the edge, return false and swap dimensions
- if (point < distance || point > outerSize - distance) {
- retVal = false;
- // Align left/top
- }
- else if (point < innerSize / 2) {
- ret[dim] = 1;
- // Align right/bottom
- }
- else if (point > outerSize - scaledInnerSize / 2) {
- ret[dim] = outerSize - scaledInnerSize - 2;
- // Align center
- }
- else {
- ret[dim] = point - innerSize / 2;
- }
- return retVal;
- },
- /*
- * Swap the dimensions
- */
- swap = function (count) {
- var temp = first;
- first = second;
- second = temp;
- swapped = count;
- }, run = function () {
- if (firstDimension.apply(0, first) !== false) {
- if (secondDimension.apply(0, second) === false &&
- !swapped) {
- swap(true);
- run();
- }
- }
- else if (!swapped) {
- swap(true);
- run();
- }
- else {
- ret.x = ret.y = 0;
- }
- };
- // Under these conditions, prefer the tooltip on the side of the point
- if (chart.inverted || this.len > 1) {
- swap();
- }
- run();
- return ret;
- };
- /**
- * Get the best X date format based on the closest point range on the axis.
- *
- * @private
- * @function Highcharts.Tooltip#getXDateFormat
- *
- * @param {Highcharts.Point} point
- *
- * @param {Highcharts.TooltipOptions} options
- *
- * @param {Highcharts.Axis} xAxis
- *
- * @return {string}
- */
- Tooltip.prototype.getXDateFormat = function (point, options, xAxis) {
- var xDateFormat,
- dateTimeLabelFormats = options.dateTimeLabelFormats,
- closestPointRange = xAxis && xAxis.closestPointRange;
- if (closestPointRange) {
- xDateFormat = this.getDateFormat(closestPointRange, point.x, xAxis.options.startOfWeek, dateTimeLabelFormats);
- }
- else {
- xDateFormat = dateTimeLabelFormats.day;
- }
- return xDateFormat || dateTimeLabelFormats.year; // #2546, 2581
- };
- /**
- * Hides the tooltip with a fade out animation.
- *
- * @function Highcharts.Tooltip#hide
- *
- * @param {number} [delay]
- * The fade out in milliseconds. If no value is provided the value
- * of the tooltip.hideDelay option is used. A value of 0 disables
- * the fade out animation.
- */
- Tooltip.prototype.hide = function (delay) {
- var tooltip = this;
- // disallow duplicate timers (#1728, #1766)
- U.clearTimeout(this.hideTimer);
- delay = pick(delay, this.options.hideDelay, 500);
- if (!this.isHidden) {
- this.hideTimer = syncTimeout(function () {
- // If there is a delay, do fadeOut with the default duration. If
- // the hideDelay is 0, we assume no animation is wanted, so we
- // pass 0 duration. #12994.
- tooltip.getLabel().fadeOut(delay ? void 0 : delay);
- tooltip.isHidden = true;
- }, delay);
- }
- };
- /**
- * @private
- * @function Highcharts.Tooltip#init
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.TooltipOptions} options
- * Tooltip options.
- */
- Tooltip.prototype.init = function (chart, options) {
- /**
- * Chart of the tooltip.
- *
- * @readonly
- * @name Highcharts.Tooltip#chart
- * @type {Highcharts.Chart}
- */
- this.chart = chart;
- /**
- * Used tooltip options.
- *
- * @readonly
- * @name Highcharts.Tooltip#options
- * @type {Highcharts.TooltipOptions}
- */
- this.options = options;
- /**
- * List of crosshairs.
- *
- * @private
- * @readonly
- * @name Highcharts.Tooltip#crosshairs
- * @type {Array<null>}
- */
- this.crosshairs = [];
- /**
- * Current values of x and y when animating.
- *
- * @private
- * @readonly
- * @name Highcharts.Tooltip#now
- * @type {Highcharts.PositionObject}
- */
- this.now = { x: 0, y: 0 };
- /**
- * Tooltips are initially hidden.
- *
- * @private
- * @readonly
- * @name Highcharts.Tooltip#isHidden
- * @type {boolean}
- */
- this.isHidden = true;
- /**
- * True, if the tooltip is split into one label per series, with the
- * header close to the axis.
- *
- * @readonly
- * @name Highcharts.Tooltip#split
- * @type {boolean|undefined}
- */
- this.split = options.split && !chart.inverted && !chart.polar;
- /**
- * When the tooltip is shared, the entire plot area will capture mouse
- * movement or touch events.
- *
- * @readonly
- * @name Highcharts.Tooltip#shared
- * @type {boolean|undefined}
- */
- this.shared = options.shared || this.split;
- /**
- * Whether to allow the tooltip to render outside the chart's SVG
- * element box. By default (false), the tooltip is rendered within the
- * chart's SVG element, which results in the tooltip being aligned
- * inside the chart area.
- *
- * @readonly
- * @name Highcharts.Tooltip#outside
- * @type {boolean}
- *
- * @todo
- * Split tooltip does not support outside in the first iteration. Should
- * not be too complicated to implement.
- */
- this.outside = pick(options.outside, Boolean(chart.scrollablePixelsX || chart.scrollablePixelsY));
- };
- /**
- * Returns true, if the pointer is in contact with the tooltip tracker.
- */
- Tooltip.prototype.isStickyOnContact = function () {
- return !!(!this.followPointer &&
- this.options.stickOnContact &&
- this.inContact);
- };
- /**
- * Moves the tooltip with a soft animation to a new position.
- *
- * @private
- * @function Highcharts.Tooltip#move
- *
- * @param {number} x
- *
- * @param {number} y
- *
- * @param {number} anchorX
- *
- * @param {number} anchorY
- */
- Tooltip.prototype.move = function (x, y, anchorX, anchorY) {
- var tooltip = this,
- now = tooltip.now,
- animate = tooltip.options.animation !== false &&
- !tooltip.isHidden &&
- // When we get close to the target position, abort animation and
- // land on the right place (#3056)
- (Math.abs(x - now.x) > 1 || Math.abs(y - now.y) > 1),
- skipAnchor = tooltip.followPointer || tooltip.len > 1;
- // Get intermediate values for animation
- extend(now, {
- x: animate ? (2 * now.x + x) / 3 : x,
- y: animate ? (now.y + y) / 2 : y,
- anchorX: skipAnchor ?
- void 0 :
- animate ? (2 * now.anchorX + anchorX) / 3 : anchorX,
- anchorY: skipAnchor ?
- void 0 :
- animate ? (now.anchorY + anchorY) / 2 : anchorY
- });
- // Move to the intermediate value
- tooltip.getLabel().attr(now);
- tooltip.drawTracker();
- // Run on next tick of the mouse tracker
- if (animate) {
- // Never allow two timeouts
- U.clearTimeout(this.tooltipTimeout);
- // Set the fixed interval ticking for the smooth tooltip
- this.tooltipTimeout = setTimeout(function () {
- // The interval function may still be running during destroy,
- // so check that the chart is really there before calling.
- if (tooltip) {
- tooltip.move(x, y, anchorX, anchorY);
- }
- }, 32);
- }
- };
- /**
- * Refresh the tooltip's text and position.
- *
- * @function Highcharts.Tooltip#refresh
- *
- * @param {Highcharts.Point|Array<Highcharts.Point>} pointOrPoints
- * Either a point or an array of points.
- *
- * @param {Highcharts.PointerEventObject} [mouseEvent]
- * Mouse event, that is responsible for the refresh and should be
- * used for the tooltip update.
- */
- Tooltip.prototype.refresh = function (pointOrPoints, mouseEvent) {
- var tooltip = this,
- chart = this.chart,
- options = tooltip.options,
- x,
- y,
- point = pointOrPoints,
- anchor,
- textConfig = {},
- text,
- pointConfig = [],
- formatter = options.formatter || tooltip.defaultFormatter,
- shared = tooltip.shared,
- currentSeries,
- styledMode = chart.styledMode;
- if (!options.enabled) {
- return;
- }
- U.clearTimeout(this.hideTimer);
- // get the reference point coordinates (pie charts use tooltipPos)
- tooltip.followPointer = splat(point)[0].series.tooltipOptions
- .followPointer;
- anchor = tooltip.getAnchor(point, mouseEvent);
- x = anchor[0];
- y = anchor[1];
- // shared tooltip, array is sent over
- if (shared &&
- !(point.series &&
- point.series.noSharedTooltip)) {
- chart.pointer.applyInactiveState(point);
- // Now set hover state for the choosen ones:
- point.forEach(function (item) {
- item.setState('hover');
- pointConfig.push(item.getLabelConfig());
- });
- textConfig = {
- x: point[0].category,
- y: point[0].y
- };
- textConfig.points = pointConfig;
- point = point[0];
- // single point tooltip
- }
- else {
- textConfig = point.getLabelConfig();
- }
- this.len = pointConfig.length; // #6128
- text = formatter.call(textConfig, tooltip);
- // register the current series
- currentSeries = point.series;
- this.distance = pick(currentSeries.tooltipOptions.distance, 16);
- // update the inner HTML
- if (text === false) {
- this.hide();
- }
- else {
- // update text
- if (tooltip.split) {
- this.renderSplit(text, splat(pointOrPoints));
- }
- else {
- var label = tooltip.getLabel();
- // Prevent the tooltip from flowing over the chart box (#6659)
- if (!options.style.width || styledMode) {
- label.css({
- width: this.chart.spacingBox.width + 'px'
- });
- }
- label.attr({
- text: text && text.join ?
- text.join('') :
- text
- });
- // Set the stroke color of the box to reflect the point
- label.removeClass(/highcharts-color-[\d]+/g)
- .addClass('highcharts-color-' +
- pick(point.colorIndex, currentSeries.colorIndex));
- if (!styledMode) {
- label.attr({
- stroke: (options.borderColor ||
- point.color ||
- currentSeries.color ||
- '#666666')
- });
- }
- tooltip.updatePosition({
- plotX: x,
- plotY: y,
- negative: point.negative,
- ttBelow: point.ttBelow,
- h: anchor[2] || 0
- });
- }
- // show it
- if (tooltip.isHidden && tooltip.label) {
- tooltip.label.attr({
- opacity: 1
- }).show();
- }
- tooltip.isHidden = false;
- }
- fireEvent(this, 'refresh');
- };
- /**
- * Render the split tooltip. Loops over each point's text and adds
- * a label next to the point, then uses the distribute function to
- * find best non-overlapping positions.
- *
- * @private
- * @function Highcharts.Tooltip#renderSplit
- *
- * @param {string|Array<(boolean|string)>} labels
- *
- * @param {Array<Highcharts.Point>} points
- */
- Tooltip.prototype.renderSplit = function (labels, points) {
- var tooltip = this;
- var chart = tooltip.chart,
- _a = tooltip.chart,
- chartWidth = _a.chartWidth,
- chartHeight = _a.chartHeight,
- plotHeight = _a.plotHeight,
- plotLeft = _a.plotLeft,
- plotTop = _a.plotTop,
- pointer = _a.pointer,
- ren = _a.renderer,
- _b = _a.scrollablePixelsY,
- scrollablePixelsY = _b === void 0 ? 0 : _b,
- _c = _a.scrollingContainer,
- _d = _c === void 0 ? { scrollLeft: 0,
- scrollTop: 0 } : _c,
- scrollLeft = _d.scrollLeft,
- scrollTop = _d.scrollTop,
- styledMode = _a.styledMode,
- distance = tooltip.distance,
- options = tooltip.options,
- positioner = tooltip.options.positioner;
- // The area which the tooltip should be limited to. Limit to scrollable
- // plot area if enabled, otherwise limit to the chart container.
- var bounds = {
- left: scrollLeft,
- right: scrollLeft + chartWidth,
- top: scrollTop,
- bottom: scrollTop + chartHeight
- };
- var tooltipLabel = tooltip.getLabel();
- var headerTop = Boolean(chart.xAxis[0] && chart.xAxis[0].opposite);
- var distributionBoxTop = plotTop + scrollTop;
- var headerHeight = 0;
- var adjustedPlotHeight = plotHeight - scrollablePixelsY;
- /**
- * Calculates the anchor position for the partial tooltip
- *
- * @private
- * @param {Highcharts.Point} point The point related to the tooltip
- * @return {object} Returns an object with anchorX and anchorY
- */
- function getAnchor(point) {
- var isHeader = point.isHeader,
- _a = point.plotX,
- plotX = _a === void 0 ? 0 : _a,
- _b = point.plotY,
- plotY = _b === void 0 ? 0 : _b,
- series = point.series;
- var anchorX;
- var anchorY;
- if (isHeader) {
- // Set anchorX to plotX
- anchorX = plotLeft + plotX;
- // Set anchorY to center of visible plot area.
- anchorY = plotTop + plotHeight / 2;
- }
- else {
- var xAxis = series.xAxis,
- yAxis = series.yAxis;
- // Set anchorX to plotX. Limit to within xAxis.
- anchorX = xAxis.pos + clamp(plotX, -distance, xAxis.len + distance);
- // Set anchorY, limit to the scrollable plot area
- if (yAxis.pos + plotY >= scrollTop + plotTop &&
- yAxis.pos + plotY <= scrollTop + plotTop + plotHeight - scrollablePixelsY) {
- anchorY = yAxis.pos + plotY;
- }
- }
- // Limit values to plot area
- anchorX = clamp(anchorX, bounds.left - distance, bounds.right + distance);
- return { anchorX: anchorX, anchorY: anchorY };
- }
- /**
- * Calculates the position of the partial tooltip
- *
- * @private
- * @param {number} anchorX The partial tooltip anchor x position
- * @param {number} anchorY The partial tooltip anchor y position
- * @param {boolean} isHeader Whether the partial tooltip is a header
- * @param {number} boxWidth Width of the partial tooltip
- * @return {Highcharts.PositionObject} Returns the partial tooltip x and
- * y position
- */
- function defaultPositioner(anchorX, anchorY, isHeader, boxWidth, alignedLeft) {
- if (alignedLeft === void 0) { alignedLeft = true; }
- var y;
- var x;
- if (isHeader) {
- y = headerTop ? 0 : adjustedPlotHeight;
- x = clamp(anchorX - (boxWidth / 2), bounds.left, bounds.right - boxWidth);
- }
- else {
- y = anchorY - distributionBoxTop;
- x = alignedLeft ?
- anchorX - boxWidth - distance :
- anchorX + distance;
- x = clamp(x, alignedLeft ? x : bounds.left, bounds.right);
- }
- // NOTE: y is relative to distributionBoxTop
- return { x: x, y: y };
- }
- /**
- * Updates the attributes and styling of the partial tooltip. Creates a
- * new partial tooltip if it does not exists.
- *
- * @private
- * @param {Highcharts.SVGElement|undefined} partialTooltip
- * The partial tooltip to update
- * @param {Highcharts.Point} point
- * The point related to the partial tooltip
- * @param {boolean|string} str The text for the partial tooltip
- * @return {Highcharts.SVGElement} Returns the updated partial tooltip
- */
- function updatePartialTooltip(partialTooltip, point, str) {
- var tt = partialTooltip;
- var isHeader = point.isHeader,
- series = point.series;
- var colorClass = 'highcharts-color-' + pick(point.colorIndex, series.colorIndex, 'none');
- if (!tt) {
- var attribs = {
- padding: options.padding,
- r: options.borderRadius
- };
- if (!styledMode) {
- attribs.fill = options.backgroundColor;
- attribs['stroke-width'] = options.borderWidth;
- }
- tt = ren
- .label('', 0, 0, (options[isHeader ? 'headerShape' : 'shape']) ||
- 'callout', void 0, void 0, options.useHTML)
- .addClass((isHeader ? 'highcharts-tooltip-header ' : '') +
- 'highcharts-tooltip-box ' +
- colorClass)
- .attr(attribs)
- .add(tooltipLabel);
- }
- tt.isActive = true;
- tt.attr({
- text: str
- });
- if (!styledMode) {
- tt.css(options.style)
- .shadow(options.shadow)
- .attr({
- stroke: (options.borderColor ||
- point.color ||
- series.color ||
- '#333333')
- });
- }
- return tt;
- }
- // Graceful degradation for legacy formatters
- if (isString(labels)) {
- labels = [false, labels];
- }
- // Create the individual labels for header and points, ignore footer
- var boxes = labels.slice(0,
- points.length + 1).reduce(function (boxes,
- str,
- i) {
- if (str !== false && str !== '') {
- var point = (points[i - 1] ||
- {
- // Item 0 is the header. Instead of this, we could also
- // use the crosshair label
- isHeader: true,
- plotX: points[0].plotX,
- plotY: plotHeight,
- series: {}
- });
- var isHeader = point.isHeader;
- // Store the tooltip label referance on the series
- var owner = isHeader ? tooltip : point.series;
- var tt = owner.tt = updatePartialTooltip(owner.tt,
- point,
- str);
- // Get X position now, so we can move all to the other side in
- // case of overflow
- var bBox = tt.getBBox();
- var boxWidth = bBox.width + tt.strokeWidth();
- if (isHeader) {
- headerHeight = bBox.height;
- adjustedPlotHeight += headerHeight;
- if (headerTop) {
- distributionBoxTop -= headerHeight;
- }
- }
- var _a = getAnchor(point),
- anchorX = _a.anchorX,
- anchorY = _a.anchorY;
- if (typeof anchorY === 'number') {
- var size = bBox.height + 1;
- var boxPosition = (positioner ?
- positioner.call(tooltip,
- boxWidth,
- size,
- point) :
- defaultPositioner(anchorX,
- anchorY,
- isHeader,
- boxWidth));
- boxes.push({
- // 0-align to the top, 1-align to the bottom
- align: positioner ? 0 : void 0,
- anchorX: anchorX,
- anchorY: anchorY,
- boxWidth: boxWidth,
- point: point,
- rank: pick(boxPosition.rank, isHeader ? 1 : 0),
- size: size,
- target: boxPosition.y,
- tt: tt,
- x: boxPosition.x
- });
- }
- else {
- // Hide tooltips which anchorY is outside the visible plot
- // area
- tt.isActive = false;
- }
- }
- return boxes;
- }, []);
- // If overflow left then align all labels to the right
- if (!positioner && boxes.some(function (box) { return box.x < bounds.left; })) {
- boxes = boxes.map(function (box) {
- var _a = defaultPositioner(box.anchorX,
- box.anchorY,
- box.point.isHeader,
- box.boxWidth,
- false),
- x = _a.x,
- y = _a.y;
- return extend(box, {
- target: y,
- x: x
- });
- });
- }
- // Clean previous run (for missing points)
- tooltip.cleanSplit();
- // Distribute and put in place
- H.distribute(boxes, adjustedPlotHeight);
- boxes.forEach(function (box) {
- var anchorX = box.anchorX,
- anchorY = box.anchorY,
- pos = box.pos,
- x = box.x;
- // Put the label in place
- box.tt.attr({
- visibility: typeof pos === 'undefined' ? 'hidden' : 'inherit',
- x: x,
- /* NOTE: y should equal pos to be consistent with !split
- * tooltip, but is currently relative to plotTop. Is left as is
- * to avoid breaking change. Remove distributionBoxTop to make
- * it consistent.
- */
- y: pos + distributionBoxTop,
- anchorX: anchorX,
- anchorY: anchorY
- });
- });
- /* If we have a seperate tooltip container, then update the necessary
- * container properties.
- * Test that tooltip has its own container and renderer before executing
- * the operation.
- */
- var container = tooltip.container,
- outside = tooltip.outside,
- renderer = tooltip.renderer;
- if (outside && container && renderer) {
- // Set container size to fit the tooltip
- var _e = tooltipLabel.getBBox(),
- width = _e.width,
- height = _e.height,
- x = _e.x,
- y = _e.y;
- renderer.setSize(width + x, height + y, false);
- // Position the tooltip container to the chart container
- var chartPosition = pointer.getChartPosition();
- container.style.left = chartPosition.left + 'px';
- container.style.top = chartPosition.top + 'px';
- }
- };
- /**
- * If the `stickOnContact` option is active, this will add a tracker shape.
- *
- * @private
- * @function Highcharts.Tooltip#drawTracker
- */
- Tooltip.prototype.drawTracker = function () {
- var tooltip = this;
- if (tooltip.followPointer ||
- !tooltip.options.stickOnContact) {
- if (tooltip.tracker) {
- tooltip.tracker.destroy();
- }
- return;
- }
- var chart = tooltip.chart;
- var label = tooltip.label;
- var point = chart.hoverPoint;
- if (!label || !point) {
- return;
- }
- var box = {
- x: 0,
- y: 0,
- width: 0,
- height: 0
- };
- // Combine anchor and tooltip
- var anchorPos = this.getAnchor(point);
- var labelBBox = label.getBBox();
- anchorPos[0] += chart.plotLeft - label.translateX;
- anchorPos[1] += chart.plotTop - label.translateY;
- // When the mouse pointer is between the anchor point and the label,
- // the label should stick.
- box.x = Math.min(0, anchorPos[0]);
- box.y = Math.min(0, anchorPos[1]);
- box.width = (anchorPos[0] < 0 ?
- Math.max(Math.abs(anchorPos[0]), (labelBBox.width - anchorPos[0])) :
- Math.max(Math.abs(anchorPos[0]), labelBBox.width));
- box.height = (anchorPos[1] < 0 ?
- Math.max(Math.abs(anchorPos[1]), (labelBBox.height - Math.abs(anchorPos[1]))) :
- Math.max(Math.abs(anchorPos[1]), labelBBox.height));
- if (tooltip.tracker) {
- tooltip.tracker.attr(box);
- }
- else {
- tooltip.tracker = label.renderer
- .rect(box)
- .addClass('highcharts-tracker')
- .add(label);
- if (!chart.styledMode) {
- tooltip.tracker.attr({
- fill: 'rgba(0,0,0,0)'
- });
- }
- }
- };
- /**
- * @private
- */
- Tooltip.prototype.styledModeFormat = function (formatString) {
- return formatString
- .replace('style="font-size: 10px"', 'class="highcharts-header"')
- .replace(/style="color:{(point|series)\.color}"/g, 'class="highcharts-color-{$1.colorIndex}"');
- };
- /**
- * Format the footer/header of the tooltip
- * #3397: abstraction to enable formatting of footer and header
- *
- * @private
- * @function Highcharts.Tooltip#tooltipFooterHeaderFormatter
- * @param {Highcharts.PointLabelObject} labelConfig
- * @param {boolean} [isFooter]
- * @return {string}
- */
- Tooltip.prototype.tooltipFooterHeaderFormatter = function (labelConfig, isFooter) {
- var footOrHead = isFooter ? 'footer' : 'header',
- series = labelConfig.series,
- tooltipOptions = series.tooltipOptions,
- xDateFormat = tooltipOptions.xDateFormat,
- xAxis = series.xAxis,
- isDateTime = (xAxis &&
- xAxis.options.type === 'datetime' &&
- isNumber(labelConfig.key)),
- formatString = tooltipOptions[footOrHead + 'Format'],
- e = {
- isFooter: isFooter,
- labelConfig: labelConfig
- };
- fireEvent(this, 'headerFormatter', e, function (e) {
- // Guess the best date format based on the closest point distance
- // (#568, #3418)
- if (isDateTime && !xDateFormat) {
- xDateFormat = this.getXDateFormat(labelConfig, tooltipOptions, xAxis);
- }
- // Insert the footer date format if any
- if (isDateTime && xDateFormat) {
- ((labelConfig.point && labelConfig.point.tooltipDateKeys) ||
- ['key']).forEach(function (key) {
- formatString = formatString.replace('{point.' + key + '}', '{point.' + key + ':' + xDateFormat + '}');
- });
- }
- // Replace default header style with class name
- if (series.chart.styledMode) {
- formatString = this.styledModeFormat(formatString);
- }
- e.text = format(formatString, {
- point: labelConfig,
- series: series
- }, this.chart);
- });
- return e.text;
- };
- /**
- * Updates the tooltip with the provided tooltip options.
- *
- * @function Highcharts.Tooltip#update
- *
- * @param {Highcharts.TooltipOptions} options
- * The tooltip options to update.
- */
- Tooltip.prototype.update = function (options) {
- this.destroy();
- // Update user options (#6218)
- merge(true, this.chart.options.tooltip.userOptions, options);
- this.init(this.chart, merge(true, this.options, options));
- };
- /**
- * Find the new position and perform the move
- *
- * @private
- * @function Highcharts.Tooltip#updatePosition
- *
- * @param {Highcharts.Point} point
- */
- Tooltip.prototype.updatePosition = function (point) {
- var chart = this.chart,
- pointer = chart.pointer,
- label = this.getLabel(),
- pos,
- anchorX = point.plotX + chart.plotLeft,
- anchorY = point.plotY + chart.plotTop,
- pad;
- // Needed for outside: true (#11688)
- var chartPosition = pointer.getChartPosition();
- pos = (this.options.positioner || this.getPosition).call(this, label.width, label.height, point);
- // Set the renderer size dynamically to prevent document size to change
- if (this.outside) {
- pad = (this.options.borderWidth || 0) + 2 * this.distance;
- this.renderer.setSize(label.width + pad, label.height + pad, false);
- // Anchor and tooltip container need scaling if chart container has
- // scale transform/css zoom. #11329.
- var containerScaling = chart.containerScaling;
- if (containerScaling) {
- css(this.container, {
- transform: "scale(" + containerScaling.scaleX + ", " + containerScaling.scaleY + ")"
- });
- anchorX *= containerScaling.scaleX;
- anchorY *= containerScaling.scaleY;
- }
- anchorX += chartPosition.left - pos.x;
- anchorY += chartPosition.top - pos.y;
- }
- // do the move
- this.move(Math.round(pos.x), Math.round(pos.y || 0), // can be undefined (#3977)
- anchorX, anchorY);
- };
- return Tooltip;
- }());
- H.Tooltip = Tooltip;
- return H.Tooltip;
- });
- _registerModule(_modules, 'Core/Pointer.js', [_modules['Core/Color.js'], _modules['Core/Globals.js'], _modules['Core/Tooltip.js'], _modules['Core/Utilities.js']], function (Color, H, Tooltip, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var color = Color.parse;
- var charts = H.charts,
- noop = H.noop;
- var addEvent = U.addEvent,
- attr = U.attr,
- css = U.css,
- defined = U.defined,
- extend = U.extend,
- find = U.find,
- fireEvent = U.fireEvent,
- isNumber = U.isNumber,
- isObject = U.isObject,
- objectEach = U.objectEach,
- offset = U.offset,
- pick = U.pick,
- splat = U.splat;
- /**
- * One position in relation to an axis.
- *
- * @interface Highcharts.PointerAxisCoordinateObject
- */ /**
- * Related axis.
- *
- * @name Highcharts.PointerAxisCoordinateObject#axis
- * @type {Highcharts.Axis}
- */ /**
- * Axis value.
- *
- * @name Highcharts.PointerAxisCoordinateObject#value
- * @type {number}
- */
- /**
- * Positions in terms of axis values.
- *
- * @interface Highcharts.PointerAxisCoordinatesObject
- */ /**
- * Positions on the x-axis.
- * @name Highcharts.PointerAxisCoordinatesObject#xAxis
- * @type {Array<Highcharts.PointerAxisCoordinateObject>}
- */ /**
- * Positions on the y-axis.
- * @name Highcharts.PointerAxisCoordinatesObject#yAxis
- * @type {Array<Highcharts.PointerAxisCoordinateObject>}
- */
- /**
- * Pointer coordinates.
- *
- * @interface Highcharts.PointerCoordinatesObject
- */ /**
- * @name Highcharts.PointerCoordinatesObject#chartX
- * @type {number}
- */ /**
- * @name Highcharts.PointerCoordinatesObject#chartY
- * @type {number}
- */
- /**
- * A native browser mouse or touch event, extended with position information
- * relative to the {@link Chart.container}.
- *
- * @interface Highcharts.PointerEventObject
- * @extends global.PointerEvent
- */ /**
- * The X coordinate of the pointer interaction relative to the chart.
- *
- * @name Highcharts.PointerEventObject#chartX
- * @type {number}
- */ /**
- * The Y coordinate of the pointer interaction relative to the chart.
- *
- * @name Highcharts.PointerEventObject#chartY
- * @type {number}
- */
- /**
- * Axis-specific data of a selection.
- *
- * @interface Highcharts.SelectDataObject
- */ /**
- * @name Highcharts.SelectDataObject#axis
- * @type {Highcharts.Axis}
- */ /**
- * @name Highcharts.SelectDataObject#max
- * @type {number}
- */ /**
- * @name Highcharts.SelectDataObject#min
- * @type {number}
- */
- /**
- * Object for select events.
- *
- * @interface Highcharts.SelectEventObject
- */ /**
- * @name Highcharts.SelectEventObject#originalEvent
- * @type {global.Event}
- */ /**
- * @name Highcharts.SelectEventObject#xAxis
- * @type {Array<Highcharts.SelectDataObject>}
- */ /**
- * @name Highcharts.SelectEventObject#yAxis
- * @type {Array<Highcharts.SelectDataObject>}
- */
- ''; // detach doclets above
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The mouse and touch tracker object. Each {@link Chart} item has one
- * assosiated Pointer item that can be accessed from the {@link Chart.pointer}
- * property.
- *
- * @class
- * @name Highcharts.Pointer
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.Options} options
- * The root options object. The pointer uses options from the chart and
- * tooltip structures.
- */
- var Pointer = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Pointer(chart, options) {
- this.lastValidTouch = {};
- this.pinchDown = [];
- this.runChartClick = false;
- this.chart = chart;
- this.hasDragged = false;
- this.options = options;
- this.unbindContainerMouseLeave = function () { };
- this.unbindContainerMouseEnter = function () { };
- this.init(chart, options);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Set inactive state to all series that are not currently hovered,
- * or, if `inactiveOtherPoints` is set to true, set inactive state to
- * all points within that series.
- *
- * @private
- * @function Highcharts.Pointer#applyInactiveState
- * @param {Array<Highcharts.Point>} points
- * Currently hovered points
- */
- Pointer.prototype.applyInactiveState = function (points) {
- var activeSeries = [],
- series;
- // Get all active series from the hovered points
- (points || []).forEach(function (item) {
- series = item.series;
- // Include itself
- activeSeries.push(series);
- // Include parent series
- if (series.linkedParent) {
- activeSeries.push(series.linkedParent);
- }
- // Include all child series
- if (series.linkedSeries) {
- activeSeries = activeSeries.concat(series.linkedSeries);
- }
- // Include navigator series
- if (series.navigatorSeries) {
- activeSeries.push(series.navigatorSeries);
- }
- });
- // Now loop over all series, filtering out active series
- this.chart.series.forEach(function (inactiveSeries) {
- if (activeSeries.indexOf(inactiveSeries) === -1) {
- // Inactive series
- inactiveSeries.setState('inactive', true);
- }
- else if (inactiveSeries.options.inactiveOtherPoints) {
- // Active series, but other points should be inactivated
- inactiveSeries.setAllPointsToState('inactive');
- }
- });
- };
- /**
- * Destroys the Pointer object and disconnects DOM events.
- *
- * @function Highcharts.Pointer#destroy
- */
- Pointer.prototype.destroy = function () {
- var pointer = this;
- if (typeof pointer.unDocMouseMove !== 'undefined') {
- pointer.unDocMouseMove();
- }
- this.unbindContainerMouseLeave();
- if (!H.chartCount) {
- if (H.unbindDocumentMouseUp) {
- H.unbindDocumentMouseUp = H.unbindDocumentMouseUp();
- }
- if (H.unbindDocumentTouchEnd) {
- H.unbindDocumentTouchEnd = H.unbindDocumentTouchEnd();
- }
- }
- // memory and CPU leak
- clearInterval(pointer.tooltipTimeout);
- objectEach(pointer, function (_val, prop) {
- pointer[prop] = void 0;
- });
- };
- /**
- * Perform a drag operation in response to a mousemove event while the mouse
- * is down.
- *
- * @private
- * @function Highcharts.Pointer#drag
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.drag = function (e) {
- var chart = this.chart,
- chartOptions = chart.options.chart,
- chartX = e.chartX,
- chartY = e.chartY,
- zoomHor = this.zoomHor,
- zoomVert = this.zoomVert,
- plotLeft = chart.plotLeft,
- plotTop = chart.plotTop,
- plotWidth = chart.plotWidth,
- plotHeight = chart.plotHeight,
- clickedInside,
- size,
- selectionMarker = this.selectionMarker,
- mouseDownX = (this.mouseDownX || 0),
- mouseDownY = (this.mouseDownY || 0),
- panningEnabled = isObject(chartOptions.panning) ?
- chartOptions.panning && chartOptions.panning.enabled :
- chartOptions.panning,
- panKey = (chartOptions.panKey && e[chartOptions.panKey + 'Key']);
- // If the device supports both touch and mouse (like IE11), and we are
- // touch-dragging inside the plot area, don't handle the mouse event.
- // #4339.
- if (selectionMarker && selectionMarker.touch) {
- return;
- }
- // If the mouse is outside the plot area, adjust to cooordinates
- // inside to prevent the selection marker from going outside
- if (chartX < plotLeft) {
- chartX = plotLeft;
- }
- else if (chartX > plotLeft + plotWidth) {
- chartX = plotLeft + plotWidth;
- }
- if (chartY < plotTop) {
- chartY = plotTop;
- }
- else if (chartY > plotTop + plotHeight) {
- chartY = plotTop + plotHeight;
- }
- // determine if the mouse has moved more than 10px
- this.hasDragged = Math.sqrt(Math.pow(mouseDownX - chartX, 2) +
- Math.pow(mouseDownY - chartY, 2));
- if (this.hasDragged > 10) {
- clickedInside = chart.isInsidePlot(mouseDownX - plotLeft, mouseDownY - plotTop);
- // make a selection
- if (chart.hasCartesianSeries &&
- (this.zoomX || this.zoomY) &&
- clickedInside &&
- !panKey) {
- if (!selectionMarker) {
- this.selectionMarker = selectionMarker =
- chart.renderer.rect(plotLeft, plotTop, zoomHor ? 1 : plotWidth, zoomVert ? 1 : plotHeight, 0)
- .attr({
- 'class': 'highcharts-selection-marker',
- zIndex: 7
- })
- .add();
- if (!chart.styledMode) {
- selectionMarker.attr({
- fill: (chartOptions.selectionMarkerFill ||
- color('#335cad')
- .setOpacity(0.25).get())
- });
- }
- }
- }
- // adjust the width of the selection marker
- if (selectionMarker && zoomHor) {
- size = chartX - mouseDownX;
- selectionMarker.attr({
- width: Math.abs(size),
- x: (size > 0 ? 0 : size) + mouseDownX
- });
- }
- // adjust the height of the selection marker
- if (selectionMarker && zoomVert) {
- size = chartY - mouseDownY;
- selectionMarker.attr({
- height: Math.abs(size),
- y: (size > 0 ? 0 : size) + mouseDownY
- });
- }
- // panning
- if (clickedInside &&
- !selectionMarker &&
- panningEnabled) {
- chart.pan(e, chartOptions.panning);
- }
- }
- };
- /**
- * Start a drag operation.
- *
- * @private
- * @function Highcharts.Pointer#dragStart
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.dragStart = function (e) {
- var chart = this.chart;
- // Record the start position
- chart.mouseIsDown = e.type;
- chart.cancelClick = false;
- chart.mouseDownX = this.mouseDownX = e.chartX;
- chart.mouseDownY = this.mouseDownY = e.chartY;
- };
- /**
- * On mouse up or touch end across the entire document, drop the selection.
- *
- * @private
- * @function Highcharts.Pointer#drop
- *
- * @param {global.Event} e
- */
- Pointer.prototype.drop = function (e) {
- var pointer = this,
- chart = this.chart,
- hasPinched = this.hasPinched;
- if (this.selectionMarker) {
- var selectionData = {
- originalEvent: e,
- xAxis: [],
- yAxis: []
- },
- selectionBox = this.selectionMarker,
- selectionLeft = selectionBox.attr ?
- selectionBox.attr('x') :
- selectionBox.x,
- selectionTop = selectionBox.attr ?
- selectionBox.attr('y') :
- selectionBox.y,
- selectionWidth = selectionBox.attr ?
- selectionBox.attr('width') :
- selectionBox.width,
- selectionHeight = selectionBox.attr ?
- selectionBox.attr('height') :
- selectionBox.height,
- runZoom;
- // a selection has been made
- if (this.hasDragged || hasPinched) {
- // record each axis' min and max
- chart.axes.forEach(function (axis) {
- if (axis.zoomEnabled &&
- defined(axis.min) &&
- (hasPinched ||
- pointer[{
- xAxis: 'zoomX',
- yAxis: 'zoomY'
- }[axis.coll]]) &&
- isNumber(selectionLeft) &&
- isNumber(selectionTop)) { // #859, #3569
- var horiz = axis.horiz,
- minPixelPadding = e.type === 'touchend' ?
- axis.minPixelPadding :
- 0, // #1207, #3075
- selectionMin = axis.toValue((horiz ? selectionLeft : selectionTop) +
- minPixelPadding),
- selectionMax = axis.toValue((horiz ?
- selectionLeft + selectionWidth :
- selectionTop + selectionHeight) - minPixelPadding);
- selectionData[axis.coll].push({
- axis: axis,
- // Min/max for reversed axes
- min: Math.min(selectionMin, selectionMax),
- max: Math.max(selectionMin, selectionMax)
- });
- runZoom = true;
- }
- });
- if (runZoom) {
- fireEvent(chart, 'selection', selectionData, function (args) {
- chart.zoom(extend(args, hasPinched ?
- { animation: false } :
- null));
- });
- }
- }
- if (isNumber(chart.index)) {
- this.selectionMarker = this.selectionMarker.destroy();
- }
- // Reset scaling preview
- if (hasPinched) {
- this.scaleGroups();
- }
- }
- // Reset all. Check isNumber because it may be destroyed on mouse up
- // (#877)
- if (chart && isNumber(chart.index)) {
- css(chart.container, { cursor: chart._cursor });
- chart.cancelClick = this.hasDragged > 10; // #370
- chart.mouseIsDown = this.hasDragged = this.hasPinched = false;
- this.pinchDown = [];
- }
- };
- /**
- * Finds the closest point to a set of coordinates, using the k-d-tree
- * algorithm.
- *
- * @function Highcharts.Pointer#findNearestKDPoint
- *
- * @param {Array<Highcharts.Series>} series
- * All the series to search in.
- *
- * @param {boolean|undefined} shared
- * Whether it is a shared tooltip or not.
- *
- * @param {Highcharts.PointerEventObject} e
- * The pointer event object, containing chart coordinates of the
- * pointer.
- *
- * @return {Highcharts.Point|undefined}
- * The point closest to given coordinates.
- */
- Pointer.prototype.findNearestKDPoint = function (series, shared, e) {
- var chart = this.chart;
- var hoverPoint = chart.hoverPoint;
- var tooltip = chart.tooltip;
- if (hoverPoint &&
- tooltip &&
- tooltip.isStickyOnContact()) {
- return hoverPoint;
- }
- var closest;
- /** @private */
- function sort(p1, p2) {
- var isCloserX = p1.distX - p2.distX,
- isCloser = p1.dist - p2.dist,
- isAbove = (p2.series.group && p2.series.group.zIndex) -
- (p1.series.group && p1.series.group.zIndex),
- result;
- // We have two points which are not in the same place on xAxis
- // and shared tooltip:
- if (isCloserX !== 0 && shared) { // #5721
- result = isCloserX;
- // Points are not exactly in the same place on x/yAxis:
- }
- else if (isCloser !== 0) {
- result = isCloser;
- // The same xAxis and yAxis position, sort by z-index:
- }
- else if (isAbove !== 0) {
- result = isAbove;
- // The same zIndex, sort by array index:
- }
- else {
- result =
- p1.series.index > p2.series.index ?
- -1 :
- 1;
- }
- return result;
- }
- series.forEach(function (s) {
- var noSharedTooltip = s.noSharedTooltip && shared,
- compareX = (!noSharedTooltip &&
- s.options.findNearestPointBy.indexOf('y') < 0),
- point = s.searchPoint(e,
- compareX);
- if ( // Check that we actually found a point on the series.
- isObject(point, true) &&
- // Use the new point if it is closer.
- (!isObject(closest, true) ||
- (sort(closest, point) > 0))) {
- closest = point;
- }
- });
- return closest;
- };
- /**
- * @private
- * @function Highcharts.Pointer#getChartCoordinatesFromPoint
- * @param {Highcharts.Point} point
- * @param {boolean} [inverted]
- * @return {Highcharts.PointerCoordinatesObject|undefined}
- */
- Pointer.prototype.getChartCoordinatesFromPoint = function (point, inverted) {
- var series = point.series,
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- plotX = pick(point.clientX,
- point.plotX),
- shapeArgs = point.shapeArgs;
- if (xAxis && yAxis) {
- return inverted ? {
- chartX: xAxis.len + xAxis.pos - plotX,
- chartY: yAxis.len + yAxis.pos - point.plotY
- } : {
- chartX: plotX + xAxis.pos,
- chartY: point.plotY + yAxis.pos
- };
- }
- if (shapeArgs && shapeArgs.x && shapeArgs.y) {
- // E.g. pies do not have axes
- return {
- chartX: shapeArgs.x,
- chartY: shapeArgs.y
- };
- }
- };
- /**
- * Return the cached chartPosition if it is available on the Pointer,
- * otherwise find it. Running offset is quite expensive, so it should be
- * avoided when we know the chart hasn't moved.
- *
- * @function Highcharts.Pointer#getChartPosition
- *
- * @return {Highcharts.OffsetObject}
- * The offset of the chart container within the page
- */
- Pointer.prototype.getChartPosition = function () {
- return (this.chartPosition ||
- (this.chartPosition = offset(this.chart.container)));
- };
- /**
- * Get the click position in terms of axis values.
- *
- * @function Highcharts.Pointer#getCoordinates
- *
- * @param {Highcharts.PointerEventObject} e
- * Pointer event, extended with `chartX` and `chartY` properties.
- *
- * @return {Highcharts.PointerAxisCoordinatesObject}
- */
- Pointer.prototype.getCoordinates = function (e) {
- var coordinates = {
- xAxis: [],
- yAxis: []
- };
- this.chart.axes.forEach(function (axis) {
- coordinates[axis.isXAxis ? 'xAxis' : 'yAxis'].push({
- axis: axis,
- value: axis.toValue(e[axis.horiz ? 'chartX' : 'chartY'])
- });
- });
- return coordinates;
- };
- /**
- * Calculates what is the current hovered point/points and series.
- *
- * @private
- * @function Highcharts.Pointer#getHoverData
- *
- * @param {Highcharts.Point|undefined} existingHoverPoint
- * The point currrently beeing hovered.
- *
- * @param {Highcharts.Series|undefined} existingHoverSeries
- * The series currently beeing hovered.
- *
- * @param {Array<Highcharts.Series>} series
- * All the series in the chart.
- *
- * @param {boolean} isDirectTouch
- * Is the pointer directly hovering the point.
- *
- * @param {boolean|undefined} shared
- * Whether it is a shared tooltip or not.
- *
- * @param {Highcharts.PointerEventObject} [e]
- * The triggering event, containing chart coordinates of the pointer.
- *
- * @return {object}
- * Object containing resulting hover data: hoverPoint, hoverSeries,
- * and hoverPoints.
- */
- Pointer.prototype.getHoverData = function (existingHoverPoint, existingHoverSeries, series, isDirectTouch, shared, e) {
- var hoverPoint,
- hoverPoints = [],
- hoverSeries = existingHoverSeries,
- useExisting = !!(isDirectTouch && existingHoverPoint),
- notSticky = hoverSeries && !hoverSeries.stickyTracking,
- // Which series to look in for the hover point
- searchSeries,
- // Parameters needed for beforeGetHoverData event.
- eventArgs = {
- chartX: e ? e.chartX : void 0,
- chartY: e ? e.chartY : void 0,
- shared: shared
- },
- filter = function (s) {
- return (s.visible &&
- !(!shared && s.directTouch) && // #3821
- pick(s.options.enableMouseTracking,
- true));
- };
- // Find chart.hoverPane and update filter method in polar.
- fireEvent(this, 'beforeGetHoverData', eventArgs);
- searchSeries = notSticky ?
- // Only search on hovered series if it has stickyTracking false
- [hoverSeries] :
- // Filter what series to look in.
- series.filter(function (s) {
- return eventArgs.filter ? eventArgs.filter(s) : filter(s) &&
- s.stickyTracking;
- });
- // Use existing hovered point or find the one closest to coordinates.
- hoverPoint = useExisting || !e ?
- existingHoverPoint :
- this.findNearestKDPoint(searchSeries, shared, e);
- // Assign hover series
- hoverSeries = hoverPoint && hoverPoint.series;
- // If we have a hoverPoint, assign hoverPoints.
- if (hoverPoint) {
- // When tooltip is shared, it displays more than one point
- if (shared && !hoverSeries.noSharedTooltip) {
- searchSeries = series.filter(function (s) {
- return eventArgs.filter ?
- eventArgs.filter(s) : filter(s) && !s.noSharedTooltip;
- });
- // Get all points with the same x value as the hoverPoint
- searchSeries.forEach(function (s) {
- var point = find(s.points,
- function (p) {
- return p.x === hoverPoint.x && !p.isNull;
- });
- if (isObject(point)) {
- /*
- * Boost returns a minimal point. Convert it to a usable
- * point for tooltip and states.
- */
- if (s.chart.isBoosting) {
- point = s.getPoint(point);
- }
- hoverPoints.push(point);
- }
- });
- }
- else {
- hoverPoints.push(hoverPoint);
- }
- }
- // Check whether the hoverPoint is inside pane we are hovering over.
- eventArgs = { hoverPoint: hoverPoint };
- fireEvent(this, 'afterGetHoverData', eventArgs);
- return {
- hoverPoint: eventArgs.hoverPoint,
- hoverSeries: hoverSeries,
- hoverPoints: hoverPoints
- };
- };
- /**
- * @private
- * @function Highcharts.Pointer#getPointFromEvent
- *
- * @param {global.Event} e
- *
- * @return {Highcharts.Point|undefined}
- */
- Pointer.prototype.getPointFromEvent = function (e) {
- var target = e.target,
- point;
- while (target && !point) {
- point = target.point;
- target = target.parentNode;
- }
- return point;
- };
- /**
- * @private
- * @function Highcharts.Pointer#onTrackerMouseOut
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.onTrackerMouseOut = function (e) {
- var chart = this.chart;
- var relatedTarget = e.relatedTarget || e.toElement;
- var series = chart.hoverSeries;
- this.isDirectTouch = false;
- if (series &&
- relatedTarget &&
- !series.stickyTracking &&
- !this.inClass(relatedTarget, 'highcharts-tooltip') &&
- (!this.inClass(relatedTarget, 'highcharts-series-' + series.index) || // #2499, #4465, #5553
- !this.inClass(relatedTarget, 'highcharts-tracker'))) {
- series.onMouseOut();
- }
- };
- /**
- * Utility to detect whether an element has, or has a parent with, a
- * specificclass name. Used on detection of tracker objects and on deciding
- * whether hovering the tooltip should cause the active series to mouse out.
- *
- * @function Highcharts.Pointer#inClass
- *
- * @param {Highcharts.SVGDOMElement|Highcharts.HTMLDOMElement} element
- * The element to investigate.
- *
- * @param {string} className
- * The class name to look for.
- *
- * @return {boolean|undefined}
- * True if either the element or one of its parents has the given
- * class name.
- */
- Pointer.prototype.inClass = function (element, className) {
- var elemClassName;
- while (element) {
- elemClassName = attr(element, 'class');
- if (elemClassName) {
- if (elemClassName.indexOf(className) !== -1) {
- return true;
- }
- if (elemClassName.indexOf('highcharts-container') !== -1) {
- return false;
- }
- }
- element = element.parentNode;
- }
- };
- /**
- * Initialize the Pointer.
- *
- * @private
- * @function Highcharts.Pointer#init
- *
- * @param {Highcharts.Chart} chart
- * The Chart instance.
- *
- * @param {Highcharts.Options} options
- * The root options object. The pointer uses options from the chart
- * and tooltip structures.
- *
- * @return {void}
- */
- Pointer.prototype.init = function (chart, options) {
- // Store references
- this.options = options;
- this.chart = chart;
- // Do we need to handle click on a touch device?
- this.runChartClick =
- options.chart.events &&
- !!options.chart.events.click;
- this.pinchDown = [];
- this.lastValidTouch = {};
- if (Tooltip) {
- /**
- * Tooltip object for points of series.
- *
- * @name Highcharts.Chart#tooltip
- * @type {Highcharts.Tooltip}
- */
- chart.tooltip = new Tooltip(chart, options.tooltip);
- this.followTouchMove = pick(options.tooltip.followTouchMove, true);
- }
- this.setDOMEvents();
- };
- /**
- * Takes a browser event object and extends it with custom Highcharts
- * properties `chartX` and `chartY` in order to work on the internal
- * coordinate system.
- *
- * @function Highcharts.Pointer#normalize
- *
- * @param {global.MouseEvent|global.PointerEvent|global.TouchEvent} e
- * Event object in standard browsers.
- *
- * @param {Highcharts.OffsetObject} [chartPosition]
- * Additional chart offset.
- *
- * @return {Highcharts.PointerEventObject}
- * A browser event with extended properties `chartX` and `chartY`.
- */
- Pointer.prototype.normalize = function (e, chartPosition) {
- var touches = e.touches;
- // iOS (#2757)
- var ePos = (touches ?
- touches.length ?
- touches.item(0) :
- (pick(// #13534
- touches.changedTouches,
- e.changedTouches))[0] :
- e);
- // Get mouse position
- if (!chartPosition) {
- chartPosition = this.getChartPosition();
- }
- var chartX = ePos.pageX - chartPosition.left,
- chartY = ePos.pageY - chartPosition.top;
- // #11329 - when there is scaling on a parent element, we need to take
- // this into account
- var containerScaling = this.chart.containerScaling;
- if (containerScaling) {
- chartX /= containerScaling.scaleX;
- chartY /= containerScaling.scaleY;
- }
- return extend(e, {
- chartX: Math.round(chartX),
- chartY: Math.round(chartY)
- });
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerClick
- */
- Pointer.prototype.onContainerClick = function (e) {
- var chart = this.chart;
- var hoverPoint = chart.hoverPoint;
- var pEvt = this.normalize(e);
- var plotLeft = chart.plotLeft;
- var plotTop = chart.plotTop;
- if (!chart.cancelClick) {
- // On tracker click, fire the series and point events. #783, #1583
- if (hoverPoint &&
- this.inClass(pEvt.target, 'highcharts-tracker')) {
- // the series click event
- fireEvent(hoverPoint.series, 'click', extend(pEvt, {
- point: hoverPoint
- }));
- // the point click event
- if (chart.hoverPoint) { // it may be destroyed (#1844)
- hoverPoint.firePointEvent('click', pEvt);
- }
- // When clicking outside a tracker, fire a chart event
- }
- else {
- extend(pEvt, this.getCoordinates(pEvt));
- // fire a click event in the chart
- if (chart.isInsidePlot((pEvt.chartX - plotLeft), (pEvt.chartY - plotTop))) {
- fireEvent(chart, 'click', pEvt);
- }
- }
- }
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerMouseDown
- *
- * @param {global.MouseEvent} e
- */
- Pointer.prototype.onContainerMouseDown = function (e) {
- var isPrimaryButton = ((e.buttons || e.button) & 1) === 1;
- // Normalize before the 'if' for the legacy IE (#7850)
- e = this.normalize(e);
- // #11635, Firefox does not reliable fire move event after click scroll
- if (H.isFirefox &&
- e.button !== 0) {
- this.onContainerMouseMove(e);
- }
- // #11635, limiting to primary button (incl. IE 8 support)
- if (typeof e.button === 'undefined' ||
- isPrimaryButton) {
- this.zoomOption(e);
- // #295, #13737 solve conflict between container drag and chart zoom
- if (isPrimaryButton &&
- e.preventDefault) {
- e.preventDefault();
- }
- this.dragStart(e);
- }
- };
- /**
- * When mouse leaves the container, hide the tooltip.
- *
- * @private
- * @function Highcharts.Pointer#onContainerMouseLeave
- *
- * @param {global.MouseEvent} e
- *
- * @return {void}
- */
- Pointer.prototype.onContainerMouseLeave = function (e) {
- var chart = charts[pick(H.hoverChartIndex, -1)];
- var tooltip = this.chart.tooltip;
- e = this.normalize(e);
- // #4886, MS Touch end fires mouseleave but with no related target
- if (chart &&
- (e.relatedTarget || e.toElement)) {
- chart.pointer.reset();
- // Also reset the chart position, used in #149 fix
- chart.pointer.chartPosition = void 0;
- }
- if ( // #11635, Firefox wheel scroll does not fire out events consistently
- tooltip &&
- !tooltip.isHidden) {
- this.reset();
- }
- };
- /**
- * When mouse enters the container, delete pointer's chartPosition.
- *
- * @private
- * @function Highcharts.Pointer#onContainerMouseEnter
- *
- * @param {global.MouseEvent} e
- *
- * @return {void}
- */
- Pointer.prototype.onContainerMouseEnter = function (e) {
- delete this.chartPosition;
- };
- /**
- * The mousemove, touchmove and touchstart event handler
- *
- * @private
- * @function Highcharts.Pointer#onContainerMouseMove
- *
- * @param {global.MouseEvent} e
- *
- * @return {void}
- */
- Pointer.prototype.onContainerMouseMove = function (e) {
- var chart = this.chart;
- var pEvt = this.normalize(e);
- this.setHoverChartIndex();
- // In IE8 we apparently need this returnValue set to false in order to
- // avoid text being selected. But in Chrome, e.returnValue is prevented,
- // plus we don't need to run e.preventDefault to prevent selected text
- // in modern browsers. So we set it conditionally. Remove it when IE8 is
- // no longer needed. #2251, #3224.
- if (!pEvt.preventDefault) {
- pEvt.returnValue = false;
- }
- if (chart.mouseIsDown === 'mousedown') {
- this.drag(pEvt);
- }
- // Show the tooltip and run mouse over events (#977)
- if (!chart.openMenu &&
- (this.inClass(pEvt.target, 'highcharts-tracker') ||
- chart.isInsidePlot((pEvt.chartX - chart.plotLeft), (pEvt.chartY - chart.plotTop)))) {
- this.runPointActions(pEvt);
- }
- };
- /**
- * @private
- * @function Highcharts.Pointer#onDocumentTouchEnd
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.onDocumentTouchEnd = function (e) {
- if (charts[H.hoverChartIndex]) {
- charts[H.hoverChartIndex].pointer.drop(e);
- }
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerTouchMove
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.onContainerTouchMove = function (e) {
- this.touch(e);
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerTouchStart
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.onContainerTouchStart = function (e) {
- this.zoomOption(e);
- this.touch(e, true);
- };
- /**
- * Special handler for mouse move that will hide the tooltip when the mouse
- * leaves the plotarea. Issue #149 workaround. The mouseleave event does not
- * always fire.
- *
- * @private
- * @function Highcharts.Pointer#onDocumentMouseMove
- *
- * @param {global.MouseEvent} e
- *
- * @return {void}
- */
- Pointer.prototype.onDocumentMouseMove = function (e) {
- var chart = this.chart;
- var chartPosition = this.chartPosition;
- var pEvt = this.normalize(e,
- chartPosition);
- var tooltip = chart.tooltip;
- // If we're outside, hide the tooltip
- if (chartPosition &&
- (!tooltip ||
- !tooltip.isStickyOnContact()) &&
- !chart.isInsidePlot(pEvt.chartX - chart.plotLeft, pEvt.chartY - chart.plotTop) &&
- !this.inClass(pEvt.target, 'highcharts-tracker')) {
- this.reset();
- }
- };
- /**
- * @private
- * @function Highcharts.Pointer#onDocumentMouseUp
- *
- * @param {global.MouseEvent} e
- *
- * @return {void}
- */
- Pointer.prototype.onDocumentMouseUp = function (e) {
- var chart = charts[pick(H.hoverChartIndex, -1)];
- if (chart) {
- chart.pointer.drop(e);
- }
- };
- /**
- * Handle touch events with two touches
- *
- * @private
- * @function Highcharts.Pointer#pinch
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- Pointer.prototype.pinch = function (e) {
- var self = this,
- chart = self.chart,
- pinchDown = self.pinchDown,
- touches = (e.touches || []),
- touchesLength = touches.length,
- lastValidTouch = self.lastValidTouch,
- hasZoom = self.hasZoom,
- selectionMarker = self.selectionMarker,
- transform = {},
- fireClickEvent = touchesLength === 1 && ((self.inClass(e.target, 'highcharts-tracker') &&
- chart.runTrackerClick) ||
- self.runChartClick),
- clip = {};
- // Don't initiate panning until the user has pinched. This prevents us
- // from blocking page scrolling as users scroll down a long page
- // (#4210).
- if (touchesLength > 1) {
- self.initiated = true;
- }
- // On touch devices, only proceed to trigger click if a handler is
- // defined
- if (hasZoom && self.initiated && !fireClickEvent) {
- e.preventDefault();
- }
- // Normalize each touch
- [].map.call(touches, function (e) {
- return self.normalize(e);
- });
- // Register the touch start position
- if (e.type === 'touchstart') {
- [].forEach.call(touches, function (e, i) {
- pinchDown[i] = { chartX: e.chartX, chartY: e.chartY };
- });
- lastValidTouch.x = [pinchDown[0].chartX, pinchDown[1] &&
- pinchDown[1].chartX];
- lastValidTouch.y = [pinchDown[0].chartY, pinchDown[1] &&
- pinchDown[1].chartY];
- // Identify the data bounds in pixels
- chart.axes.forEach(function (axis) {
- if (axis.zoomEnabled) {
- var bounds = chart.bounds[axis.horiz ? 'h' : 'v'],
- minPixelPadding = axis.minPixelPadding,
- min = axis.toPixels(Math.min(pick(axis.options.min,
- axis.dataMin),
- axis.dataMin)),
- max = axis.toPixels(Math.max(pick(axis.options.max,
- axis.dataMax),
- axis.dataMax)),
- absMin = Math.min(min,
- max),
- absMax = Math.max(min,
- max);
- // Store the bounds for use in the touchmove handler
- bounds.min = Math.min(axis.pos, absMin - minPixelPadding);
- bounds.max = Math.max(axis.pos + axis.len, absMax + minPixelPadding);
- }
- });
- self.res = true; // reset on next move
- // Optionally move the tooltip on touchmove
- }
- else if (self.followTouchMove && touchesLength === 1) {
- this.runPointActions(self.normalize(e));
- // Event type is touchmove, handle panning and pinching
- }
- else if (pinchDown.length) { // can be 0 when releasing, if touchend
- // fires first
- // Set the marker
- if (!selectionMarker) {
- self.selectionMarker = selectionMarker = extend({
- destroy: noop,
- touch: true
- }, chart.plotBox);
- }
- self.pinchTranslate(pinchDown, touches, transform, selectionMarker, clip, lastValidTouch);
- self.hasPinched = hasZoom;
- // Scale and translate the groups to provide visual feedback during
- // pinching
- self.scaleGroups(transform, clip);
- if (self.res) {
- self.res = false;
- this.reset(false, 0);
- }
- }
- };
- /**
- * Run translation operations
- *
- * @private
- * @function Highcharts.Pointer#pinchTranslate
- *
- * @param {Array<*>} pinchDown
- *
- * @param {Array<Highcharts.PointerEventObject>} touches
- *
- * @param {*} transform
- *
- * @param {*} selectionMarker
- *
- * @param {*} clip
- *
- * @param {*} lastValidTouch
- *
- * @return {void}
- */
- Pointer.prototype.pinchTranslate = function (pinchDown, touches, transform, selectionMarker, clip, lastValidTouch) {
- if (this.zoomHor) {
- this.pinchTranslateDirection(true, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch);
- }
- if (this.zoomVert) {
- this.pinchTranslateDirection(false, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch);
- }
- };
- /**
- * Run translation operations for each direction (horizontal and vertical)
- * independently.
- *
- * @private
- * @function Highcharts.Pointer#pinchTranslateDirection
- *
- * @param {boolean} horiz
- *
- * @param {Array<*>} pinchDown
- *
- * @param {Array<Highcharts.PointerEventObject>} touches
- *
- * @param {*} transform
- *
- * @param {*} selectionMarker
- *
- * @param {*} clip
- *
- * @param {*} lastValidTouch
- *
- * @param {number|undefined} [forcedScale=1]
- *
- * @return {void}
- */
- Pointer.prototype.pinchTranslateDirection = function (horiz, pinchDown, touches, transform, selectionMarker, clip, lastValidTouch, forcedScale) {
- var chart = this.chart, xy = horiz ? 'x' : 'y', XY = horiz ? 'X' : 'Y', sChartXY = ('chart' + XY), wh = horiz ? 'width' : 'height', plotLeftTop = chart['plot' + (horiz ? 'Left' : 'Top')], selectionWH, selectionXY, clipXY, scale = forcedScale || 1, inverted = chart.inverted, bounds = chart.bounds[horiz ? 'h' : 'v'], singleTouch = pinchDown.length === 1, touch0Start = pinchDown[0][sChartXY], touch0Now = touches[0][sChartXY], touch1Start = !singleTouch && pinchDown[1][sChartXY], touch1Now = !singleTouch && touches[1][sChartXY], outOfBounds, transformScale, scaleKey, setScale = function () {
- // Don't zoom if fingers are too close on this axis
- if (typeof touch1Now === 'number' &&
- Math.abs(touch0Start - touch1Start) > 20) {
- scale = forcedScale ||
- Math.abs(touch0Now - touch1Now) /
- Math.abs(touch0Start - touch1Start);
- }
- clipXY = ((plotLeftTop - touch0Now) / scale) + touch0Start;
- selectionWH = chart['plot' + (horiz ? 'Width' : 'Height')] / scale;
- };
- // Set the scale, first pass
- setScale();
- // The clip position (x or y) is altered if out of bounds, the selection
- // position is not
- selectionXY = clipXY;
- // Out of bounds
- if (selectionXY < bounds.min) {
- selectionXY = bounds.min;
- outOfBounds = true;
- }
- else if (selectionXY + selectionWH > bounds.max) {
- selectionXY = bounds.max - selectionWH;
- outOfBounds = true;
- }
- // Is the chart dragged off its bounds, determined by dataMin and
- // dataMax?
- if (outOfBounds) {
- // Modify the touchNow position in order to create an elastic drag
- // movement. This indicates to the user that the chart is responsive
- // but can't be dragged further.
- touch0Now -= 0.8 * (touch0Now - lastValidTouch[xy][0]);
- if (typeof touch1Now === 'number') {
- touch1Now -= 0.8 * (touch1Now - lastValidTouch[xy][1]);
- }
- // Set the scale, second pass to adapt to the modified touchNow
- // positions
- setScale();
- }
- else {
- lastValidTouch[xy] = [touch0Now, touch1Now];
- }
- // Set geometry for clipping, selection and transformation
- if (!inverted) {
- clip[xy] = clipXY - plotLeftTop;
- clip[wh] = selectionWH;
- }
- scaleKey = inverted ? (horiz ? 'scaleY' : 'scaleX') : 'scale' + XY;
- transformScale = inverted ? 1 / scale : scale;
- selectionMarker[wh] = selectionWH;
- selectionMarker[xy] = selectionXY;
- transform[scaleKey] = scale;
- transform['translate' + XY] = (transformScale * plotLeftTop) +
- (touch0Now - (transformScale * touch0Start));
- };
- /**
- * Reset the tracking by hiding the tooltip, the hover series state and the
- * hover point
- *
- * @function Highcharts.Pointer#reset
- *
- * @param {boolean} [allowMove]
- * Instead of destroying the tooltip altogether, allow moving it if
- * possible.
- *
- * @param {number} [delay]
- *
- * @return {void}
- */
- Pointer.prototype.reset = function (allowMove, delay) {
- var pointer = this,
- chart = pointer.chart,
- hoverSeries = chart.hoverSeries,
- hoverPoint = chart.hoverPoint,
- hoverPoints = chart.hoverPoints,
- tooltip = chart.tooltip,
- tooltipPoints = tooltip && tooltip.shared ?
- hoverPoints :
- hoverPoint;
- // Check if the points have moved outside the plot area (#1003, #4736,
- // #5101)
- if (allowMove && tooltipPoints) {
- splat(tooltipPoints).forEach(function (point) {
- if (point.series.isCartesian &&
- typeof point.plotX === 'undefined') {
- allowMove = false;
- }
- });
- }
- // Just move the tooltip, #349
- if (allowMove) {
- if (tooltip && tooltipPoints && splat(tooltipPoints).length) {
- tooltip.refresh(tooltipPoints);
- if (tooltip.shared && hoverPoints) { // #8284
- hoverPoints.forEach(function (point) {
- point.setState(point.state, true);
- if (point.series.isCartesian) {
- if (point.series.xAxis.crosshair) {
- point.series.xAxis
- .drawCrosshair(null, point);
- }
- if (point.series.yAxis.crosshair) {
- point.series.yAxis
- .drawCrosshair(null, point);
- }
- }
- });
- }
- else if (hoverPoint) { // #2500
- hoverPoint.setState(hoverPoint.state, true);
- chart.axes.forEach(function (axis) {
- if (axis.crosshair &&
- hoverPoint.series[axis.coll] === axis) {
- axis.drawCrosshair(null, hoverPoint);
- }
- });
- }
- }
- // Full reset
- }
- else {
- if (hoverPoint) {
- hoverPoint.onMouseOut();
- }
- if (hoverPoints) {
- hoverPoints.forEach(function (point) {
- point.setState();
- });
- }
- if (hoverSeries) {
- hoverSeries.onMouseOut();
- }
- if (tooltip) {
- tooltip.hide(delay);
- }
- if (pointer.unDocMouseMove) {
- pointer.unDocMouseMove = pointer.unDocMouseMove();
- }
- // Remove crosshairs
- chart.axes.forEach(function (axis) {
- axis.hideCrosshair();
- });
- pointer.hoverX = chart.hoverPoints = chart.hoverPoint = null;
- }
- };
- /**
- * With line type charts with a single tracker, get the point closest to the
- * mouse. Run Point.onMouseOver and display tooltip for the point or points.
- *
- * @private
- * @function Highcharts.Pointer#runPointActions
- *
- * @param {global.Event} e
- *
- * @param {Highcharts.PointerEventObject} [p]
- *
- * @return {void}
- *
- * @fires Highcharts.Point#event:mouseOut
- * @fires Highcharts.Point#event:mouseOver
- */
- Pointer.prototype.runPointActions = function (e, p) {
- var pointer = this,
- chart = pointer.chart,
- series = chart.series,
- tooltip = (chart.tooltip && chart.tooltip.options.enabled ?
- chart.tooltip :
- void 0),
- shared = (tooltip ?
- tooltip.shared :
- false),
- hoverPoint = p || chart.hoverPoint,
- hoverSeries = hoverPoint && hoverPoint.series || chart.hoverSeries,
- // onMouseOver or already hovering a series with directTouch
- isDirectTouch = (!e || e.type !== 'touchmove') && (!!p || ((hoverSeries && hoverSeries.directTouch) &&
- pointer.isDirectTouch)),
- hoverData = this.getHoverData(hoverPoint,
- hoverSeries,
- series,
- isDirectTouch,
- shared,
- e),
- useSharedTooltip,
- followPointer,
- anchor,
- points;
- // Update variables from hoverData.
- hoverPoint = hoverData.hoverPoint;
- points = hoverData.hoverPoints;
- hoverSeries = hoverData.hoverSeries;
- followPointer = hoverSeries && hoverSeries.tooltipOptions.followPointer;
- useSharedTooltip = (shared &&
- hoverSeries &&
- !hoverSeries.noSharedTooltip);
- // Refresh tooltip for kdpoint if new hover point or tooltip was hidden
- // #3926, #4200
- if (hoverPoint &&
- // !(hoverSeries && hoverSeries.directTouch) &&
- (hoverPoint !== chart.hoverPoint || (tooltip && tooltip.isHidden))) {
- (chart.hoverPoints || []).forEach(function (p) {
- if (points.indexOf(p) === -1) {
- p.setState();
- }
- });
- // Set normal state to previous series
- if (chart.hoverSeries !== hoverSeries) {
- hoverSeries.onMouseOver();
- }
- pointer.applyInactiveState(points);
- // Do mouseover on all points (#3919, #3985, #4410, #5622)
- (points || []).forEach(function (p) {
- p.setState('hover');
- });
- // If tracking is on series in stead of on each point,
- // fire mouseOver on hover point. // #4448
- if (chart.hoverPoint) {
- chart.hoverPoint.firePointEvent('mouseOut');
- }
- // Hover point may have been destroyed in the event handlers (#7127)
- if (!hoverPoint.series) {
- return;
- }
- /**
- * Contains all hovered points.
- *
- * @name Highcharts.Chart#hoverPoints
- * @type {Array<Highcharts.Point>|null}
- */
- chart.hoverPoints = points;
- /**
- * Contains the original hovered point.
- *
- * @name Highcharts.Chart#hoverPoint
- * @type {Highcharts.Point|null}
- */
- chart.hoverPoint = hoverPoint;
- /**
- * Hover state should not be lost when axis is updated (#12569)
- * Axis.update runs pointer.reset which uses chart.hoverPoint.state
- * to apply state which does not exist in hoverPoint yet.
- * The mouseOver event should be triggered when hoverPoint
- * is correct.
- */
- hoverPoint.firePointEvent('mouseOver');
- // Draw tooltip if necessary
- if (tooltip) {
- tooltip.refresh(useSharedTooltip ? points : hoverPoint, e);
- }
- // Update positions (regardless of kdpoint or hoverPoint)
- }
- else if (followPointer && tooltip && !tooltip.isHidden) {
- anchor = tooltip.getAnchor([{}], e);
- tooltip.updatePosition({ plotX: anchor[0], plotY: anchor[1] });
- }
- // Start the event listener to pick up the tooltip and crosshairs
- if (!pointer.unDocMouseMove) {
- pointer.unDocMouseMove = addEvent(chart.container.ownerDocument, 'mousemove', function (e) {
- var chart = charts[H.hoverChartIndex];
- if (chart) {
- chart.pointer.onDocumentMouseMove(e);
- }
- });
- }
- // Issues related to crosshair #4927, #5269 #5066, #5658
- chart.axes.forEach(function drawAxisCrosshair(axis) {
- var snap = pick((axis.crosshair || {}).snap,
- true);
- var point;
- if (snap) {
- point = chart.hoverPoint; // #13002
- if (!point || point.series[axis.coll] !== axis) {
- point = find(points, function (p) {
- return p.series[axis.coll] === axis;
- });
- }
- }
- // Axis has snapping crosshairs, and one of the hover points belongs
- // to axis. Always call drawCrosshair when it is not snap.
- if (point || !snap) {
- axis.drawCrosshair(e, point);
- // Axis has snapping crosshairs, but no hover point belongs to axis
- }
- else {
- axis.hideCrosshair();
- }
- });
- };
- /**
- * Scale series groups to a certain scale and translation.
- *
- * @private
- * @function Highcharts.Pointer#scaleGroups
- *
- * @param {Highcharts.SeriesPlotBoxObject} [attribs]
- *
- * @param {boolean} [clip]
- *
- * @return {void}
- */
- Pointer.prototype.scaleGroups = function (attribs, clip) {
- var chart = this.chart,
- seriesAttribs;
- // Scale each series
- chart.series.forEach(function (series) {
- seriesAttribs = attribs || series.getPlotBox(); // #1701
- if (series.xAxis && series.xAxis.zoomEnabled && series.group) {
- series.group.attr(seriesAttribs);
- if (series.markerGroup) {
- series.markerGroup.attr(seriesAttribs);
- series.markerGroup.clip(clip ? chart.clipRect : null);
- }
- if (series.dataLabelsGroup) {
- series.dataLabelsGroup.attr(seriesAttribs);
- }
- }
- });
- // Clip
- chart.clipRect.attr(clip || chart.clipBox);
- };
- /**
- * Set the JS DOM events on the container and document. This method should
- * contain a one-to-one assignment between methods and their handlers. Any
- * advanced logic should be moved to the handler reflecting the event's
- * name.
- *
- * @private
- * @function Highcharts.Pointer#setDOMEvents
- *
- * @return {void}
- */
- Pointer.prototype.setDOMEvents = function () {
- var container = this.chart.container,
- ownerDoc = container.ownerDocument;
- container.onmousedown = this.onContainerMouseDown.bind(this);
- container.onmousemove = this.onContainerMouseMove.bind(this);
- container.onclick = this.onContainerClick.bind(this);
- this.unbindContainerMouseEnter = addEvent(container, 'mouseenter', this.onContainerMouseEnter.bind(this));
- this.unbindContainerMouseLeave = addEvent(container, 'mouseleave', this.onContainerMouseLeave.bind(this));
- if (!H.unbindDocumentMouseUp) {
- H.unbindDocumentMouseUp = addEvent(ownerDoc, 'mouseup', this.onDocumentMouseUp.bind(this));
- }
- if (H.hasTouch) {
- addEvent(container, 'touchstart', this.onContainerTouchStart.bind(this));
- addEvent(container, 'touchmove', this.onContainerTouchMove.bind(this));
- if (!H.unbindDocumentTouchEnd) {
- H.unbindDocumentTouchEnd = addEvent(ownerDoc, 'touchend', this.onDocumentTouchEnd.bind(this));
- }
- }
- };
- /**
- * Sets the index of the hovered chart and leaves the previous hovered
- * chart, to reset states like tooltip.
- *
- * @private
- * @function Highcharts.Pointer#setHoverChartIndex
- */
- Pointer.prototype.setHoverChartIndex = function () {
- var chart = this.chart;
- var hoverChart = H.charts[pick(H.hoverChartIndex, -1)];
- if (hoverChart &&
- hoverChart !== chart) {
- hoverChart.pointer.onContainerMouseLeave({ relatedTarget: true });
- }
- if (!hoverChart ||
- !hoverChart.mouseIsDown) {
- H.hoverChartIndex = chart.index;
- }
- };
- /**
- * General touch handler shared by touchstart and touchmove.
- *
- * @private
- * @function Highcharts.Pointer#touch
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @param {boolean} [start]
- *
- * @return {void}
- */
- Pointer.prototype.touch = function (e, start) {
- var chart = this.chart,
- hasMoved,
- pinchDown,
- isInside;
- this.setHoverChartIndex();
- if (e.touches.length === 1) {
- e = this.normalize(e);
- isInside = chart.isInsidePlot(e.chartX - chart.plotLeft, e.chartY - chart.plotTop);
- if (isInside && !chart.openMenu) {
- // Run mouse events and display tooltip etc
- if (start) {
- this.runPointActions(e);
- }
- // Android fires touchmove events after the touchstart even if
- // the finger hasn't moved, or moved only a pixel or two. In iOS
- // however, the touchmove doesn't fire unless the finger moves
- // more than ~4px. So we emulate this behaviour in Android by
- // checking how much it moved, and cancelling on small
- // distances. #3450.
- if (e.type === 'touchmove') {
- pinchDown = this.pinchDown;
- hasMoved = pinchDown[0] ? Math.sqrt(// #5266
- Math.pow(pinchDown[0].chartX - e.chartX, 2) +
- Math.pow(pinchDown[0].chartY - e.chartY, 2)) >= 4 : false;
- }
- if (pick(hasMoved, true)) {
- this.pinch(e);
- }
- }
- else if (start) {
- // Hide the tooltip on touching outside the plot area (#1203)
- this.reset();
- }
- }
- else if (e.touches.length === 2) {
- this.pinch(e);
- }
- };
- /**
- * Resolve the zoomType option, this is reset on all touch start and mouse
- * down events.
- *
- * @private
- * @function Highcharts.Pointer#zoomOption
- *
- * @param {global.Event} e
- * Event object.
- *
- * @param {void}
- */
- Pointer.prototype.zoomOption = function (e) {
- var chart = this.chart,
- options = chart.options.chart,
- zoomType = options.zoomType || '',
- inverted = chart.inverted,
- zoomX,
- zoomY;
- // Look for the pinchType option
- if (/touch/.test(e.type)) {
- zoomType = pick(options.pinchType, zoomType);
- }
- this.zoomX = zoomX = /x/.test(zoomType);
- this.zoomY = zoomY = /y/.test(zoomType);
- this.zoomHor = (zoomX && !inverted) || (zoomY && inverted);
- this.zoomVert = (zoomY && !inverted) || (zoomX && inverted);
- this.hasZoom = zoomX || zoomY;
- };
- return Pointer;
- }());
- H.Pointer = Pointer;
- return Pointer;
- });
- _registerModule(_modules, 'Core/MSPointer.js', [_modules['Core/Globals.js'], _modules['Core/Pointer.js'], _modules['Core/Utilities.js']], function (H, Pointer, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var __extends = (this && this.__extends) || (function () {
- var extendStatics = function (d,
- b) {
- extendStatics = Object.setPrototypeOf ||
- ({ __proto__: [] } instanceof Array && function (d,
- b) { d.__proto__ = b; }) ||
- function (d,
- b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
- return extendStatics(d, b);
- };
- return function (d, b) {
- extendStatics(d, b);
- function __() { this.constructor = d; }
- d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
- };
- })();
- var charts = H.charts,
- doc = H.doc,
- noop = H.noop,
- win = H.win;
- var addEvent = U.addEvent,
- css = U.css,
- objectEach = U.objectEach,
- removeEvent = U.removeEvent;
- /* globals MSPointerEvent, PointerEvent */
- // The touches object keeps track of the points being touched at all times
- var touches = {};
- var hasPointerEvent = !!win.PointerEvent;
- /* eslint-disable valid-jsdoc */
- /** @private */
- function getWebkitTouches() {
- var fake = [];
- fake.item = function (i) {
- return this[i];
- };
- objectEach(touches, function (touch) {
- fake.push({
- pageX: touch.pageX,
- pageY: touch.pageY,
- target: touch.target
- });
- });
- return fake;
- }
- /** @private */
- function translateMSPointer(e, method, wktype, func) {
- var p;
- if ((e.pointerType === 'touch' ||
- e.pointerType === e.MSPOINTER_TYPE_TOUCH) && charts[H.hoverChartIndex]) {
- func(e);
- p = charts[H.hoverChartIndex].pointer;
- p[method]({
- type: wktype,
- target: e.currentTarget,
- preventDefault: noop,
- touches: getWebkitTouches()
- });
- }
- }
- /** @private */
- var MSPointer = /** @class */ (function (_super) {
- __extends(MSPointer, _super);
- function MSPointer() {
- return _super !== null && _super.apply(this, arguments) || this;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Add or remove the MS Pointer specific events
- *
- * @private
- * @function Highcharts.Pointer#batchMSEvents
- *
- * @param {Function} fn
- *
- * @return {void}
- */
- MSPointer.prototype.batchMSEvents = function (fn) {
- fn(this.chart.container, hasPointerEvent ? 'pointerdown' : 'MSPointerDown', this.onContainerPointerDown);
- fn(this.chart.container, hasPointerEvent ? 'pointermove' : 'MSPointerMove', this.onContainerPointerMove);
- fn(doc, hasPointerEvent ? 'pointerup' : 'MSPointerUp', this.onDocumentPointerUp);
- };
- // Destroy MS events also
- MSPointer.prototype.destroy = function () {
- this.batchMSEvents(removeEvent);
- _super.prototype.destroy.call(this);
- };
- // Disable default IE actions for pinch and such on chart element
- MSPointer.prototype.init = function (chart, options) {
- _super.prototype.init.call(this, chart, options);
- if (this.hasZoom) { // #4014
- css(chart.container, {
- '-ms-touch-action': 'none',
- 'touch-action': 'none'
- });
- }
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerPointerDown
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- MSPointer.prototype.onContainerPointerDown = function (e) {
- translateMSPointer(e, 'onContainerTouchStart', 'touchstart', function (e) {
- touches[e.pointerId] = {
- pageX: e.pageX,
- pageY: e.pageY,
- target: e.currentTarget
- };
- });
- };
- /**
- * @private
- * @function Highcharts.Pointer#onContainerPointerMove
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- MSPointer.prototype.onContainerPointerMove = function (e) {
- translateMSPointer(e, 'onContainerTouchMove', 'touchmove', function (e) {
- touches[e.pointerId] = ({ pageX: e.pageX, pageY: e.pageY });
- if (!touches[e.pointerId].target) {
- touches[e.pointerId].target = e.currentTarget;
- }
- });
- };
- /**
- * @private
- * @function Highcharts.Pointer#onDocumentPointerUp
- *
- * @param {Highcharts.PointerEventObject} e
- *
- * @return {void}
- */
- MSPointer.prototype.onDocumentPointerUp = function (e) {
- translateMSPointer(e, 'onDocumentTouchEnd', 'touchend', function (e) {
- delete touches[e.pointerId];
- });
- };
- // Add IE specific touch events to chart
- MSPointer.prototype.setDOMEvents = function () {
- _super.prototype.setDOMEvents.call(this);
- if (this.hasZoom || this.followTouchMove) {
- this.batchMSEvents(addEvent);
- }
- };
- return MSPointer;
- }(Pointer));
- return MSPointer;
- });
- _registerModule(_modules, 'Core/Legend.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Gets fired when the legend item belonging to a point is clicked. The default
- * action is to toggle the visibility of the point. This can be prevented by
- * returning `false` or calling `event.preventDefault()`.
- *
- * @callback Highcharts.PointLegendItemClickCallbackFunction
- *
- * @param {Highcharts.Point} this
- * The point on which the event occured.
- *
- * @param {Highcharts.PointLegendItemClickEventObject} event
- * The event that occured.
- */
- /**
- * Information about the legend click event.
- *
- * @interface Highcharts.PointLegendItemClickEventObject
- */ /**
- * Related browser event.
- * @name Highcharts.PointLegendItemClickEventObject#browserEvent
- * @type {Highcharts.PointerEvent}
- */ /**
- * Prevent the default action of toggle the visibility of the point.
- * @name Highcharts.PointLegendItemClickEventObject#preventDefault
- * @type {Function}
- */ /**
- * Related point.
- * @name Highcharts.PointLegendItemClickEventObject#target
- * @type {Highcharts.Point}
- */ /**
- * Event type.
- * @name Highcharts.PointLegendItemClickEventObject#type
- * @type {"legendItemClick"}
- */
- /**
- * Gets fired when the legend item belonging to a series is clicked. The default
- * action is to toggle the visibility of the series. This can be prevented by
- * returning `false` or calling `event.preventDefault()`.
- *
- * @callback Highcharts.SeriesLegendItemClickCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {Highcharts.SeriesLegendItemClickEventObject} event
- * The event that occured.
- */
- /**
- * Information about the legend click event.
- *
- * @interface Highcharts.SeriesLegendItemClickEventObject
- */ /**
- * Related browser event.
- * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent
- * @type {Highcharts.PointerEvent}
- */ /**
- * Prevent the default action of toggle the visibility of the series.
- * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault
- * @type {Function}
- */ /**
- * Related series.
- * @name Highcharts.SeriesLegendItemClickEventObject#target
- * @type {Highcharts.Series}
- */ /**
- * Event type.
- * @name Highcharts.SeriesLegendItemClickEventObject#type
- * @type {"legendItemClick"}
- */
- var addEvent = U.addEvent,
- animObject = U.animObject,
- css = U.css,
- defined = U.defined,
- discardElement = U.discardElement,
- find = U.find,
- fireEvent = U.fireEvent,
- format = U.format,
- isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- relativeLength = U.relativeLength,
- setAnimation = U.setAnimation,
- stableSort = U.stableSort,
- syncTimeout = U.syncTimeout,
- wrap = U.wrap;
- var isFirefox = H.isFirefox,
- marginNames = H.marginNames,
- win = H.win;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The overview of the chart's series. The legend object is instanciated
- * internally in the chart constructor, and is available from the `chart.legend`
- * property. Each chart has only one legend.
- *
- * @class
- * @name Highcharts.Legend
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.LegendOptions} options
- * Legend options.
- */
- var Legend = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Legend(chart, options) {
- /* *
- *
- * Properties
- *
- * */
- this.allItems = [];
- this.box = void 0;
- this.contentGroup = void 0;
- this.display = false;
- this.group = void 0;
- this.initialItemY = 0;
- this.itemHeight = 0;
- this.itemMarginBottom = 0;
- this.itemMarginTop = 0;
- this.itemX = 0;
- this.itemY = 0;
- this.lastItemY = 0;
- this.lastLineHeight = 0;
- this.legendHeight = 0;
- this.legendWidth = 0;
- this.maxItemWidth = 0;
- this.maxLegendWidth = 0;
- this.offsetWidth = 0;
- this.options = {};
- this.padding = 0;
- this.pages = [];
- this.proximate = false;
- this.scrollGroup = void 0;
- this.symbolHeight = 0;
- this.symbolWidth = 0;
- this.titleHeight = 0;
- this.totalItemWidth = 0;
- this.widthOption = 0;
- this.chart = chart;
- this.init(chart, options);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Initialize the legend.
- *
- * @private
- * @function Highcharts.Legend#init
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.LegendOptions} options
- * Legend options.
- */
- Legend.prototype.init = function (chart, options) {
- /**
- * Chart of this legend.
- *
- * @readonly
- * @name Highcharts.Legend#chart
- * @type {Highcharts.Chart}
- */
- this.chart = chart;
- this.setOptions(options);
- if (options.enabled) {
- // Render it
- this.render();
- // move checkboxes
- addEvent(this.chart, 'endResize', function () {
- this.legend.positionCheckboxes();
- });
- if (this.proximate) {
- this.unchartrender = addEvent(this.chart, 'render', function () {
- this.legend.proximatePositions();
- this.legend.positionItems();
- });
- }
- else if (this.unchartrender) {
- this.unchartrender();
- }
- }
- };
- /**
- * @private
- * @function Highcharts.Legend#setOptions
- * @param {Highcharts.LegendOptions} options
- */
- Legend.prototype.setOptions = function (options) {
- var padding = pick(options.padding, 8);
- /**
- * Legend options.
- *
- * @readonly
- * @name Highcharts.Legend#options
- * @type {Highcharts.LegendOptions}
- */
- this.options = options;
- if (!this.chart.styledMode) {
- this.itemStyle = options.itemStyle;
- this.itemHiddenStyle = merge(this.itemStyle, options.itemHiddenStyle);
- }
- this.itemMarginTop = options.itemMarginTop || 0;
- this.itemMarginBottom = options.itemMarginBottom || 0;
- this.padding = padding;
- this.initialItemY = padding - 5; // 5 is pixels above the text
- this.symbolWidth = pick(options.symbolWidth, 16);
- this.pages = [];
- this.proximate = options.layout === 'proximate' && !this.chart.inverted;
- this.baseline = void 0; // #12705: baseline has to be reset on every update
- };
- /**
- * Update the legend with new options. Equivalent to running `chart.update`
- * with a legend configuration option.
- *
- * @sample highcharts/legend/legend-update/
- * Legend update
- *
- * @function Highcharts.Legend#update
- *
- * @param {Highcharts.LegendOptions} options
- * Legend options.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the axis is altered. If doing more
- * operations on the chart, it is a good idea to set redraw to false and
- * call {@link Chart#redraw} after. Whether to redraw the chart.
- *
- * @fires Highcharts.Legends#event:afterUpdate
- */
- Legend.prototype.update = function (options, redraw) {
- var chart = this.chart;
- this.setOptions(merge(true, this.options, options));
- this.destroy();
- chart.isDirtyLegend = chart.isDirtyBox = true;
- if (pick(redraw, true)) {
- chart.redraw();
- }
- fireEvent(this, 'afterUpdate');
- };
- /**
- * Set the colors for the legend item.
- *
- * @private
- * @function Highcharts.Legend#colorizeItem
- * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item
- * A Series or Point instance
- * @param {boolean} [visible=false]
- * Dimmed or colored
- *
- * @todo
- * Make events official: Fires the event `afterColorizeItem`.
- */
- Legend.prototype.colorizeItem = function (item, visible) {
- item.legendGroup[visible ? 'removeClass' : 'addClass']('highcharts-legend-item-hidden');
- if (!this.chart.styledMode) {
- var legend = this,
- options = legend.options,
- legendItem = item.legendItem,
- legendLine = item.legendLine,
- legendSymbol = item.legendSymbol,
- hiddenColor = legend.itemHiddenStyle.color,
- textColor = visible ?
- options.itemStyle.color :
- hiddenColor,
- symbolColor = visible ?
- (item.color || hiddenColor) :
- hiddenColor,
- markerOptions = item.options && item.options.marker,
- symbolAttr = { fill: symbolColor };
- if (legendItem) {
- legendItem.css({
- fill: textColor,
- color: textColor // #1553, oldIE
- });
- }
- if (legendLine) {
- legendLine.attr({ stroke: symbolColor });
- }
- if (legendSymbol) {
- // Apply marker options
- if (markerOptions && legendSymbol.isMarker) { // #585
- symbolAttr = item.pointAttribs();
- if (!visible) {
- // #6769
- symbolAttr.stroke = symbolAttr.fill = hiddenColor;
- }
- }
- legendSymbol.attr(symbolAttr);
- }
- }
- fireEvent(this, 'afterColorizeItem', { item: item, visible: visible });
- };
- /**
- * @private
- * @function Highcharts.Legend#positionItems
- */
- Legend.prototype.positionItems = function () {
- // Now that the legend width and height are established, put the items
- // in the final position
- this.allItems.forEach(this.positionItem, this);
- if (!this.chart.isResizing) {
- this.positionCheckboxes();
- }
- };
- /**
- * Position the legend item.
- *
- * @private
- * @function Highcharts.Legend#positionItem
- * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item
- * The item to position
- */
- Legend.prototype.positionItem = function (item) {
- var _this = this;
- var legend = this,
- options = legend.options,
- symbolPadding = options.symbolPadding,
- ltr = !options.rtl,
- legendItemPos = item._legendItemPos,
- itemX = legendItemPos[0],
- itemY = legendItemPos[1],
- checkbox = item.checkbox,
- legendGroup = item.legendGroup;
- if (legendGroup && legendGroup.element) {
- var attribs = {
- translateX: ltr ?
- itemX :
- legend.legendWidth - itemX - 2 * symbolPadding - 4,
- translateY: itemY
- };
- var complete = function () {
- fireEvent(_this, 'afterPositionItem', { item: item });
- };
- if (defined(legendGroup.translateY)) {
- legendGroup.animate(attribs, void 0, complete);
- }
- else {
- legendGroup.attr(attribs);
- complete();
- }
- }
- if (checkbox) {
- checkbox.x = itemX;
- checkbox.y = itemY;
- }
- };
- /**
- * Destroy a single legend item, used internally on removing series items.
- *
- * @private
- * @function Highcharts.Legend#destroyItem
- * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item
- * The item to remove
- */
- Legend.prototype.destroyItem = function (item) {
- var checkbox = item.checkbox;
- // destroy SVG elements
- ['legendItem', 'legendLine', 'legendSymbol', 'legendGroup'].forEach(function (key) {
- if (item[key]) {
- item[key] = item[key].destroy();
- }
- });
- if (checkbox) {
- discardElement(item.checkbox);
- }
- };
- /**
- * Destroy the legend. Used internally. To reflow objects, `chart.redraw`
- * must be called after destruction.
- *
- * @private
- * @function Highcharts.Legend#destroy
- */
- Legend.prototype.destroy = function () {
- /**
- * @private
- * @param {string} key
- * @return {void}
- */
- function destroyItems(key) {
- if (this[key]) {
- this[key] = this[key].destroy();
- }
- }
- // Destroy items
- this.getAllItems().forEach(function (item) {
- ['legendItem', 'legendGroup'].forEach(destroyItems, item);
- });
- // Destroy legend elements
- [
- 'clipRect',
- 'up',
- 'down',
- 'pager',
- 'nav',
- 'box',
- 'title',
- 'group'
- ].forEach(destroyItems, this);
- this.display = null; // Reset in .render on update.
- };
- /**
- * Position the checkboxes after the width is determined.
- *
- * @private
- * @function Highcharts.Legend#positionCheckboxes
- */
- Legend.prototype.positionCheckboxes = function () {
- var alignAttr = this.group && this.group.alignAttr,
- translateY,
- clipHeight = this.clipHeight || this.legendHeight,
- titleHeight = this.titleHeight;
- if (alignAttr) {
- translateY = alignAttr.translateY;
- this.allItems.forEach(function (item) {
- var checkbox = item.checkbox,
- top;
- if (checkbox) {
- top = translateY + titleHeight + checkbox.y +
- (this.scrollOffset || 0) + 3;
- css(checkbox, {
- left: (alignAttr.translateX + item.checkboxOffset +
- checkbox.x - 20) + 'px',
- top: top + 'px',
- display: this.proximate || (top > translateY - 6 &&
- top < translateY + clipHeight - 6) ?
- '' :
- 'none'
- });
- }
- }, this);
- }
- };
- /**
- * Render the legend title on top of the legend.
- *
- * @private
- * @function Highcharts.Legend#renderTitle
- */
- Legend.prototype.renderTitle = function () {
- var options = this.options,
- padding = this.padding,
- titleOptions = options.title,
- titleHeight = 0,
- bBox;
- if (titleOptions.text) {
- if (!this.title) {
- /**
- * SVG element of the legend title.
- *
- * @readonly
- * @name Highcharts.Legend#title
- * @type {Highcharts.SVGElement}
- */
- this.title = this.chart.renderer.label(titleOptions.text, padding - 3, padding - 4, null, null, null, options.useHTML, null, 'legend-title')
- .attr({ zIndex: 1 });
- if (!this.chart.styledMode) {
- this.title.css(titleOptions.style);
- }
- this.title.add(this.group);
- }
- // Set the max title width (#7253)
- if (!titleOptions.width) {
- this.title.css({
- width: this.maxLegendWidth + 'px'
- });
- }
- bBox = this.title.getBBox();
- titleHeight = bBox.height;
- this.offsetWidth = bBox.width; // #1717
- this.contentGroup.attr({ translateY: titleHeight });
- }
- this.titleHeight = titleHeight;
- };
- /**
- * Set the legend item text.
- *
- * @function Highcharts.Legend#setText
- * @param {Highcharts.Point|Highcharts.Series} item
- * The item for which to update the text in the legend.
- */
- Legend.prototype.setText = function (item) {
- var options = this.options;
- item.legendItem.attr({
- text: options.labelFormat ?
- format(options.labelFormat, item, this.chart) :
- options.labelFormatter.call(item)
- });
- };
- /**
- * Render a single specific legend item. Called internally from the `render`
- * function.
- *
- * @private
- * @function Highcharts.Legend#renderItem
- * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item
- * The item to render.
- */
- Legend.prototype.renderItem = function (item) {
- var legend = this,
- chart = legend.chart,
- renderer = chart.renderer,
- options = legend.options,
- horizontal = options.layout === 'horizontal',
- symbolWidth = legend.symbolWidth,
- symbolPadding = options.symbolPadding,
- itemStyle = legend.itemStyle,
- itemHiddenStyle = legend.itemHiddenStyle,
- itemDistance = horizontal ? pick(options.itemDistance, 20) : 0,
- ltr = !options.rtl,
- bBox,
- li = item.legendItem,
- isSeries = !item.series,
- series = !isSeries && item.series.drawLegendSymbol ?
- item.series :
- item,
- seriesOptions = series.options,
- showCheckbox = legend.createCheckboxForItem &&
- seriesOptions &&
- seriesOptions.showCheckbox,
- // full width minus text width
- itemExtraWidth = symbolWidth + symbolPadding +
- itemDistance + (showCheckbox ? 20 : 0),
- useHTML = options.useHTML,
- itemClassName = item.options.className;
- if (!li) { // generate it once, later move it
- // Generate the group box, a group to hold the symbol and text. Text
- // is to be appended in Legend class.
- item.legendGroup = renderer
- .g('legend-item')
- .addClass('highcharts-' + series.type + '-series ' +
- 'highcharts-color-' + item.colorIndex +
- (itemClassName ? ' ' + itemClassName : '') +
- (isSeries ?
- ' highcharts-series-' + item.index :
- ''))
- .attr({ zIndex: 1 })
- .add(legend.scrollGroup);
- // Generate the list item text and add it to the group
- item.legendItem = li = renderer.text('', ltr ?
- symbolWidth + symbolPadding :
- -symbolPadding, legend.baseline || 0, useHTML);
- if (!chart.styledMode) {
- // merge to prevent modifying original (#1021)
- li.css(merge(item.visible ?
- itemStyle :
- itemHiddenStyle));
- }
- li
- .attr({
- align: ltr ? 'left' : 'right',
- zIndex: 2
- })
- .add(item.legendGroup);
- // Get the baseline for the first item - the font size is equal for
- // all
- if (!legend.baseline) {
- legend.fontMetrics = renderer.fontMetrics(chart.styledMode ? 12 : itemStyle.fontSize, li);
- legend.baseline =
- legend.fontMetrics.f + 3 + legend.itemMarginTop;
- li.attr('y', legend.baseline);
- }
- // Draw the legend symbol inside the group box
- legend.symbolHeight =
- options.symbolHeight || legend.fontMetrics.f;
- series.drawLegendSymbol(legend, item);
- if (legend.setItemEvents) {
- legend.setItemEvents(item, li, useHTML);
- }
- }
- // Add the HTML checkbox on top
- if (showCheckbox && !item.checkbox && legend.createCheckboxForItem) {
- legend.createCheckboxForItem(item);
- }
- // Colorize the items
- legend.colorizeItem(item, item.visible);
- // Take care of max width and text overflow (#6659)
- if (chart.styledMode || !itemStyle.width) {
- li.css({
- width: ((options.itemWidth ||
- legend.widthOption ||
- chart.spacingBox.width) - itemExtraWidth) + 'px'
- });
- }
- // Always update the text
- legend.setText(item);
- // calculate the positions for the next line
- bBox = li.getBBox();
- item.itemWidth = item.checkboxOffset =
- options.itemWidth ||
- item.legendItemWidth ||
- bBox.width + itemExtraWidth;
- legend.maxItemWidth = Math.max(legend.maxItemWidth, item.itemWidth);
- legend.totalItemWidth += item.itemWidth;
- legend.itemHeight = item.itemHeight = Math.round(item.legendItemHeight || bBox.height || legend.symbolHeight);
- };
- /**
- * Get the position of the item in the layout. We now know the
- * maxItemWidth from the previous loop.
- *
- * @private
- * @function Highcharts.Legend#layoutItem
- * @param {Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series} item
- */
- Legend.prototype.layoutItem = function (item) {
- var options = this.options,
- padding = this.padding,
- horizontal = options.layout === 'horizontal',
- itemHeight = item.itemHeight,
- itemMarginBottom = this.itemMarginBottom,
- itemMarginTop = this.itemMarginTop,
- itemDistance = horizontal ? pick(options.itemDistance, 20) : 0,
- maxLegendWidth = this.maxLegendWidth,
- itemWidth = (options.alignColumns &&
- this.totalItemWidth > maxLegendWidth) ?
- this.maxItemWidth :
- item.itemWidth;
- // If the item exceeds the width, start a new line
- if (horizontal &&
- this.itemX - padding + itemWidth > maxLegendWidth) {
- this.itemX = padding;
- if (this.lastLineHeight) { // Not for the first line (#10167)
- this.itemY += (itemMarginTop +
- this.lastLineHeight +
- itemMarginBottom);
- }
- this.lastLineHeight = 0; // reset for next line (#915, #3976)
- }
- // Set the edge positions
- this.lastItemY = itemMarginTop + this.itemY + itemMarginBottom;
- this.lastLineHeight = Math.max(// #915
- itemHeight, this.lastLineHeight);
- // cache the position of the newly generated or reordered items
- item._legendItemPos = [this.itemX, this.itemY];
- // advance
- if (horizontal) {
- this.itemX += itemWidth;
- }
- else {
- this.itemY +=
- itemMarginTop + itemHeight + itemMarginBottom;
- this.lastLineHeight = itemHeight;
- }
- // the width of the widest item
- this.offsetWidth = this.widthOption || Math.max((horizontal ? this.itemX - padding - (item.checkbox ?
- // decrease by itemDistance only when no checkbox #4853
- 0 :
- itemDistance) : itemWidth) + padding, this.offsetWidth);
- };
- /**
- * Get all items, which is one item per series for most series and one
- * item per point for pie series and its derivatives. Fires the event
- * `afterGetAllItems`.
- *
- * @private
- * @function Highcharts.Legend#getAllItems
- * @return {Array<(Highcharts.BubbleLegend|Highcharts.Point|Highcharts.Series)>}
- * The current items in the legend.
- * @fires Highcharts.Legend#event:afterGetAllItems
- */
- Legend.prototype.getAllItems = function () {
- var allItems = [];
- this.chart.series.forEach(function (series) {
- var seriesOptions = series && series.options;
- // Handle showInLegend. If the series is linked to another series,
- // defaults to false.
- if (series && pick(seriesOptions.showInLegend, !defined(seriesOptions.linkedTo) ? void 0 : false, true)) {
- // Use points or series for the legend item depending on
- // legendType
- allItems = allItems.concat(series.legendItems ||
- (seriesOptions.legendType === 'point' ?
- series.data :
- series));
- }
- });
- fireEvent(this, 'afterGetAllItems', { allItems: allItems });
- return allItems;
- };
- /**
- * Get a short, three letter string reflecting the alignment and layout.
- *
- * @private
- * @function Highcharts.Legend#getAlignment
- * @return {string}
- * The alignment, empty string if floating
- */
- Legend.prototype.getAlignment = function () {
- var options = this.options;
- // Use the first letter of each alignment option in order to detect
- // the side. (#4189 - use charAt(x) notation instead of [x] for IE7)
- if (this.proximate) {
- return options.align.charAt(0) + 'tv';
- }
- return options.floating ? '' : (options.align.charAt(0) +
- options.verticalAlign.charAt(0) +
- options.layout.charAt(0));
- };
- /**
- * Adjust the chart margins by reserving space for the legend on only one
- * side of the chart. If the position is set to a corner, top or bottom is
- * reserved for horizontal legends and left or right for vertical ones.
- *
- * @private
- * @function Highcharts.Legend#adjustMargins
- * @param {Array<number>} margin
- * @param {Array<number>} spacing
- */
- Legend.prototype.adjustMargins = function (margin, spacing) {
- var chart = this.chart,
- options = this.options,
- alignment = this.getAlignment();
- if (alignment) {
- ([
- /(lth|ct|rth)/,
- /(rtv|rm|rbv)/,
- /(rbh|cb|lbh)/,
- /(lbv|lm|ltv)/
- ]).forEach(function (alignments, side) {
- if (alignments.test(alignment) && !defined(margin[side])) {
- // Now we have detected on which side of the chart we should
- // reserve space for the legend
- chart[marginNames[side]] = Math.max(chart[marginNames[side]], (chart.legend[(side + 1) % 2 ? 'legendHeight' : 'legendWidth'] +
- [1, -1, -1, 1][side] * options[(side % 2) ? 'x' : 'y'] +
- pick(options.margin, 12) +
- spacing[side] +
- (chart.titleOffset[side] || 0)));
- }
- });
- }
- };
- /**
- * @private
- * @function Highcharts.Legend#proximatePositions
- */
- Legend.prototype.proximatePositions = function () {
- var chart = this.chart,
- boxes = [],
- alignLeft = this.options.align === 'left';
- this.allItems.forEach(function (item) {
- var lastPoint,
- height,
- useFirstPoint = alignLeft,
- target,
- top;
- if (item.yAxis) {
- if (item.xAxis.options.reversed) {
- useFirstPoint = !useFirstPoint;
- }
- if (item.points) {
- lastPoint = find(useFirstPoint ?
- item.points :
- item.points.slice(0).reverse(), function (item) {
- return isNumber(item.plotY);
- });
- }
- height = this.itemMarginTop +
- item.legendItem.getBBox().height +
- this.itemMarginBottom;
- top = item.yAxis.top - chart.plotTop;
- if (item.visible) {
- target = lastPoint ?
- lastPoint.plotY :
- item.yAxis.height;
- target += top - 0.3 * height;
- }
- else {
- target = top + item.yAxis.height;
- }
- boxes.push({
- target: target,
- size: height,
- item: item
- });
- }
- }, this);
- H.distribute(boxes, chart.plotHeight);
- boxes.forEach(function (box) {
- box.item._legendItemPos[1] =
- chart.plotTop - chart.spacing[0] + box.pos;
- });
- };
- /**
- * Render the legend. This method can be called both before and after
- * `chart.render`. If called after, it will only rearrange items instead
- * of creating new ones. Called internally on initial render and after
- * redraws.
- *
- * @private
- * @function Highcharts.Legend#render
- */
- Legend.prototype.render = function () {
- var legend = this,
- chart = legend.chart,
- renderer = chart.renderer,
- legendGroup = legend.group,
- allItems,
- display,
- legendWidth,
- legendHeight,
- box = legend.box,
- options = legend.options,
- padding = legend.padding,
- allowedWidth;
- legend.itemX = padding;
- legend.itemY = legend.initialItemY;
- legend.offsetWidth = 0;
- legend.lastItemY = 0;
- legend.widthOption = relativeLength(options.width, chart.spacingBox.width - padding);
- // Compute how wide the legend is allowed to be
- allowedWidth =
- chart.spacingBox.width - 2 * padding - options.x;
- if (['rm', 'lm'].indexOf(legend.getAlignment().substring(0, 2)) > -1) {
- allowedWidth /= 2;
- }
- legend.maxLegendWidth = legend.widthOption || allowedWidth;
- if (!legendGroup) {
- /**
- * SVG group of the legend.
- *
- * @readonly
- * @name Highcharts.Legend#group
- * @type {Highcharts.SVGElement}
- */
- legend.group = legendGroup = renderer.g('legend')
- .attr({ zIndex: 7 })
- .add();
- legend.contentGroup = renderer.g()
- .attr({ zIndex: 1 }) // above background
- .add(legendGroup);
- legend.scrollGroup = renderer.g()
- .add(legend.contentGroup);
- }
- legend.renderTitle();
- // add each series or point
- allItems = legend.getAllItems();
- // sort by legendIndex
- stableSort(allItems, function (a, b) {
- return ((a.options && a.options.legendIndex) || 0) -
- ((b.options && b.options.legendIndex) || 0);
- });
- // reversed legend
- if (options.reversed) {
- allItems.reverse();
- }
- /**
- * All items for the legend, which is an array of series for most series
- * and an array of points for pie series and its derivatives.
- *
- * @readonly
- * @name Highcharts.Legend#allItems
- * @type {Array<(Highcharts.Point|Highcharts.Series)>}
- */
- legend.allItems = allItems;
- legend.display = display = !!allItems.length;
- // Render the items. First we run a loop to set the text and properties
- // and read all the bounding boxes. The next loop computes the item
- // positions based on the bounding boxes.
- legend.lastLineHeight = 0;
- legend.maxItemWidth = 0;
- legend.totalItemWidth = 0;
- legend.itemHeight = 0;
- allItems.forEach(legend.renderItem, legend);
- allItems.forEach(legend.layoutItem, legend);
- // Get the box
- legendWidth = (legend.widthOption || legend.offsetWidth) + padding;
- legendHeight = legend.lastItemY + legend.lastLineHeight +
- legend.titleHeight;
- legendHeight = legend.handleOverflow(legendHeight);
- legendHeight += padding;
- // Draw the border and/or background
- if (!box) {
- /**
- * SVG element of the legend box.
- *
- * @readonly
- * @name Highcharts.Legend#box
- * @type {Highcharts.SVGElement}
- */
- legend.box = box = renderer.rect()
- .addClass('highcharts-legend-box')
- .attr({
- r: options.borderRadius
- })
- .add(legendGroup);
- box.isNew = true;
- }
- // Presentational
- if (!chart.styledMode) {
- box
- .attr({
- stroke: options.borderColor,
- 'stroke-width': options.borderWidth || 0,
- fill: options.backgroundColor || 'none'
- })
- .shadow(options.shadow);
- }
- if (legendWidth > 0 && legendHeight > 0) {
- box[box.isNew ? 'attr' : 'animate'](box.crisp.call({}, {
- x: 0,
- y: 0,
- width: legendWidth,
- height: legendHeight
- }, box.strokeWidth()));
- box.isNew = false;
- }
- // hide the border if no items
- box[display ? 'show' : 'hide']();
- // Open for responsiveness
- if (chart.styledMode && legendGroup.getStyle('display') === 'none') {
- legendWidth = legendHeight = 0;
- }
- legend.legendWidth = legendWidth;
- legend.legendHeight = legendHeight;
- if (display) {
- legend.align();
- }
- if (!this.proximate) {
- this.positionItems();
- }
- fireEvent(this, 'afterRender');
- };
- /**
- * Align the legend to chart's box.
- *
- * @private
- * @function Highcharts.align
- * @param {Highcharts.BBoxObject} alignTo
- * @return {void}
- */
- Legend.prototype.align = function (alignTo) {
- if (alignTo === void 0) { alignTo = this.chart.spacingBox; }
- var chart = this.chart,
- options = this.options;
- // If aligning to the top and the layout is horizontal, adjust for
- // the title (#7428)
- var y = alignTo.y;
- if (/(lth|ct|rth)/.test(this.getAlignment()) &&
- chart.titleOffset[0] > 0) {
- y += chart.titleOffset[0];
- }
- else if (/(lbh|cb|rbh)/.test(this.getAlignment()) &&
- chart.titleOffset[2] > 0) {
- y -= chart.titleOffset[2];
- }
- if (y !== alignTo.y) {
- alignTo = merge(alignTo, { y: y });
- }
- this.group.align(merge(options, {
- width: this.legendWidth,
- height: this.legendHeight,
- verticalAlign: this.proximate ? 'top' : options.verticalAlign
- }), true, alignTo);
- };
- /**
- * Set up the overflow handling by adding navigation with up and down arrows
- * below the legend.
- *
- * @private
- * @function Highcharts.Legend#handleOverflow
- * @param {number} legendHeight
- * @return {number}
- */
- Legend.prototype.handleOverflow = function (legendHeight) {
- var legend = this,
- chart = this.chart,
- renderer = chart.renderer,
- options = this.options,
- optionsY = options.y,
- alignTop = options.verticalAlign === 'top',
- padding = this.padding,
- spaceHeight = (chart.spacingBox.height +
- (alignTop ? -optionsY : optionsY) - padding),
- maxHeight = options.maxHeight,
- clipHeight,
- clipRect = this.clipRect,
- navOptions = options.navigation,
- animation = pick(navOptions.animation,
- true),
- arrowSize = navOptions.arrowSize || 12,
- nav = this.nav,
- pages = this.pages,
- lastY,
- allItems = this.allItems,
- clipToHeight = function (height) {
- if (typeof height === 'number') {
- clipRect.attr({
- height: height
- });
- }
- else if (clipRect) { // Reset (#5912)
- legend.clipRect = clipRect.destroy();
- legend.contentGroup.clip();
- }
- // useHTML
- if (legend.contentGroup.div) {
- legend.contentGroup.div.style.clip = height ?
- 'rect(' + padding + 'px,9999px,' +
- (padding + height) + 'px,0)' :
- 'auto';
- }
- }, addTracker = function (key) {
- legend[key] = renderer
- .circle(0, 0, arrowSize * 1.3)
- .translate(arrowSize / 2, arrowSize / 2)
- .add(nav);
- if (!chart.styledMode) {
- legend[key].attr('fill', 'rgba(0,0,0,0.0001)');
- }
- return legend[key];
- };
- // Adjust the height
- if (options.layout === 'horizontal' &&
- options.verticalAlign !== 'middle' &&
- !options.floating) {
- spaceHeight /= 2;
- }
- if (maxHeight) {
- spaceHeight = Math.min(spaceHeight, maxHeight);
- }
- // Reset the legend height and adjust the clipping rectangle
- pages.length = 0;
- if (legendHeight > spaceHeight &&
- navOptions.enabled !== false) {
- this.clipHeight = clipHeight =
- Math.max(spaceHeight - 20 - this.titleHeight - padding, 0);
- this.currentPage = pick(this.currentPage, 1);
- this.fullHeight = legendHeight;
- // Fill pages with Y positions so that the top of each a legend item
- // defines the scroll top for each page (#2098)
- allItems.forEach(function (item, i) {
- var y = item._legendItemPos[1],
- h = Math.round(item.legendItem.getBBox().height),
- len = pages.length;
- if (!len || (y - pages[len - 1] > clipHeight &&
- (lastY || y) !== pages[len - 1])) {
- pages.push(lastY || y);
- len++;
- }
- // Keep track of which page each item is on
- item.pageIx = len - 1;
- if (lastY) {
- allItems[i - 1].pageIx = len - 1;
- }
- if (i === allItems.length - 1 &&
- y + h - pages[len - 1] > clipHeight &&
- y !== lastY // #2617
- ) {
- pages.push(y);
- item.pageIx = len;
- }
- if (y !== lastY) {
- lastY = y;
- }
- });
- // Only apply clipping if needed. Clipping causes blurred legend in
- // PDF export (#1787)
- if (!clipRect) {
- clipRect = legend.clipRect =
- renderer.clipRect(0, padding, 9999, 0);
- legend.contentGroup.clip(clipRect);
- }
- clipToHeight(clipHeight);
- // Add navigation elements
- if (!nav) {
- this.nav = nav = renderer.g()
- .attr({ zIndex: 1 })
- .add(this.group);
- this.up = renderer
- .symbol('triangle', 0, 0, arrowSize, arrowSize)
- .add(nav);
- addTracker('upTracker')
- .on('click', function () {
- legend.scroll(-1, animation);
- });
- this.pager = renderer.text('', 15, 10)
- .addClass('highcharts-legend-navigation');
- if (!chart.styledMode) {
- this.pager.css(navOptions.style);
- }
- this.pager.add(nav);
- this.down = renderer
- .symbol('triangle-down', 0, 0, arrowSize, arrowSize)
- .add(nav);
- addTracker('downTracker')
- .on('click', function () {
- legend.scroll(1, animation);
- });
- }
- // Set initial position
- legend.scroll(0);
- legendHeight = spaceHeight;
- // Reset
- }
- else if (nav) {
- clipToHeight();
- this.nav = nav.destroy(); // #6322
- this.scrollGroup.attr({
- translateY: 1
- });
- this.clipHeight = 0; // #1379
- }
- return legendHeight;
- };
- /**
- * Scroll the legend by a number of pages.
- *
- * @private
- * @function Highcharts.Legend#scroll
- *
- * @param {number} scrollBy
- * The number of pages to scroll.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Whether and how to apply animation.
- *
- * @return {void}
- */
- Legend.prototype.scroll = function (scrollBy, animation) {
- var _this = this;
- var chart = this.chart,
- pages = this.pages,
- pageCount = pages.length,
- currentPage = this.currentPage + scrollBy,
- clipHeight = this.clipHeight,
- navOptions = this.options.navigation,
- pager = this.pager,
- padding = this.padding;
- // When resizing while looking at the last page
- if (currentPage > pageCount) {
- currentPage = pageCount;
- }
- if (currentPage > 0) {
- if (typeof animation !== 'undefined') {
- setAnimation(animation, chart);
- }
- this.nav.attr({
- translateX: padding,
- translateY: clipHeight + this.padding + 7 + this.titleHeight,
- visibility: 'visible'
- });
- [this.up, this.upTracker].forEach(function (elem) {
- elem.attr({
- 'class': currentPage === 1 ?
- 'highcharts-legend-nav-inactive' :
- 'highcharts-legend-nav-active'
- });
- });
- pager.attr({
- text: currentPage + '/' + pageCount
- });
- [this.down, this.downTracker].forEach(function (elem) {
- elem.attr({
- // adjust to text width
- x: 18 + this.pager.getBBox().width,
- 'class': currentPage === pageCount ?
- 'highcharts-legend-nav-inactive' :
- 'highcharts-legend-nav-active'
- });
- }, this);
- if (!chart.styledMode) {
- this.up
- .attr({
- fill: currentPage === 1 ?
- navOptions.inactiveColor :
- navOptions.activeColor
- });
- this.upTracker
- .css({
- cursor: currentPage === 1 ? 'default' : 'pointer'
- });
- this.down
- .attr({
- fill: currentPage === pageCount ?
- navOptions.inactiveColor :
- navOptions.activeColor
- });
- this.downTracker
- .css({
- cursor: currentPage === pageCount ?
- 'default' :
- 'pointer'
- });
- }
- this.scrollOffset = -pages[currentPage - 1] + this.initialItemY;
- this.scrollGroup.animate({
- translateY: this.scrollOffset
- });
- this.currentPage = currentPage;
- this.positionCheckboxes();
- // Fire event after scroll animation is complete
- var animOptions = animObject(pick(animation,
- chart.renderer.globalAnimation,
- true));
- syncTimeout(function () {
- fireEvent(_this, 'afterScroll', { currentPage: currentPage });
- }, animOptions.duration);
- }
- };
- return Legend;
- }());
- // Workaround for #2030, horizontal legend items not displaying in IE11 Preview,
- // and for #2580, a similar drawing flaw in Firefox 26.
- // Explore if there's a general cause for this. The problem may be related
- // to nested group elements, as the legend item texts are within 4 group
- // elements.
- if (/Trident\/7\.0/.test(win.navigator && win.navigator.userAgent) ||
- isFirefox) {
- wrap(Legend.prototype, 'positionItem', function (proceed, item) {
- var legend = this,
- // If chart destroyed in sync, this is undefined (#2030)
- runPositionItem = function () {
- if (item._legendItemPos) {
- proceed.call(legend,
- item);
- }
- };
- // Do it now, for export and to get checkbox placement
- runPositionItem();
- // Do it after to work around the core issue
- if (!legend.bubbleLegend) {
- setTimeout(runPositionItem);
- }
- });
- }
- H.Legend = Legend;
- return H.Legend;
- });
- _registerModule(_modules, 'Core/Chart/Chart.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Legend.js'], _modules['Core/MSPointer.js'], _modules['Core/Options.js'], _modules['Core/Pointer.js'], _modules['Core/Time.js'], _modules['Core/Utilities.js']], function (Axis, H, Legend, MSPointer, O, Pointer, Time, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var charts = H.charts,
- doc = H.doc,
- seriesTypes = H.seriesTypes,
- win = H.win;
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- animate = U.animate,
- animObject = U.animObject,
- attr = U.attr,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- discardElement = U.discardElement,
- erase = U.erase,
- error = U.error,
- extend = U.extend,
- find = U.find,
- fireEvent = U.fireEvent,
- getStyle = U.getStyle,
- isArray = U.isArray,
- isFunction = U.isFunction,
- isNumber = U.isNumber,
- isObject = U.isObject,
- isString = U.isString,
- merge = U.merge,
- numberFormat = U.numberFormat,
- objectEach = U.objectEach,
- pick = U.pick,
- pInt = U.pInt,
- relativeLength = U.relativeLength,
- removeEvent = U.removeEvent,
- setAnimation = U.setAnimation,
- splat = U.splat,
- syncTimeout = U.syncTimeout,
- uniqueKey = U.uniqueKey;
- /**
- * Callback for chart constructors.
- *
- * @callback Highcharts.ChartCallbackFunction
- *
- * @param {Highcharts.Chart} chart
- * Created chart.
- */
- /**
- * Format a number and return a string based on input settings.
- *
- * @callback Highcharts.NumberFormatterCallbackFunction
- *
- * @param {number} number
- * The input number to format.
- *
- * @param {number} decimals
- * The amount of decimals. A value of -1 preserves the amount in the
- * input number.
- *
- * @param {string} [decimalPoint]
- * The decimal point, defaults to the one given in the lang options, or
- * a dot.
- *
- * @param {string} [thousandsSep]
- * The thousands separator, defaults to the one given in the lang
- * options, or a space character.
- *
- * @return {string} The formatted number.
- */
- /**
- * The chart title. The title has an `update` method that allows modifying the
- * options directly or indirectly via `chart.update`.
- *
- * @interface Highcharts.TitleObject
- * @extends Highcharts.SVGElement
- */ /**
- * Modify options for the title.
- *
- * @function Highcharts.TitleObject#update
- *
- * @param {Highcharts.TitleOptions} titleOptions
- * Options to modify.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the title is altered. If doing more
- * operations on the chart, it is a good idea to set redraw to false and
- * call {@link Chart#redraw} after.
- */
- /**
- * The chart subtitle. The subtitle has an `update` method that
- * allows modifying the options directly or indirectly via
- * `chart.update`.
- *
- * @interface Highcharts.SubtitleObject
- * @extends Highcharts.SVGElement
- */ /**
- * Modify options for the subtitle.
- *
- * @function Highcharts.SubtitleObject#update
- *
- * @param {Highcharts.SubtitleOptions} subtitleOptions
- * Options to modify.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the subtitle is altered. If doing
- * more operations on the chart, it is a good idea to set redraw to false
- * and call {@link Chart#redraw} after.
- */
- /**
- * The chart caption. The caption has an `update` method that
- * allows modifying the options directly or indirectly via
- * `chart.update`.
- *
- * @interface Highcharts.CaptionObject
- * @extends Highcharts.SVGElement
- */ /**
- * Modify options for the caption.
- *
- * @function Highcharts.CaptionObject#update
- *
- * @param {Highcharts.CaptionOptions} captionOptions
- * Options to modify.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the caption is altered. If doing
- * more operations on the chart, it is a good idea to set redraw to false
- * and call {@link Chart#redraw} after.
- */
- var marginNames = H.marginNames;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Chart class. The recommended constructor is {@link Highcharts#chart}.
- *
- * @example
- * var chart = Highcharts.chart('container', {
- * title: {
- * text: 'My chart'
- * },
- * series: [{
- * data: [1, 3, 2, 4]
- * }]
- * })
- *
- * @class
- * @name Highcharts.Chart
- *
- * @param {string|Highcharts.HTMLDOMElement} [renderTo]
- * The DOM element to render to, or its id.
- *
- * @param {Highcharts.Options} options
- * The chart options structure.
- *
- * @param {Highcharts.ChartCallbackFunction} [callback]
- * Function to run when the chart has loaded and and all external images
- * are loaded. Defining a
- * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load)
- * handler is equivalent.
- */
- var Chart = /** @class */ (function () {
- function Chart(a, b, c) {
- this.axes = void 0;
- this.axisOffset = void 0;
- this.bounds = void 0;
- this.chartHeight = void 0;
- this.chartWidth = void 0;
- this.clipBox = void 0;
- this.colorCounter = void 0;
- this.container = void 0;
- this.index = void 0;
- this.isResizing = void 0;
- this.labelCollectors = void 0;
- this.legend = void 0;
- this.margin = void 0;
- this.numberFormatter = void 0;
- this.options = void 0;
- this.plotBox = void 0;
- this.plotHeight = void 0;
- this.plotLeft = void 0;
- this.plotTop = void 0;
- this.plotWidth = void 0;
- this.pointCount = void 0;
- this.pointer = void 0;
- this.renderer = void 0;
- this.renderTo = void 0;
- this.series = void 0;
- this.spacing = void 0;
- this.spacingBox = void 0;
- this.symbolCounter = void 0;
- this.time = void 0;
- this.titleOffset = void 0;
- this.userOptions = void 0;
- this.xAxis = void 0;
- this.yAxis = void 0;
- this.getArgs(a, b, c);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Handle the arguments passed to the constructor.
- *
- * @private
- * @function Highcharts.Chart#getArgs
- *
- * @param {...Array<*>} arguments
- * All arguments for the constructor.
- *
- * @fires Highcharts.Chart#event:init
- * @fires Highcharts.Chart#event:afterInit
- */
- Chart.prototype.getArgs = function (a, b, c) {
- // Remove the optional first argument, renderTo, and
- // set it on this.
- if (isString(a) || a.nodeName) {
- this.renderTo = a;
- this.init(b, c);
- }
- else {
- this.init(a, b);
- }
- };
- /**
- * Overridable function that initializes the chart. The constructor's
- * arguments are passed on directly.
- *
- * @function Highcharts.Chart#init
- *
- * @param {Highcharts.Options} userOptions
- * Custom options.
- *
- * @param {Function} [callback]
- * Function to run when the chart has loaded and and all external
- * images are loaded.
- *
- * @return {void}
- *
- * @fires Highcharts.Chart#event:init
- * @fires Highcharts.Chart#event:afterInit
- */
- Chart.prototype.init = function (userOptions, callback) {
- // Handle regular options
- var options,
- // skip merging data points to increase performance
- seriesOptions = userOptions.series,
- userPlotOptions = userOptions.plotOptions || {};
- // Fire the event with a default function
- fireEvent(this, 'init', { args: arguments }, function () {
- userOptions.series = null;
- options = merge(defaultOptions, userOptions); // do the merge
- var optionsChart = options.chart || {};
- // Override (by copy of user options) or clear tooltip options
- // in chart.options.plotOptions (#6218)
- objectEach(options.plotOptions, function (typeOptions, type) {
- if (isObject(typeOptions)) { // #8766
- typeOptions.tooltip = (userPlotOptions[type] && // override by copy:
- merge(userPlotOptions[type].tooltip)) || void 0; // or clear
- }
- });
- // User options have higher priority than default options
- // (#6218). In case of exporting: path is changed
- options.tooltip.userOptions = (userOptions.chart &&
- userOptions.chart.forExport &&
- userOptions.tooltip.userOptions) || userOptions.tooltip;
- // set back the series data
- options.series = userOptions.series = seriesOptions;
- /**
- * The original options given to the constructor or a chart factory
- * like {@link Highcharts.chart} and {@link Highcharts.stockChart}.
- *
- * @name Highcharts.Chart#userOptions
- * @type {Highcharts.Options}
- */
- this.userOptions = userOptions;
- var chartEvents = optionsChart.events;
- this.margin = [];
- this.spacing = [];
- // Pixel data bounds for touch zoom
- this.bounds = { h: {}, v: {} };
- // An array of functions that returns labels that should be
- // considered for anti-collision
- this.labelCollectors = [];
- this.callback = callback;
- this.isResizing = 0;
- /**
- * The options structure for the chart after merging
- * {@link #defaultOptions} and {@link #userOptions}. It contains
- * members for the sub elements like series, legend, tooltip etc.
- *
- * @name Highcharts.Chart#options
- * @type {Highcharts.Options}
- */
- this.options = options;
- /**
- * All the axes in the chart.
- *
- * @see Highcharts.Chart.xAxis
- * @see Highcharts.Chart.yAxis
- *
- * @name Highcharts.Chart#axes
- * @type {Array<Highcharts.Axis>}
- */
- this.axes = [];
- /**
- * All the current series in the chart.
- *
- * @name Highcharts.Chart#series
- * @type {Array<Highcharts.Series>}
- */
- this.series = [];
- /**
- * The `Time` object associated with the chart. Since v6.0.5,
- * time settings can be applied individually for each chart. If
- * no individual settings apply, the `Time` object is shared by
- * all instances.
- *
- * @name Highcharts.Chart#time
- * @type {Highcharts.Time}
- */
- this.time =
- userOptions.time && Object.keys(userOptions.time).length ?
- new Time(userOptions.time) :
- H.time;
- /**
- * Callback function to override the default function that formats
- * all the numbers in the chart. Returns a string with the formatted
- * number.
- *
- * @name Highcharts.Chart#numberFormatter
- * @type {Highcharts.NumberFormatterCallbackFunction}
- */
- this.numberFormatter = optionsChart.numberFormatter || numberFormat;
- /**
- * Whether the chart is in styled mode, meaning all presentatinoal
- * attributes are avoided.
- *
- * @name Highcharts.Chart#styledMode
- * @type {boolean}
- */
- this.styledMode = optionsChart.styledMode;
- this.hasCartesianSeries = optionsChart.showAxes;
- var chart = this;
- /**
- * Index position of the chart in the {@link Highcharts#charts}
- * property.
- *
- * @name Highcharts.Chart#index
- * @type {number}
- * @readonly
- */
- chart.index = charts.length; // Add the chart to the global lookup
- charts.push(chart);
- H.chartCount++;
- // Chart event handlers
- if (chartEvents) {
- objectEach(chartEvents, function (event, eventType) {
- if (isFunction(event)) {
- addEvent(chart, eventType, event);
- }
- });
- }
- /**
- * A collection of the X axes in the chart.
- *
- * @name Highcharts.Chart#xAxis
- * @type {Array<Highcharts.Axis>}
- */
- chart.xAxis = [];
- /**
- * A collection of the Y axes in the chart.
- *
- * @name Highcharts.Chart#yAxis
- * @type {Array<Highcharts.Axis>}
- *
- * @todo
- * Make events official: Fire the event `afterInit`.
- */
- chart.yAxis = [];
- chart.pointCount = chart.colorCounter = chart.symbolCounter = 0;
- // Fire after init but before first render, before axes and series
- // have been initialized.
- fireEvent(chart, 'afterInit');
- chart.firstRender();
- });
- };
- /**
- * Internal function to unitialize an individual series.
- *
- * @private
- * @function Highcharts.Chart#initSeries
- */
- Chart.prototype.initSeries = function (options) {
- var chart = this,
- optionsChart = chart.options.chart,
- type = (options.type ||
- optionsChart.type ||
- optionsChart.defaultSeriesType),
- series,
- Constr = seriesTypes[type];
- // No such series type
- if (!Constr) {
- error(17, true, chart, { missingModuleFor: type });
- }
- series = new Constr();
- series.init(this, options);
- return series;
- };
- /**
- * Internal function to set data for all series with enabled sorting.
- *
- * @private
- * @function Highcharts.Chart#setSeriesData
- */
- Chart.prototype.setSeriesData = function () {
- this.getSeriesOrderByLinks().forEach(function (series) {
- // We need to set data for series with sorting after series init
- if (!series.points && !series.data && series.enabledDataSorting) {
- series.setData(series.options.data, false);
- }
- });
- };
- /**
- * Sort and return chart series in order depending on the number of linked
- * series.
- *
- * @private
- * @function Highcharts.Series#getSeriesOrderByLinks
- * @return {Array<Highcharts.Series>}
- */
- Chart.prototype.getSeriesOrderByLinks = function () {
- return this.series.concat().sort(function (a, b) {
- if (a.linkedSeries.length || b.linkedSeries.length) {
- return b.linkedSeries.length - a.linkedSeries.length;
- }
- return 0;
- });
- };
- /**
- * Order all series above a given index. When series are added and ordered
- * by configuration, only the last series is handled (#248, #1123, #2456,
- * #6112). This function is called on series initialization and destroy.
- *
- * @private
- * @function Highcharts.Series#orderSeries
- * @param {number} [fromIndex]
- * If this is given, only the series above this index are handled.
- */
- Chart.prototype.orderSeries = function (fromIndex) {
- var series = this.series,
- i = fromIndex || 0;
- for (; i < series.length; i++) {
- if (series[i]) {
- /**
- * Contains the series' index in the `Chart.series` array.
- *
- * @name Highcharts.Series#index
- * @type {number}
- * @readonly
- */
- series[i].index = i;
- series[i].name = series[i].getName();
- }
- }
- };
- /**
- * Check whether a given point is within the plot area.
- *
- * @function Highcharts.Chart#isInsidePlot
- *
- * @param {number} plotX
- * Pixel x relative to the plot area.
- *
- * @param {number} plotY
- * Pixel y relative to the plot area.
- *
- * @param {boolean} [inverted]
- * Whether the chart is inverted.
- *
- * @return {boolean}
- * Returns true if the given point is inside the plot area.
- */
- Chart.prototype.isInsidePlot = function (plotX, plotY, inverted) {
- var x = inverted ? plotY : plotX,
- y = inverted ? plotX : plotY,
- e = {
- x: x,
- y: y,
- isInsidePlot: x >= 0 &&
- x <= this.plotWidth &&
- y >= 0 &&
- y <= this.plotHeight
- };
- fireEvent(this, 'afterIsInsidePlot', e);
- return e.isInsidePlot;
- };
- /**
- * Redraw the chart after changes have been done to the data, axis extremes
- * chart size or chart elements. All methods for updating axes, series or
- * points have a parameter for redrawing the chart. This is `true` by
- * default. But in many cases you want to do more than one operation on the
- * chart before redrawing, for example add a number of points. In those
- * cases it is a waste of resources to redraw the chart for each new point
- * added. So you add the points and call `chart.redraw()` after.
- *
- * @function Highcharts.Chart#redraw
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * If or how to apply animation to the redraw.
- *
- * @fires Highcharts.Chart#event:afterSetExtremes
- * @fires Highcharts.Chart#event:beforeRedraw
- * @fires Highcharts.Chart#event:predraw
- * @fires Highcharts.Chart#event:redraw
- * @fires Highcharts.Chart#event:render
- * @fires Highcharts.Chart#event:updatedData
- */
- Chart.prototype.redraw = function (animation) {
- fireEvent(this, 'beforeRedraw');
- var chart = this,
- axes = chart.axes,
- series = chart.series,
- pointer = chart.pointer,
- legend = chart.legend,
- legendUserOptions = chart.userOptions.legend,
- redrawLegend = chart.isDirtyLegend,
- hasStackedSeries,
- hasDirtyStacks,
- hasCartesianSeries = chart.hasCartesianSeries,
- isDirtyBox = chart.isDirtyBox,
- i,
- serie,
- renderer = chart.renderer,
- isHiddenChart = renderer.isHidden(),
- afterRedraw = [];
- // Handle responsive rules, not only on resize (#6130)
- if (chart.setResponsive) {
- chart.setResponsive(false);
- }
- // Set the global animation. When chart.hasRendered is not true, the
- // redraw call comes from a responsive rule and animation should not
- // occur.
- setAnimation(chart.hasRendered ? animation : false, chart);
- if (isHiddenChart) {
- chart.temporaryDisplay();
- }
- // Adjust title layout (reflow multiline text)
- chart.layOutTitles();
- // link stacked series
- i = series.length;
- while (i--) {
- serie = series[i];
- if (serie.options.stacking) {
- hasStackedSeries = true;
- if (serie.isDirty) {
- hasDirtyStacks = true;
- break;
- }
- }
- }
- if (hasDirtyStacks) { // mark others as dirty
- i = series.length;
- while (i--) {
- serie = series[i];
- if (serie.options.stacking) {
- serie.isDirty = true;
- }
- }
- }
- // Handle updated data in the series
- series.forEach(function (serie) {
- if (serie.isDirty) {
- if (serie.options.legendType === 'point') {
- if (serie.updateTotals) {
- serie.updateTotals();
- }
- redrawLegend = true;
- }
- else if (legendUserOptions &&
- (legendUserOptions.labelFormatter ||
- legendUserOptions.labelFormat)) {
- redrawLegend = true; // #2165
- }
- }
- if (serie.isDirtyData) {
- fireEvent(serie, 'updatedData');
- }
- });
- // handle added or removed series
- if (redrawLegend && legend && legend.options.enabled) {
- // draw legend graphics
- legend.render();
- chart.isDirtyLegend = false;
- }
- // reset stacks
- if (hasStackedSeries) {
- chart.getStacks();
- }
- if (hasCartesianSeries) {
- // set axes scales
- axes.forEach(function (axis) {
- // Don't do setScale again if we're only resizing. Regression
- // #13507. But we need it after chart.update (responsive), as
- // axis is initialized again (#12137).
- if (!chart.isResizing || !isNumber(axis.min)) {
- axis.updateNames();
- axis.setScale();
- }
- });
- }
- chart.getMargins(); // #3098
- if (hasCartesianSeries) {
- // If one axis is dirty, all axes must be redrawn (#792, #2169)
- axes.forEach(function (axis) {
- if (axis.isDirty) {
- isDirtyBox = true;
- }
- });
- // redraw axes
- axes.forEach(function (axis) {
- // Fire 'afterSetExtremes' only if extremes are set
- var key = axis.min + ',' + axis.max;
- if (axis.extKey !== key) { // #821, #4452
- axis.extKey = key;
- // prevent a recursive call to chart.redraw() (#1119)
- afterRedraw.push(function () {
- fireEvent(axis, 'afterSetExtremes', extend(axis.eventArgs, axis.getExtremes())); // #747, #751
- delete axis.eventArgs;
- });
- }
- if (isDirtyBox || hasStackedSeries) {
- axis.redraw();
- }
- });
- }
- // the plot areas size has changed
- if (isDirtyBox) {
- chart.drawChartBox();
- }
- // Fire an event before redrawing series, used by the boost module to
- // clear previous series renderings.
- fireEvent(chart, 'predraw');
- // redraw affected series
- series.forEach(function (serie) {
- if ((isDirtyBox || serie.isDirty) && serie.visible) {
- serie.redraw();
- }
- // Set it here, otherwise we will have unlimited 'updatedData' calls
- // for a hidden series after setData(). Fixes #6012
- serie.isDirtyData = false;
- });
- // move tooltip or reset
- if (pointer) {
- pointer.reset(true);
- }
- // redraw if canvas
- renderer.draw();
- // Fire the events
- fireEvent(chart, 'redraw');
- fireEvent(chart, 'render');
- if (isHiddenChart) {
- chart.temporaryDisplay(true);
- }
- // Fire callbacks that are put on hold until after the redraw
- afterRedraw.forEach(function (callback) {
- callback.call();
- });
- };
- /**
- * Get an axis, series or point object by `id` as given in the configuration
- * options. Returns `undefined` if no item is found.
- *
- * @sample highcharts/plotoptions/series-id/
- * Get series by id
- *
- * @function Highcharts.Chart#get
- *
- * @param {string} id
- * The id as given in the configuration options.
- *
- * @return {Highcharts.Axis|Highcharts.Series|Highcharts.Point|undefined}
- * The retrieved item.
- */
- Chart.prototype.get = function (id) {
- var ret,
- series = this.series,
- i;
- /**
- * @private
- * @param {Highcharts.Axis|Highcharts.Series} item
- * @return {boolean}
- */
- function itemById(item) {
- return (item.id === id ||
- (item.options && item.options.id === id));
- }
- ret =
- // Search axes
- find(this.axes, itemById) ||
- // Search series
- find(this.series, itemById);
- // Search points
- for (i = 0; !ret && i < series.length; i++) {
- ret = find(series[i].points || [], itemById);
- }
- return ret;
- };
- /**
- * Create the Axis instances based on the config options.
- *
- * @private
- * @function Highcharts.Chart#getAxes
- * @fires Highcharts.Chart#event:afterGetAxes
- * @fires Highcharts.Chart#event:getAxes
- */
- Chart.prototype.getAxes = function () {
- var chart = this,
- options = this.options,
- xAxisOptions = options.xAxis = splat(options.xAxis || {}),
- yAxisOptions = options.yAxis = splat(options.yAxis || {}),
- optionsArray;
- fireEvent(this, 'getAxes');
- // make sure the options are arrays and add some members
- xAxisOptions.forEach(function (axis, i) {
- axis.index = i;
- axis.isX = true;
- });
- yAxisOptions.forEach(function (axis, i) {
- axis.index = i;
- });
- // concatenate all axis options into one array
- optionsArray = xAxisOptions.concat(yAxisOptions);
- optionsArray.forEach(function (axisOptions) {
- new Axis(chart, axisOptions); // eslint-disable-line no-new
- });
- fireEvent(this, 'afterGetAxes');
- };
- /**
- * Returns an array of all currently selected points in the chart. Points
- * can be selected by clicking or programmatically by the
- * {@link Highcharts.Point#select}
- * function.
- *
- * @sample highcharts/plotoptions/series-allowpointselect-line/
- * Get selected points
- *
- * @function Highcharts.Chart#getSelectedPoints
- *
- * @return {Array<Highcharts.Point>}
- * The currently selected points.
- */
- Chart.prototype.getSelectedPoints = function () {
- var points = [];
- this.series.forEach(function (serie) {
- // For one-to-one points inspect series.data in order to retrieve
- // points outside the visible range (#6445). For grouped data,
- // inspect the generated series.points.
- points = points.concat(serie.getPointsCollection().filter(function (point) {
- return pick(point.selectedStaging, point.selected);
- }));
- });
- return points;
- };
- /**
- * Returns an array of all currently selected series in the chart. Series
- * can be selected either programmatically by the
- * {@link Highcharts.Series#select}
- * function or by checking the checkbox next to the legend item if
- * [series.showCheckBox](https://api.highcharts.com/highcharts/plotOptions.series.showCheckbox)
- * is true.
- *
- * @sample highcharts/members/chart-getselectedseries/
- * Get selected series
- *
- * @function Highcharts.Chart#getSelectedSeries
- *
- * @return {Array<Highcharts.Series>}
- * The currently selected series.
- */
- Chart.prototype.getSelectedSeries = function () {
- return this.series.filter(function (serie) {
- return serie.selected;
- });
- };
- /**
- * Set a new title or subtitle for the chart.
- *
- * @sample highcharts/members/chart-settitle/
- * Set title text and styles
- *
- * @function Highcharts.Chart#setTitle
- *
- * @param {Highcharts.TitleOptions} [titleOptions]
- * New title options. The title text itself is set by the
- * `titleOptions.text` property.
- *
- * @param {Highcharts.SubtitleOptions} [subtitleOptions]
- * New subtitle options. The subtitle text itself is set by the
- * `subtitleOptions.text` property.
- *
- * @param {boolean} [redraw]
- * Whether to redraw the chart or wait for a later call to
- * `chart.redraw()`.
- */
- Chart.prototype.setTitle = function (titleOptions, subtitleOptions, redraw) {
- this.applyDescription('title', titleOptions);
- this.applyDescription('subtitle', subtitleOptions);
- // The initial call also adds the caption. On update, chart.update will
- // relay to Chart.setCaption.
- this.applyDescription('caption', void 0);
- this.layOutTitles(redraw);
- };
- /**
- * Apply a title, subtitle or caption for the chart
- *
- * @private
- * @function Highcharts.Chart#applyDescription
- * @param name {string}
- * Either title, subtitle or caption
- * @param {Highcharts.TitleOptions|Highcharts.SubtitleOptions|Highcharts.CaptionOptions|undefined} explicitOptions
- * The options to set, will be merged with default options.
- */
- Chart.prototype.applyDescription = function (name, explicitOptions) {
- var chart = this;
- // Default style
- var style = name === 'title' ? {
- color: '#333333',
- fontSize: this.options.isStock ? '16px' : '18px' // #2944
- } : {
- color: '#666666'
- };
- // Merge default options with explicit options
- var options = this.options[name] = merge(
- // Default styles
- (!this.styledMode && { style: style }),
- this.options[name],
- explicitOptions);
- var elem = this[name];
- if (elem && explicitOptions) {
- this[name] = elem = elem.destroy(); // remove old
- }
- if (options && !elem) {
- elem = this.renderer.text(options.text, 0, 0, options.useHTML)
- .attr({
- align: options.align,
- 'class': 'highcharts-' + name,
- zIndex: options.zIndex || 4
- })
- .add();
- // Update methods, shortcut to Chart.setTitle, Chart.setSubtitle and
- // Chart.setCaption
- elem.update = function (updateOptions) {
- var fn = {
- title: 'setTitle',
- subtitle: 'setSubtitle',
- caption: 'setCaption'
- }[name];
- chart[fn](updateOptions);
- };
- // Presentational
- if (!this.styledMode) {
- elem.css(options.style);
- }
- /**
- * The chart title. The title has an `update` method that allows
- * modifying the options directly or indirectly via
- * `chart.update`.
- *
- * @sample highcharts/members/title-update/
- * Updating titles
- *
- * @name Highcharts.Chart#title
- * @type {Highcharts.TitleObject}
- */
- /**
- * The chart subtitle. The subtitle has an `update` method that
- * allows modifying the options directly or indirectly via
- * `chart.update`.
- *
- * @name Highcharts.Chart#subtitle
- * @type {Highcharts.SubtitleObject}
- */
- this[name] = elem;
- }
- };
- /**
- * Internal function to lay out the chart title, subtitle and caption, and
- * cache the full offset height for use in `getMargins`. The result is
- * stored in `this.titleOffset`.
- *
- * @private
- * @function Highcharts.Chart#layOutTitles
- *
- * @param {boolean} [redraw=true]
- * @fires Highcharts.Chart#event:afterLayOutTitles
- */
- Chart.prototype.layOutTitles = function (redraw) {
- var titleOffset = [0, 0, 0],
- requiresDirtyBox,
- renderer = this.renderer,
- spacingBox = this.spacingBox;
- // Lay out the title and the subtitle respectively
- ['title', 'subtitle', 'caption'].forEach(function (key) {
- var title = this[key], titleOptions = this.options[key], verticalAlign = titleOptions.verticalAlign || 'top', offset = key === 'title' ? -3 :
- // Floating subtitle (#6574)
- verticalAlign === 'top' ? titleOffset[0] + 2 : 0, titleSize, height;
- if (title) {
- if (!this.styledMode) {
- titleSize = titleOptions.style.fontSize;
- }
- titleSize = renderer.fontMetrics(titleSize, title).b;
- title
- .css({
- width: (titleOptions.width ||
- spacingBox.width + (titleOptions.widthAdjust || 0)) + 'px'
- });
- // Skip the cache for HTML (#3481, #11666)
- height = Math.round(title.getBBox(titleOptions.useHTML).height);
- title.align(extend({
- y: verticalAlign === 'bottom' ?
- titleSize :
- offset + titleSize,
- height: height
- }, titleOptions), false, 'spacingBox');
- if (!titleOptions.floating) {
- if (verticalAlign === 'top') {
- titleOffset[0] = Math.ceil(titleOffset[0] +
- height);
- }
- else if (verticalAlign === 'bottom') {
- titleOffset[2] = Math.ceil(titleOffset[2] +
- height);
- }
- }
- }
- }, this);
- // Handle title.margin and caption.margin
- if (titleOffset[0] &&
- (this.options.title.verticalAlign || 'top') === 'top') {
- titleOffset[0] += this.options.title.margin;
- }
- if (titleOffset[2] &&
- this.options.caption.verticalAlign === 'bottom') {
- titleOffset[2] += this.options.caption.margin;
- }
- requiresDirtyBox = (!this.titleOffset ||
- this.titleOffset.join(',') !== titleOffset.join(','));
- // Used in getMargins
- this.titleOffset = titleOffset;
- fireEvent(this, 'afterLayOutTitles');
- if (!this.isDirtyBox && requiresDirtyBox) {
- this.isDirtyBox = this.isDirtyLegend = requiresDirtyBox;
- // Redraw if necessary (#2719, #2744)
- if (this.hasRendered && pick(redraw, true) && this.isDirtyBox) {
- this.redraw();
- }
- }
- };
- /**
- * Internal function to get the chart width and height according to options
- * and container size. Sets {@link Chart.chartWidth} and
- * {@link Chart.chartHeight}.
- *
- * @private
- * @function Highcharts.Chart#getChartSize
- */
- Chart.prototype.getChartSize = function () {
- var chart = this,
- optionsChart = chart.options.chart,
- widthOption = optionsChart.width,
- heightOption = optionsChart.height,
- renderTo = chart.renderTo;
- // Get inner width and height
- if (!defined(widthOption)) {
- chart.containerWidth = getStyle(renderTo, 'width');
- }
- if (!defined(heightOption)) {
- chart.containerHeight = getStyle(renderTo, 'height');
- }
- /**
- * The current pixel width of the chart.
- *
- * @name Highcharts.Chart#chartWidth
- * @type {number}
- */
- chart.chartWidth = Math.max(// #1393
- 0, widthOption || chart.containerWidth || 600 // #1460
- );
- /**
- * The current pixel height of the chart.
- *
- * @name Highcharts.Chart#chartHeight
- * @type {number}
- */
- chart.chartHeight = Math.max(0, relativeLength(heightOption, chart.chartWidth) ||
- (chart.containerHeight > 1 ?
- chart.containerHeight :
- 400));
- };
- /**
- * If the renderTo element has no offsetWidth, most likely one or more of
- * its parents are hidden. Loop up the DOM tree to temporarily display the
- * parents, then save the original display properties, and when the true
- * size is retrieved, reset them. Used on first render and on redraws.
- *
- * @private
- * @function Highcharts.Chart#temporaryDisplay
- *
- * @param {boolean} [revert]
- * Revert to the saved original styles.
- */
- Chart.prototype.temporaryDisplay = function (revert) {
- var node = this.renderTo,
- tempStyle;
- if (!revert) {
- while (node && node.style) {
- // When rendering to a detached node, it needs to be temporarily
- // attached in order to read styling and bounding boxes (#5783,
- // #7024).
- if (!doc.body.contains(node) && !node.parentNode) {
- node.hcOrigDetached = true;
- doc.body.appendChild(node);
- }
- if (getStyle(node, 'display', false) === 'none' ||
- node.hcOricDetached) {
- node.hcOrigStyle = {
- display: node.style.display,
- height: node.style.height,
- overflow: node.style.overflow
- };
- tempStyle = {
- display: 'block',
- overflow: 'hidden'
- };
- if (node !== this.renderTo) {
- tempStyle.height = 0;
- }
- css(node, tempStyle);
- // If it still doesn't have an offset width after setting
- // display to block, it probably has an !important priority
- // #2631, 6803
- if (!node.offsetWidth) {
- node.style.setProperty('display', 'block', 'important');
- }
- }
- node = node.parentNode;
- if (node === doc.body) {
- break;
- }
- }
- }
- else {
- while (node && node.style) {
- if (node.hcOrigStyle) {
- css(node, node.hcOrigStyle);
- delete node.hcOrigStyle;
- }
- if (node.hcOrigDetached) {
- doc.body.removeChild(node);
- node.hcOrigDetached = false;
- }
- node = node.parentNode;
- }
- }
- };
- /**
- * Set the {@link Chart.container|chart container's} class name, in
- * addition to `highcharts-container`.
- *
- * @function Highcharts.Chart#setClassName
- *
- * @param {string} [className]
- * The additional class name.
- */
- Chart.prototype.setClassName = function (className) {
- this.container.className = 'highcharts-container ' + (className || '');
- };
- /**
- * Get the containing element, determine the size and create the inner
- * container div to hold the chart.
- *
- * @private
- * @function Highcharts.Chart#afterGetContainer
- * @fires Highcharts.Chart#event:afterGetContainer
- */
- Chart.prototype.getContainer = function () {
- var chart = this,
- container,
- options = chart.options,
- optionsChart = options.chart,
- chartWidth,
- chartHeight,
- renderTo = chart.renderTo,
- indexAttrName = 'data-highcharts-chart',
- oldChartIndex,
- Ren,
- containerId = uniqueKey(),
- containerStyle,
- key;
- if (!renderTo) {
- chart.renderTo = renderTo =
- optionsChart.renderTo;
- }
- if (isString(renderTo)) {
- chart.renderTo = renderTo =
- doc.getElementById(renderTo);
- }
- // Display an error if the renderTo is wrong
- if (!renderTo) {
- error(13, true, chart);
- }
- // If the container already holds a chart, destroy it. The check for
- // hasRendered is there because web pages that are saved to disk from
- // the browser, will preserve the data-highcharts-chart attribute and
- // the SVG contents, but not an interactive chart. So in this case,
- // charts[oldChartIndex] will point to the wrong chart if any (#2609).
- oldChartIndex = pInt(attr(renderTo, indexAttrName));
- if (isNumber(oldChartIndex) &&
- charts[oldChartIndex] &&
- charts[oldChartIndex].hasRendered) {
- charts[oldChartIndex].destroy();
- }
- // Make a reference to the chart from the div
- attr(renderTo, indexAttrName, chart.index);
- // remove previous chart
- renderTo.innerHTML = '';
- // If the container doesn't have an offsetWidth, it has or is a child of
- // a node that has display:none. We need to temporarily move it out to a
- // visible state to determine the size, else the legend and tooltips
- // won't render properly. The skipClone option is used in sparklines as
- // a micro optimization, saving about 1-2 ms each chart.
- if (!optionsChart.skipClone && !renderTo.offsetWidth) {
- chart.temporaryDisplay();
- }
- // get the width and height
- chart.getChartSize();
- chartWidth = chart.chartWidth;
- chartHeight = chart.chartHeight;
- // Allow table cells and flex-boxes to shrink without the chart blocking
- // them out (#6427)
- css(renderTo, { overflow: 'hidden' });
- // Create the inner container
- if (!chart.styledMode) {
- containerStyle = extend({
- position: 'relative',
- // needed for context menu (avoidscrollbars) and content
- // overflow in IE
- overflow: 'hidden',
- width: chartWidth + 'px',
- height: chartHeight + 'px',
- textAlign: 'left',
- lineHeight: 'normal',
- zIndex: 0,
- '-webkit-tap-highlight-color': 'rgba(0,0,0,0)',
- userSelect: 'none' // #13503
- }, optionsChart.style);
- }
- /**
- * The containing HTML element of the chart. The container is
- * dynamically inserted into the element given as the `renderTo`
- * parameter in the {@link Highcharts#chart} constructor.
- *
- * @name Highcharts.Chart#container
- * @type {Highcharts.HTMLDOMElement}
- */
- container = createElement('div', {
- id: containerId
- }, containerStyle, renderTo);
- chart.container = container;
- // cache the cursor (#1650)
- chart._cursor = container.style.cursor;
- // Initialize the renderer
- Ren = H[optionsChart.renderer] || H.Renderer;
- /**
- * The renderer instance of the chart. Each chart instance has only one
- * associated renderer.
- *
- * @name Highcharts.Chart#renderer
- * @type {Highcharts.SVGRenderer}
- */
- chart.renderer = new Ren(container, chartWidth, chartHeight, null, optionsChart.forExport, options.exporting && options.exporting.allowHTML, chart.styledMode);
- // Set the initial animation from the options
- setAnimation(void 0, chart);
- chart.setClassName(optionsChart.className);
- if (!chart.styledMode) {
- chart.renderer.setStyle(optionsChart.style);
- }
- else {
- // Initialize definitions
- for (key in options.defs) { // eslint-disable-line guard-for-in
- this.renderer.definition(options.defs[key]);
- }
- }
- // Add a reference to the charts index
- chart.renderer.chartIndex = chart.index;
- fireEvent(this, 'afterGetContainer');
- };
- /**
- * Calculate margins by rendering axis labels in a preliminary position.
- * Title, subtitle and legend have already been rendered at this stage, but
- * will be moved into their final positions.
- *
- * @private
- * @function Highcharts.Chart#getMargins
- * @fires Highcharts.Chart#event:getMargins
- */
- Chart.prototype.getMargins = function (skipAxes) {
- var _a = this,
- spacing = _a.spacing,
- margin = _a.margin,
- titleOffset = _a.titleOffset;
- this.resetMargins();
- // Adjust for title and subtitle
- if (titleOffset[0] && !defined(margin[0])) {
- this.plotTop = Math.max(this.plotTop, titleOffset[0] + spacing[0]);
- }
- if (titleOffset[2] && !defined(margin[2])) {
- this.marginBottom = Math.max(this.marginBottom, titleOffset[2] + spacing[2]);
- }
- // Adjust for legend
- if (this.legend && this.legend.display) {
- this.legend.adjustMargins(margin, spacing);
- }
- fireEvent(this, 'getMargins');
- if (!skipAxes) {
- this.getAxisMargins();
- }
- };
- /**
- * @private
- * @function Highcharts.Chart#getAxisMargins
- */
- Chart.prototype.getAxisMargins = function () {
- var chart = this,
- // [top, right, bottom, left]
- axisOffset = chart.axisOffset = [0, 0, 0, 0],
- colorAxis = chart.colorAxis,
- margin = chart.margin,
- getOffset = function (axes) {
- axes.forEach(function (axis) {
- if (axis.visible) {
- axis.getOffset();
- }
- });
- };
- // pre-render axes to get labels offset width
- if (chart.hasCartesianSeries) {
- getOffset(chart.axes);
- }
- else if (colorAxis && colorAxis.length) {
- getOffset(colorAxis);
- }
- // Add the axis offsets
- marginNames.forEach(function (m, side) {
- if (!defined(margin[side])) {
- chart[m] += axisOffset[side];
- }
- });
- chart.setChartSize();
- };
- /**
- * Reflows the chart to its container. By default, the chart reflows
- * automatically to its container following a `window.resize` event, as per
- * the [chart.reflow](https://api.highcharts.com/highcharts/chart.reflow)
- * option. However, there are no reliable events for div resize, so if the
- * container is resized without a window resize event, this must be called
- * explicitly.
- *
- * @sample highcharts/members/chart-reflow/
- * Resize div and reflow
- * @sample highcharts/chart/events-container/
- * Pop up and reflow
- *
- * @function Highcharts.Chart#reflow
- *
- * @param {global.Event} [e]
- * Event arguments. Used primarily when the function is called
- * internally as a response to window resize.
- */
- Chart.prototype.reflow = function (e) {
- var chart = this, optionsChart = chart.options.chart, renderTo = chart.renderTo, hasUserSize = (defined(optionsChart.width) &&
- defined(optionsChart.height)), width = optionsChart.width || getStyle(renderTo, 'width'), height = optionsChart.height || getStyle(renderTo, 'height'), target = e ? e.target : win;
- // Width and height checks for display:none. Target is doc in IE8 and
- // Opera, win in Firefox, Chrome and IE9.
- if (!hasUserSize &&
- !chart.isPrinting &&
- width &&
- height &&
- (target === win || target === doc)) {
- if (width !== chart.containerWidth ||
- height !== chart.containerHeight) {
- U.clearTimeout(chart.reflowTimeout);
- // When called from window.resize, e is set, else it's called
- // directly (#2224)
- chart.reflowTimeout = syncTimeout(function () {
- // Set size, it may have been destroyed in the meantime
- // (#1257)
- if (chart.container) {
- chart.setSize(void 0, void 0, false);
- }
- }, e ? 100 : 0);
- }
- chart.containerWidth = width;
- chart.containerHeight = height;
- }
- };
- /**
- * Toggle the event handlers necessary for auto resizing, depending on the
- * `chart.reflow` option.
- *
- * @private
- * @function Highcharts.Chart#setReflow
- */
- Chart.prototype.setReflow = function (reflow) {
- var chart = this;
- if (reflow !== false && !this.unbindReflow) {
- this.unbindReflow = addEvent(win, 'resize', function (e) {
- // a removed event listener still runs in Edge and IE if the
- // listener was removed while the event runs, so check if the
- // chart is not destroyed (#11609)
- if (chart.options) {
- chart.reflow(e);
- }
- });
- addEvent(this, 'destroy', this.unbindReflow);
- }
- else if (reflow === false && this.unbindReflow) {
- // Unbind and unset
- this.unbindReflow = this.unbindReflow();
- }
- // The following will add listeners to re-fit the chart before and after
- // printing (#2284). However it only works in WebKit. Should have worked
- // in Firefox, but not supported in IE.
- /*
- if (win.matchMedia) {
- win.matchMedia('print').addListener(function reflow() {
- chart.reflow();
- });
- }
- //*/
- };
- /**
- * Resize the chart to a given width and height. In order to set the width
- * only, the height argument may be skipped. To set the height only, pass
- * `undefined` for the width.
- *
- * @sample highcharts/members/chart-setsize-button/
- * Test resizing from buttons
- * @sample highcharts/members/chart-setsize-jquery-resizable/
- * Add a jQuery UI resizable
- * @sample stock/members/chart-setsize/
- * Highstock with UI resizable
- *
- * @function Highcharts.Chart#setSize
- *
- * @param {number|null} [width]
- * The new pixel width of the chart. Since v4.2.6, the argument can
- * be `undefined` in order to preserve the current value (when
- * setting height only), or `null` to adapt to the width of the
- * containing element.
- *
- * @param {number|null} [height]
- * The new pixel height of the chart. Since v4.2.6, the argument can
- * be `undefined` in order to preserve the current value, or `null`
- * in order to adapt to the height of the containing element.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Whether and how to apply animation.
- *
- * @return {void}
- *
- * @fires Highcharts.Chart#event:endResize
- * @fires Highcharts.Chart#event:resize
- */
- Chart.prototype.setSize = function (width, height, animation) {
- var chart = this,
- renderer = chart.renderer,
- globalAnimation;
- // Handle the isResizing counter
- chart.isResizing += 1;
- // set the animation for the current process
- setAnimation(animation, chart);
- globalAnimation = renderer.globalAnimation;
- chart.oldChartHeight = chart.chartHeight;
- chart.oldChartWidth = chart.chartWidth;
- if (typeof width !== 'undefined') {
- chart.options.chart.width = width;
- }
- if (typeof height !== 'undefined') {
- chart.options.chart.height = height;
- }
- chart.getChartSize();
- // Resize the container with the global animation applied if enabled
- // (#2503)
- if (!chart.styledMode) {
- (globalAnimation ? animate : css)(chart.container, {
- width: chart.chartWidth + 'px',
- height: chart.chartHeight + 'px'
- }, globalAnimation);
- }
- chart.setChartSize(true);
- renderer.setSize(chart.chartWidth, chart.chartHeight, globalAnimation);
- // handle axes
- chart.axes.forEach(function (axis) {
- axis.isDirty = true;
- axis.setScale();
- });
- chart.isDirtyLegend = true; // force legend redraw
- chart.isDirtyBox = true; // force redraw of plot and chart border
- chart.layOutTitles(); // #2857
- chart.getMargins();
- chart.redraw(globalAnimation);
- chart.oldChartHeight = null;
- fireEvent(chart, 'resize');
- // Fire endResize and set isResizing back. If animation is disabled,
- // fire without delay
- syncTimeout(function () {
- if (chart) {
- fireEvent(chart, 'endResize', null, function () {
- chart.isResizing -= 1;
- });
- }
- }, animObject(globalAnimation).duration);
- };
- /**
- * Set the public chart properties. This is done before and after the
- * pre-render to determine margin sizes.
- *
- * @private
- * @function Highcharts.Chart#setChartSize
- * @fires Highcharts.Chart#event:afterSetChartSize
- */
- Chart.prototype.setChartSize = function (skipAxes) {
- var chart = this,
- inverted = chart.inverted,
- renderer = chart.renderer,
- chartWidth = chart.chartWidth,
- chartHeight = chart.chartHeight,
- optionsChart = chart.options.chart,
- spacing = chart.spacing,
- clipOffset = chart.clipOffset,
- clipX,
- clipY,
- plotLeft,
- plotTop,
- plotWidth,
- plotHeight,
- plotBorderWidth;
- /**
- * The current left position of the plot area in pixels.
- *
- * @name Highcharts.Chart#plotLeft
- * @type {number}
- */
- chart.plotLeft = plotLeft = Math.round(chart.plotLeft);
- /**
- * The current top position of the plot area in pixels.
- *
- * @name Highcharts.Chart#plotTop
- * @type {number}
- */
- chart.plotTop = plotTop = Math.round(chart.plotTop);
- /**
- * The current width of the plot area in pixels.
- *
- * @name Highcharts.Chart#plotWidth
- * @type {number}
- */
- chart.plotWidth = plotWidth = Math.max(0, Math.round(chartWidth - plotLeft - chart.marginRight));
- /**
- * The current height of the plot area in pixels.
- *
- * @name Highcharts.Chart#plotHeight
- * @type {number}
- */
- chart.plotHeight = plotHeight = Math.max(0, Math.round(chartHeight - plotTop - chart.marginBottom));
- chart.plotSizeX = inverted ? plotHeight : plotWidth;
- chart.plotSizeY = inverted ? plotWidth : plotHeight;
- chart.plotBorderWidth = optionsChart.plotBorderWidth || 0;
- // Set boxes used for alignment
- chart.spacingBox = renderer.spacingBox = {
- x: spacing[3],
- y: spacing[0],
- width: chartWidth - spacing[3] - spacing[1],
- height: chartHeight - spacing[0] - spacing[2]
- };
- chart.plotBox = renderer.plotBox = {
- x: plotLeft,
- y: plotTop,
- width: plotWidth,
- height: plotHeight
- };
- plotBorderWidth = 2 * Math.floor(chart.plotBorderWidth / 2);
- clipX = Math.ceil(Math.max(plotBorderWidth, clipOffset[3]) / 2);
- clipY = Math.ceil(Math.max(plotBorderWidth, clipOffset[0]) / 2);
- chart.clipBox = {
- x: clipX,
- y: clipY,
- width: Math.floor(chart.plotSizeX -
- Math.max(plotBorderWidth, clipOffset[1]) / 2 -
- clipX),
- height: Math.max(0, Math.floor(chart.plotSizeY -
- Math.max(plotBorderWidth, clipOffset[2]) / 2 -
- clipY))
- };
- if (!skipAxes) {
- chart.axes.forEach(function (axis) {
- axis.setAxisSize();
- axis.setAxisTranslation();
- });
- }
- fireEvent(chart, 'afterSetChartSize', { skipAxes: skipAxes });
- };
- /**
- * Initial margins before auto size margins are applied.
- *
- * @private
- * @function Highcharts.Chart#resetMargins
- */
- Chart.prototype.resetMargins = function () {
- fireEvent(this, 'resetMargins');
- var chart = this,
- chartOptions = chart.options.chart;
- // Create margin and spacing array
- ['margin', 'spacing'].forEach(function splashArrays(target) {
- var value = chartOptions[target],
- values = isObject(value) ? value : [value,
- value,
- value,
- value];
- [
- 'Top',
- 'Right',
- 'Bottom',
- 'Left'
- ].forEach(function (sideName, side) {
- chart[target][side] = pick(chartOptions[target + sideName], values[side]);
- });
- });
- // Set margin names like chart.plotTop, chart.plotLeft,
- // chart.marginRight, chart.marginBottom.
- marginNames.forEach(function (m, side) {
- chart[m] = pick(chart.margin[side], chart.spacing[side]);
- });
- chart.axisOffset = [0, 0, 0, 0]; // top, right, bottom, left
- chart.clipOffset = [0, 0, 0, 0];
- };
- /**
- * Internal function to draw or redraw the borders and backgrounds for chart
- * and plot area.
- *
- * @private
- * @function Highcharts.Chart#drawChartBox
- * @fires Highcharts.Chart#event:afterDrawChartBox
- */
- Chart.prototype.drawChartBox = function () {
- var chart = this,
- optionsChart = chart.options.chart,
- renderer = chart.renderer,
- chartWidth = chart.chartWidth,
- chartHeight = chart.chartHeight,
- chartBackground = chart.chartBackground,
- plotBackground = chart.plotBackground,
- plotBorder = chart.plotBorder,
- chartBorderWidth,
- styledMode = chart.styledMode,
- plotBGImage = chart.plotBGImage,
- chartBackgroundColor = optionsChart.backgroundColor,
- plotBackgroundColor = optionsChart.plotBackgroundColor,
- plotBackgroundImage = optionsChart.plotBackgroundImage,
- mgn,
- bgAttr,
- plotLeft = chart.plotLeft,
- plotTop = chart.plotTop,
- plotWidth = chart.plotWidth,
- plotHeight = chart.plotHeight,
- plotBox = chart.plotBox,
- clipRect = chart.clipRect,
- clipBox = chart.clipBox,
- verb = 'animate';
- // Chart area
- if (!chartBackground) {
- chart.chartBackground = chartBackground = renderer.rect()
- .addClass('highcharts-background')
- .add();
- verb = 'attr';
- }
- if (!styledMode) {
- // Presentational
- chartBorderWidth = optionsChart.borderWidth || 0;
- mgn = chartBorderWidth + (optionsChart.shadow ? 8 : 0);
- bgAttr = {
- fill: chartBackgroundColor || 'none'
- };
- if (chartBorderWidth || chartBackground['stroke-width']) { // #980
- bgAttr.stroke = optionsChart.borderColor;
- bgAttr['stroke-width'] = chartBorderWidth;
- }
- chartBackground
- .attr(bgAttr)
- .shadow(optionsChart.shadow);
- }
- else {
- chartBorderWidth = mgn = chartBackground.strokeWidth();
- }
- chartBackground[verb]({
- x: mgn / 2,
- y: mgn / 2,
- width: chartWidth - mgn - chartBorderWidth % 2,
- height: chartHeight - mgn - chartBorderWidth % 2,
- r: optionsChart.borderRadius
- });
- // Plot background
- verb = 'animate';
- if (!plotBackground) {
- verb = 'attr';
- chart.plotBackground = plotBackground = renderer.rect()
- .addClass('highcharts-plot-background')
- .add();
- }
- plotBackground[verb](plotBox);
- if (!styledMode) {
- // Presentational attributes for the background
- plotBackground
- .attr({
- fill: plotBackgroundColor || 'none'
- })
- .shadow(optionsChart.plotShadow);
- // Create the background image
- if (plotBackgroundImage) {
- if (!plotBGImage) {
- chart.plotBGImage = renderer.image(plotBackgroundImage, plotLeft, plotTop, plotWidth, plotHeight).add();
- }
- else {
- if (plotBackgroundImage !== plotBGImage.attr('href')) {
- plotBGImage.attr('href', plotBackgroundImage);
- }
- plotBGImage.animate(plotBox);
- }
- }
- }
- // Plot clip
- if (!clipRect) {
- chart.clipRect = renderer.clipRect(clipBox);
- }
- else {
- clipRect.animate({
- width: clipBox.width,
- height: clipBox.height
- });
- }
- // Plot area border
- verb = 'animate';
- if (!plotBorder) {
- verb = 'attr';
- chart.plotBorder = plotBorder = renderer.rect()
- .addClass('highcharts-plot-border')
- .attr({
- zIndex: 1 // Above the grid
- })
- .add();
- }
- if (!styledMode) {
- // Presentational
- plotBorder.attr({
- stroke: optionsChart.plotBorderColor,
- 'stroke-width': optionsChart.plotBorderWidth || 0,
- fill: 'none'
- });
- }
- plotBorder[verb](plotBorder.crisp({
- x: plotLeft,
- y: plotTop,
- width: plotWidth,
- height: plotHeight
- }, -plotBorder.strokeWidth())); // #3282 plotBorder should be negative;
- // reset
- chart.isDirtyBox = false;
- fireEvent(this, 'afterDrawChartBox');
- };
- /**
- * Detect whether a certain chart property is needed based on inspecting its
- * options and series. This mainly applies to the chart.inverted property,
- * and in extensions to the chart.angular and chart.polar properties.
- *
- * @private
- * @function Highcharts.Chart#propFromSeries
- * @return {void}
- */
- Chart.prototype.propFromSeries = function () {
- var chart = this,
- optionsChart = chart.options.chart,
- klass,
- seriesOptions = chart.options.series,
- i,
- value;
- /**
- * The flag is set to `true` if a series of the chart is inverted.
- *
- * @name Highcharts.Chart#inverted
- * @type {boolean|undefined}
- */
- ['inverted', 'angular', 'polar'].forEach(function (key) {
- // The default series type's class
- klass = seriesTypes[(optionsChart.type ||
- optionsChart.defaultSeriesType)];
- // Get the value from available chart-wide properties
- value =
- // It is set in the options:
- optionsChart[key] ||
- // The default series class:
- (klass && klass.prototype[key]);
- // requires it
- // 4. Check if any the chart's series require it
- i = seriesOptions && seriesOptions.length;
- while (!value && i--) {
- klass = seriesTypes[seriesOptions[i].type];
- if (klass && klass.prototype[key]) {
- value = true;
- }
- }
- // Set the chart property
- chart[key] = value;
- });
- };
- /**
- * Internal function to link two or more series together, based on the
- * `linkedTo` option. This is done from `Chart.render`, and after
- * `Chart.addSeries` and `Series.remove`.
- *
- * @private
- * @function Highcharts.Chart#linkSeries
- * @fires Highcharts.Chart#event:afterLinkSeries
- */
- Chart.prototype.linkSeries = function () {
- var chart = this,
- chartSeries = chart.series;
- // Reset links
- chartSeries.forEach(function (series) {
- series.linkedSeries.length = 0;
- });
- // Apply new links
- chartSeries.forEach(function (series) {
- var linkedTo = series.options.linkedTo;
- if (isString(linkedTo)) {
- if (linkedTo === ':previous') {
- linkedTo = chart.series[series.index - 1];
- }
- else {
- linkedTo = chart.get(linkedTo);
- }
- // #3341 avoid mutual linking
- if (linkedTo && linkedTo.linkedParent !== series) {
- linkedTo.linkedSeries.push(series);
- series.linkedParent = linkedTo;
- if (linkedTo.enabledDataSorting) {
- series.setDataSortingOptions();
- }
- series.visible = pick(series.options.visible, linkedTo.options.visible, series.visible); // #3879
- }
- }
- });
- fireEvent(this, 'afterLinkSeries');
- };
- /**
- * Render series for the chart.
- *
- * @private
- * @function Highcharts.Chart#renderSeries
- */
- Chart.prototype.renderSeries = function () {
- this.series.forEach(function (serie) {
- serie.translate();
- serie.render();
- });
- };
- /**
- * Render labels for the chart.
- *
- * @private
- * @function Highcharts.Chart#renderLabels
- */
- Chart.prototype.renderLabels = function () {
- var chart = this,
- labels = chart.options.labels;
- if (labels.items) {
- labels.items.forEach(function (label) {
- var style = extend(labels.style,
- label.style),
- x = pInt(style.left) + chart.plotLeft,
- y = pInt(style.top) + chart.plotTop + 12;
- // delete to prevent rewriting in IE
- delete style.left;
- delete style.top;
- chart.renderer.text(label.html, x, y)
- .attr({ zIndex: 2 })
- .css(style)
- .add();
- });
- }
- };
- /**
- * Render all graphics for the chart. Runs internally on initialization.
- *
- * @private
- * @function Highcharts.Chart#render
- */
- Chart.prototype.render = function () {
- var chart = this,
- axes = chart.axes,
- colorAxis = chart.colorAxis,
- renderer = chart.renderer,
- options = chart.options,
- correction = 0, // correction for X axis labels
- tempWidth,
- tempHeight,
- redoHorizontal,
- redoVertical,
- renderAxes = function (axes) {
- axes.forEach(function (axis) {
- if (axis.visible) {
- axis.render();
- }
- });
- };
- // Title
- chart.setTitle();
- /**
- * The overview of the chart's series.
- *
- * @name Highcharts.Chart#legend
- * @type {Highcharts.Legend}
- */
- chart.legend = new Legend(chart, options.legend);
- // Get stacks
- if (chart.getStacks) {
- chart.getStacks();
- }
- // Get chart margins
- chart.getMargins(true);
- chart.setChartSize();
- // Record preliminary dimensions for later comparison
- tempWidth = chart.plotWidth;
- axes.some(function (axis) {
- if (axis.horiz &&
- axis.visible &&
- axis.options.labels.enabled &&
- axis.series.length) {
- // 21 is the most common correction for X axis labels
- correction = 21;
- return true;
- }
- });
- // use Math.max to prevent negative plotHeight
- chart.plotHeight = Math.max(chart.plotHeight - correction, 0);
- tempHeight = chart.plotHeight;
- // Get margins by pre-rendering axes
- axes.forEach(function (axis) {
- axis.setScale();
- });
- chart.getAxisMargins();
- // If the plot area size has changed significantly, calculate tick
- // positions again
- redoHorizontal = tempWidth / chart.plotWidth > 1.1;
- // Height is more sensitive, use lower threshold
- redoVertical = tempHeight / chart.plotHeight > 1.05;
- if (redoHorizontal || redoVertical) {
- axes.forEach(function (axis) {
- if ((axis.horiz && redoHorizontal) ||
- (!axis.horiz && redoVertical)) {
- // update to reflect the new margins
- axis.setTickInterval(true);
- }
- });
- chart.getMargins(); // second pass to check for new labels
- }
- // Draw the borders and backgrounds
- chart.drawChartBox();
- // Axes
- if (chart.hasCartesianSeries) {
- renderAxes(axes);
- }
- else if (colorAxis && colorAxis.length) {
- renderAxes(colorAxis);
- }
- // The series
- if (!chart.seriesGroup) {
- chart.seriesGroup = renderer.g('series-group')
- .attr({ zIndex: 3 })
- .add();
- }
- chart.renderSeries();
- // Labels
- chart.renderLabels();
- // Credits
- chart.addCredits();
- // Handle responsiveness
- if (chart.setResponsive) {
- chart.setResponsive();
- }
- // Handle scaling
- chart.updateContainerScaling();
- // Set flag
- chart.hasRendered = true;
- };
- /**
- * Set a new credits label for the chart.
- *
- * @sample highcharts/credits/credits-update/
- * Add and update credits
- *
- * @function Highcharts.Chart#addCredits
- *
- * @param {Highcharts.CreditsOptions} [credits]
- * A configuration object for the new credits.
- */
- Chart.prototype.addCredits = function (credits) {
- var chart = this,
- creds = merge(true,
- this.options.credits,
- credits);
- if (creds.enabled && !this.credits) {
- /**
- * The chart's credits label. The label has an `update` method that
- * allows setting new options as per the
- * [credits options set](https://api.highcharts.com/highcharts/credits).
- *
- * @name Highcharts.Chart#credits
- * @type {Highcharts.SVGElement}
- */
- this.credits = this.renderer.text(creds.text + (this.mapCredits || ''), 0, 0)
- .addClass('highcharts-credits')
- .on('click', function () {
- if (creds.href) {
- win.location.href = creds.href;
- }
- })
- .attr({
- align: creds.position.align,
- zIndex: 8
- });
- if (!chart.styledMode) {
- this.credits.css(creds.style);
- }
- this.credits
- .add()
- .align(creds.position);
- // Dynamically update
- this.credits.update = function (options) {
- chart.credits = chart.credits.destroy();
- chart.addCredits(options);
- };
- }
- };
- /**
- * Handle scaling, #11329 - when there is scaling/transform on the container
- * or on a parent element, we need to take this into account. We calculate
- * the scaling once here and it is picked up where we need to use it
- * (Pointer, Tooltip).
- *
- * @private
- * @function Highcharts.Chart#updateContainerScaling
- */
- Chart.prototype.updateContainerScaling = function () {
- var container = this.container;
- // #13342 - tooltip was not visible in Chrome, when chart
- // updates height.
- if (container.offsetWidth > 2 && // #13342
- container.offsetHeight > 2 && // #13342
- container.getBoundingClientRect) {
- var bb = container.getBoundingClientRect(),
- scaleX = bb.width / container.offsetWidth,
- scaleY = bb.height / container.offsetHeight;
- if (scaleX !== 1 || scaleY !== 1) {
- this.containerScaling = { scaleX: scaleX, scaleY: scaleY };
- }
- else {
- delete this.containerScaling;
- }
- }
- };
- /**
- * Remove the chart and purge memory. This method is called internally
- * before adding a second chart into the same container, as well as on
- * window unload to prevent leaks.
- *
- * @sample highcharts/members/chart-destroy/
- * Destroy the chart from a button
- * @sample stock/members/chart-destroy/
- * Destroy with Highstock
- *
- * @function Highcharts.Chart#destroy
- *
- * @fires Highcharts.Chart#event:destroy
- */
- Chart.prototype.destroy = function () {
- var chart = this,
- axes = chart.axes,
- series = chart.series,
- container = chart.container,
- i,
- parentNode = container && container.parentNode;
- // fire the chart.destoy event
- fireEvent(chart, 'destroy');
- // Delete the chart from charts lookup array
- if (chart.renderer.forExport) {
- erase(charts, chart); // #6569
- }
- else {
- charts[chart.index] = void 0;
- }
- H.chartCount--;
- chart.renderTo.removeAttribute('data-highcharts-chart');
- // remove events
- removeEvent(chart);
- // ==== Destroy collections:
- // Destroy axes
- i = axes.length;
- while (i--) {
- axes[i] = axes[i].destroy();
- }
- // Destroy scroller & scroller series before destroying base series
- if (this.scroller && this.scroller.destroy) {
- this.scroller.destroy();
- }
- // Destroy each series
- i = series.length;
- while (i--) {
- series[i] = series[i].destroy();
- }
- // ==== Destroy chart properties:
- [
- 'title', 'subtitle', 'chartBackground', 'plotBackground',
- 'plotBGImage', 'plotBorder', 'seriesGroup', 'clipRect', 'credits',
- 'pointer', 'rangeSelector', 'legend', 'resetZoomButton', 'tooltip',
- 'renderer'
- ].forEach(function (name) {
- var prop = chart[name];
- if (prop && prop.destroy) {
- chart[name] = prop.destroy();
- }
- });
- // Remove container and all SVG, check container as it can break in IE
- // when destroyed before finished loading
- if (container) {
- container.innerHTML = '';
- removeEvent(container);
- if (parentNode) {
- discardElement(container);
- }
- }
- // clean it all up
- objectEach(chart, function (val, key) {
- delete chart[key];
- });
- };
- /**
- * Prepare for first rendering after all data are loaded.
- *
- * @private
- * @function Highcharts.Chart#firstRender
- * @fires Highcharts.Chart#event:beforeRender
- */
- Chart.prototype.firstRender = function () {
- var chart = this,
- options = chart.options;
- // Hook for oldIE to check whether the chart is ready to render
- if (chart.isReadyToRender && !chart.isReadyToRender()) {
- return;
- }
- // Create the container
- chart.getContainer();
- chart.resetMargins();
- chart.setChartSize();
- // Set the common chart properties (mainly invert) from the given series
- chart.propFromSeries();
- // get axes
- chart.getAxes();
- // Initialize the series
- (isArray(options.series) ? options.series : []).forEach(
- // #9680
- function (serieOptions) {
- chart.initSeries(serieOptions);
- });
- chart.linkSeries();
- chart.setSeriesData();
- // Run an event after axes and series are initialized, but before
- // render. At this stage, the series data is indexed and cached in the
- // xData and yData arrays, so we can access those before rendering. Used
- // in Highstock.
- fireEvent(chart, 'beforeRender');
- // depends on inverted and on margins being set
- if (Pointer) {
- if (!H.hasTouch && (win.PointerEvent || win.MSPointerEvent)) {
- chart.pointer = new MSPointer(chart, options);
- }
- else {
- /**
- * The Pointer that keeps track of mouse and touch interaction.
- *
- * @memberof Highcharts.Chart
- * @name pointer
- * @type {Highcharts.Pointer}
- * @instance
- */
- chart.pointer = new Pointer(chart, options);
- }
- }
- chart.render();
- // Fire the load event if there are no external images
- if (!chart.renderer.imgCount && !chart.hasLoaded) {
- chart.onload();
- }
- // If the chart was rendered outside the top container, put it back in
- // (#3679)
- chart.temporaryDisplay(true);
- };
- /**
- * Internal function that runs on chart load, async if any images are loaded
- * in the chart. Runs the callbacks and triggers the `load` and `render`
- * events.
- *
- * @private
- * @function Highcharts.Chart#onload
- * @fires Highcharts.Chart#event:load
- * @fires Highcharts.Chart#event:render
- */
- Chart.prototype.onload = function () {
- // Run callbacks, first the ones registered by modules, then user's one
- this.callbacks.concat([this.callback]).forEach(function (fn) {
- // Chart destroyed in its own callback (#3600)
- if (fn && typeof this.index !== 'undefined') {
- fn.apply(this, [this]);
- }
- }, this);
- fireEvent(this, 'load');
- fireEvent(this, 'render');
- // Set up auto resize, check for not destroyed (#6068)
- if (defined(this.index)) {
- this.setReflow(this.options.chart.reflow);
- }
- // Don't run again
- this.hasLoaded = true;
- };
- return Chart;
- }());
- // Hook for adding callbacks in modules
- Chart.prototype.callbacks = [];
- /**
- * Factory function for basic charts.
- *
- * @example
- * // Render a chart in to div#container
- * var chart = Highcharts.chart('container', {
- * title: {
- * text: 'My chart'
- * },
- * series: [{
- * data: [1, 3, 2, 4]
- * }]
- * });
- *
- * @function Highcharts.chart
- *
- * @param {string|Highcharts.HTMLDOMElement} [renderTo]
- * The DOM element to render to, or its id.
- *
- * @param {Highcharts.Options} options
- * The chart options structure.
- *
- * @param {Highcharts.ChartCallbackFunction} [callback]
- * Function to run when the chart has loaded and and all external images
- * are loaded. Defining a
- * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load)
- * handler is equivalent.
- *
- * @return {Highcharts.Chart}
- * Returns the Chart object.
- */
- function chart(a, b, c) {
- return new Chart(a, b, c);
- }
- H.chart = chart;
- H.Chart = Chart;
- return Chart;
- });
- _registerModule(_modules, 'Extensions/ScrollablePlotArea.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Chart, H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * Highcharts feature to make the Y axis stay fixed when scrolling the chart
- * horizontally on mobile devices. Supports left and right side axes.
- */
- /*
- WIP on vertical scrollable plot area (#9378). To do:
- - Bottom axis positioning
- - Test with Gantt
- - Look for size optimizing the code
- - API and demos
- */
- var addEvent = U.addEvent,
- createElement = U.createElement,
- pick = U.pick,
- stop = U.stop;
- /**
- * Options for a scrollable plot area. This feature provides a minimum size for
- * the plot area of the chart. If the size gets smaller than this, typically
- * on mobile devices, a native browser scrollbar is presented. This scrollbar
- * provides smooth scrolling for the contents of the plot area, whereas the
- * title, legend and unaffected axes are fixed.
- *
- * Since v7.1.2, a scrollable plot area can be defined for either horizontal or
- * vertical scrolling, depending on whether the `minWidth` or `minHeight`
- * option is set.
- *
- * @sample highcharts/chart/scrollable-plotarea
- * Scrollable plot area
- * @sample highcharts/chart/scrollable-plotarea-vertical
- * Vertically scrollable plot area
- * @sample {gantt} highcharts/chart/scrollable-plotarea-vertical
- * Gantt chart with vertically scrollable plot area
- *
- * @since 6.1.0
- * @product highcharts gantt
- * @apioption chart.scrollablePlotArea
- */
- /**
- * The minimum height for the plot area. If it gets smaller than this, the plot
- * area will become scrollable.
- *
- * @type {number}
- * @apioption chart.scrollablePlotArea.minHeight
- */
- /**
- * The minimum width for the plot area. If it gets smaller than this, the plot
- * area will become scrollable.
- *
- * @type {number}
- * @apioption chart.scrollablePlotArea.minWidth
- */
- /**
- * The initial scrolling position of the scrollable plot area. Ranges from 0 to
- * 1, where 0 aligns the plot area to the left and 1 aligns it to the right.
- * Typically we would use 1 if the chart has right aligned Y axes.
- *
- * @type {number}
- * @apioption chart.scrollablePlotArea.scrollPositionX
- */
- /**
- * The initial scrolling position of the scrollable plot area. Ranges from 0 to
- * 1, where 0 aligns the plot area to the top and 1 aligns it to the bottom.
- *
- * @type {number}
- * @apioption chart.scrollablePlotArea.scrollPositionY
- */
- /**
- * The opacity of mask applied on one of the sides of the plot
- * area.
- *
- * @sample {highcharts} highcharts/chart/scrollable-plotarea-opacity
- * Disabled opacity for the mask
- *
- * @type {number}
- * @default 0.85
- * @since 7.1.1
- * @apioption chart.scrollablePlotArea.opacity
- */
- ''; // detach API doclets
- /* eslint-disable no-invalid-this, valid-jsdoc */
- addEvent(Chart, 'afterSetChartSize', function (e) {
- var scrollablePlotArea = this.options.chart.scrollablePlotArea,
- scrollableMinWidth = scrollablePlotArea && scrollablePlotArea.minWidth,
- scrollableMinHeight = scrollablePlotArea && scrollablePlotArea.minHeight,
- scrollablePixelsX,
- scrollablePixelsY,
- corrections;
- if (!this.renderer.forExport) {
- // The amount of pixels to scroll, the difference between chart
- // width and scrollable width
- if (scrollableMinWidth) {
- this.scrollablePixelsX = scrollablePixelsX = Math.max(0, scrollableMinWidth - this.chartWidth);
- if (scrollablePixelsX) {
- this.plotWidth += scrollablePixelsX;
- if (this.inverted) {
- this.clipBox.height += scrollablePixelsX;
- this.plotBox.height += scrollablePixelsX;
- }
- else {
- this.clipBox.width += scrollablePixelsX;
- this.plotBox.width += scrollablePixelsX;
- }
- corrections = {
- // Corrections for right side
- 1: { name: 'right', value: scrollablePixelsX }
- };
- }
- // Currently we can only do either X or Y
- }
- else if (scrollableMinHeight) {
- this.scrollablePixelsY = scrollablePixelsY = Math.max(0, scrollableMinHeight - this.chartHeight);
- if (scrollablePixelsY) {
- this.plotHeight += scrollablePixelsY;
- if (this.inverted) {
- this.clipBox.width += scrollablePixelsY;
- this.plotBox.width += scrollablePixelsY;
- }
- else {
- this.clipBox.height += scrollablePixelsY;
- this.plotBox.height += scrollablePixelsY;
- }
- corrections = {
- 2: { name: 'bottom', value: scrollablePixelsY }
- };
- }
- }
- if (corrections && !e.skipAxes) {
- this.axes.forEach(function (axis) {
- // For right and bottom axes, only fix the plot line length
- if (corrections[axis.side]) {
- // Get the plot lines right in getPlotLinePath,
- // temporarily set it to the adjusted plot width.
- axis.getPlotLinePath = function () {
- var marginName = corrections[axis.side].name,
- correctionValue = corrections[axis.side].value,
- // axis.right or axis.bottom
- margin = this[marginName],
- path;
- // Temporarily adjust
- this[marginName] = margin - correctionValue;
- path = H.Axis.prototype.getPlotLinePath.apply(this, arguments);
- // Reset
- this[marginName] = margin;
- return path;
- };
- }
- else {
- // Apply the corrected plotWidth
- axis.setAxisSize();
- axis.setAxisTranslation();
- }
- });
- }
- }
- });
- addEvent(Chart, 'render', function () {
- if (this.scrollablePixelsX || this.scrollablePixelsY) {
- if (this.setUpScrolling) {
- this.setUpScrolling();
- }
- this.applyFixed();
- }
- else if (this.fixedDiv) { // Has been in scrollable mode
- this.applyFixed();
- }
- });
- /**
- * @private
- * @function Highcharts.Chart#setUpScrolling
- * @return {void}
- */
- Chart.prototype.setUpScrolling = function () {
- var _this = this;
- var attribs = {
- WebkitOverflowScrolling: 'touch',
- overflowX: 'hidden',
- overflowY: 'hidden'
- };
- if (this.scrollablePixelsX) {
- attribs.overflowX = 'auto';
- }
- if (this.scrollablePixelsY) {
- attribs.overflowY = 'auto';
- }
- // Insert a container with position relative
- // that scrolling and fixed container renders to (#10555)
- this.scrollingParent = createElement('div', {
- className: 'highcharts-scrolling-parent'
- }, {
- position: 'relative'
- }, this.renderTo);
- // Add the necessary divs to provide scrolling
- this.scrollingContainer = createElement('div', {
- 'className': 'highcharts-scrolling'
- }, attribs, this.scrollingParent);
- // On scroll, reset the chart position because it applies to the scrolled
- // container
- addEvent(this.scrollingContainer, 'scroll', function () {
- if (_this.pointer) {
- delete _this.pointer.chartPosition;
- }
- });
- this.innerContainer = createElement('div', {
- 'className': 'highcharts-inner-container'
- }, null, this.scrollingContainer);
- // Now move the container inside
- this.innerContainer.appendChild(this.container);
- // Don't run again
- this.setUpScrolling = null;
- };
- /**
- * These elements are moved over to the fixed renderer and stay fixed when the
- * user scrolls the chart
- * @private
- */
- Chart.prototype.moveFixedElements = function () {
- var container = this.container,
- fixedRenderer = this.fixedRenderer,
- fixedSelectors = [
- '.highcharts-contextbutton',
- '.highcharts-credits',
- '.highcharts-legend',
- '.highcharts-legend-checkbox',
- '.highcharts-navigator-series',
- '.highcharts-navigator-xaxis',
- '.highcharts-navigator-yaxis',
- '.highcharts-navigator',
- '.highcharts-reset-zoom',
- '.highcharts-scrollbar',
- '.highcharts-subtitle',
- '.highcharts-title'
- ],
- axisClass;
- if (this.scrollablePixelsX && !this.inverted) {
- axisClass = '.highcharts-yaxis';
- }
- else if (this.scrollablePixelsX && this.inverted) {
- axisClass = '.highcharts-xaxis';
- }
- else if (this.scrollablePixelsY && !this.inverted) {
- axisClass = '.highcharts-xaxis';
- }
- else if (this.scrollablePixelsY && this.inverted) {
- axisClass = '.highcharts-yaxis';
- }
- fixedSelectors.push(axisClass, axisClass + '-labels');
- fixedSelectors.forEach(function (className) {
- [].forEach.call(container.querySelectorAll(className), function (elem) {
- (elem.namespaceURI === fixedRenderer.SVG_NS ?
- fixedRenderer.box :
- fixedRenderer.box.parentNode).appendChild(elem);
- elem.style.pointerEvents = 'auto';
- });
- });
- };
- /**
- * @private
- * @function Highcharts.Chart#applyFixed
- * @return {void}
- */
- Chart.prototype.applyFixed = function () {
- var _a,
- _b;
- var fixedRenderer,
- scrollableWidth,
- scrollableHeight,
- firstTime = !this.fixedDiv,
- scrollableOptions = this.options.chart.scrollablePlotArea;
- // First render
- if (firstTime) {
- this.fixedDiv = createElement('div', {
- className: 'highcharts-fixed'
- }, {
- position: 'absolute',
- overflow: 'hidden',
- pointerEvents: 'none',
- zIndex: 2,
- top: 0
- }, null, true);
- (_a = this.scrollingContainer) === null || _a === void 0 ? void 0 : _a.parentNode.insertBefore(this.fixedDiv, this.scrollingContainer);
- this.renderTo.style.overflow = 'visible';
- this.fixedRenderer = fixedRenderer = new H.Renderer(this.fixedDiv, this.chartWidth, this.chartHeight, (_b = this.options.chart) === null || _b === void 0 ? void 0 : _b.style);
- // Mask
- this.scrollableMask = fixedRenderer
- .path()
- .attr({
- fill: this.options.chart.backgroundColor || '#fff',
- 'fill-opacity': pick(scrollableOptions.opacity, 0.85),
- zIndex: -1
- })
- .addClass('highcharts-scrollable-mask')
- .add();
- this.moveFixedElements();
- addEvent(this, 'afterShowResetZoom', this.moveFixedElements);
- addEvent(this, 'afterLayOutTitles', this.moveFixedElements);
- }
- else {
- // Set the size of the fixed renderer to the visible width
- this.fixedRenderer.setSize(this.chartWidth, this.chartHeight);
- }
- // Increase the size of the scrollable renderer and background
- scrollableWidth = this.chartWidth + (this.scrollablePixelsX || 0);
- scrollableHeight = this.chartHeight + (this.scrollablePixelsY || 0);
- stop(this.container);
- this.container.style.width = scrollableWidth + 'px';
- this.container.style.height = scrollableHeight + 'px';
- this.renderer.boxWrapper.attr({
- width: scrollableWidth,
- height: scrollableHeight,
- viewBox: [0, 0, scrollableWidth, scrollableHeight].join(' ')
- });
- this.chartBackground.attr({
- width: scrollableWidth,
- height: scrollableHeight
- });
- this.scrollingContainer.style.height = this.chartHeight + 'px';
- // Set scroll position
- if (firstTime) {
- if (scrollableOptions.scrollPositionX) {
- this.scrollingContainer.scrollLeft =
- this.scrollablePixelsX *
- scrollableOptions.scrollPositionX;
- }
- if (scrollableOptions.scrollPositionY) {
- this.scrollingContainer.scrollTop =
- this.scrollablePixelsY *
- scrollableOptions.scrollPositionY;
- }
- }
- // Mask behind the left and right side
- var axisOffset = this.axisOffset,
- maskTop = this.plotTop - axisOffset[0] - 1,
- maskLeft = this.plotLeft - axisOffset[3] - 1,
- maskBottom = this.plotTop + this.plotHeight + axisOffset[2] + 1,
- maskRight = this.plotLeft + this.plotWidth + axisOffset[1] + 1,
- maskPlotRight = this.plotLeft + this.plotWidth -
- (this.scrollablePixelsX || 0),
- maskPlotBottom = this.plotTop + this.plotHeight -
- (this.scrollablePixelsY || 0),
- d;
- if (this.scrollablePixelsX) {
- d = [
- // Left side
- ['M', 0, maskTop],
- ['L', this.plotLeft - 1, maskTop],
- ['L', this.plotLeft - 1, maskBottom],
- ['L', 0, maskBottom],
- ['Z'],
- // Right side
- ['M', maskPlotRight, maskTop],
- ['L', this.chartWidth, maskTop],
- ['L', this.chartWidth, maskBottom],
- ['L', maskPlotRight, maskBottom],
- ['Z']
- ];
- }
- else if (this.scrollablePixelsY) {
- d = [
- // Top side
- ['M', maskLeft, 0],
- ['L', maskLeft, this.plotTop - 1],
- ['L', maskRight, this.plotTop - 1],
- ['L', maskRight, 0],
- ['Z'],
- // Bottom side
- ['M', maskLeft, maskPlotBottom],
- ['L', maskLeft, this.chartHeight],
- ['L', maskRight, this.chartHeight],
- ['L', maskRight, maskPlotBottom],
- ['Z']
- ];
- }
- else {
- d = [['M', 0, 0]];
- }
- if (this.redrawTrigger !== 'adjustHeight') {
- this.scrollableMask.attr({ d: d });
- }
- };
- });
- _registerModule(_modules, 'Core/Axis/StackingAxis.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- destroyObjectProperties = U.destroyObjectProperties,
- fireEvent = U.fireEvent,
- getDeferredAnimation = U.getDeferredAnimation,
- objectEach = U.objectEach,
- pick = U.pick;
- /* eslint-disable valid-jsdoc */
- /**
- * Adds stacking support to axes.
- * @private
- * @class
- */
- var StackingAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function StackingAxisAdditions(axis) {
- this.oldStacks = {};
- this.stacks = {};
- this.stacksTouched = 0;
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Build the stacks from top down
- * @private
- */
- StackingAxisAdditions.prototype.buildStacks = function () {
- var stacking = this;
- var axis = stacking.axis;
- var axisSeries = axis.series;
- var reversedStacks = pick(axis.options.reversedStacks,
- true);
- var len = axisSeries.length;
- var actualSeries,
- i;
- if (!axis.isXAxis) {
- stacking.usePercentage = false;
- i = len;
- while (i--) {
- actualSeries = axisSeries[reversedStacks ? i : len - i - 1];
- actualSeries.setStackedPoints();
- actualSeries.setGroupedPoints();
- }
- // Loop up again to compute percent and stream stack
- for (i = 0; i < len; i++) {
- axisSeries[i].modifyStacks();
- }
- fireEvent(axis, 'afterBuildStacks');
- }
- };
- /**
- * @private
- */
- StackingAxisAdditions.prototype.cleanStacks = function () {
- var stacking = this;
- var axis = stacking.axis;
- var stacks;
- if (!axis.isXAxis) {
- if (stacking.oldStacks) {
- stacks = stacking.stacks = stacking.oldStacks;
- }
- // reset stacks
- objectEach(stacks, function (type) {
- objectEach(type, function (stack) {
- stack.cumulative = stack.total;
- });
- });
- }
- };
- /**
- * Set all the stacks to initial states and destroy unused ones.
- * @private
- */
- StackingAxisAdditions.prototype.resetStacks = function () {
- var stacking = this;
- var axis = stacking.axis;
- var stacks = stacking.stacks;
- if (!axis.isXAxis) {
- objectEach(stacks, function (type) {
- objectEach(type, function (stack, key) {
- // Clean up memory after point deletion (#1044, #4320)
- if (stack.touched < stacking.stacksTouched) {
- stack.destroy();
- delete type[key];
- // Reset stacks
- }
- else {
- stack.total = null;
- stack.cumulative = null;
- }
- });
- });
- }
- };
- /**
- * @private
- */
- StackingAxisAdditions.prototype.renderStackTotals = function () {
- var stacking = this;
- var axis = stacking.axis;
- var chart = axis.chart;
- var renderer = chart.renderer;
- var stacks = stacking.stacks;
- var stackLabelsAnim = axis.options.stackLabels.animation;
- var animationConfig = getDeferredAnimation(chart,
- stackLabelsAnim);
- var stackTotalGroup = stacking.stackTotalGroup = (stacking.stackTotalGroup ||
- renderer
- .g('stack-labels')
- .attr({
- visibility: 'visible',
- zIndex: 6,
- opacity: 0
- })
- .add());
- // plotLeft/Top will change when y axis gets wider so we need to
- // translate the stackTotalGroup at every render call. See bug #506
- // and #516
- stackTotalGroup.translate(chart.plotLeft, chart.plotTop);
- // Render each stack total
- objectEach(stacks, function (type) {
- objectEach(type, function (stack) {
- stack.render(stackTotalGroup);
- });
- });
- stackTotalGroup.animate({
- opacity: 1
- }, animationConfig);
- };
- return StackingAxisAdditions;
- }());
- /**
- * Axis with stacking support.
- * @private
- * @class
- */
- var StackingAxis = /** @class */ (function () {
- function StackingAxis() {
- }
- /* *
- *
- * Static Functions
- *
- * */
- /**
- * Extends axis with stacking support.
- * @private
- */
- StackingAxis.compose = function (AxisClass) {
- var axisProto = AxisClass.prototype;
- addEvent(AxisClass, 'init', StackingAxis.onInit);
- addEvent(AxisClass, 'destroy', StackingAxis.onDestroy);
- };
- /**
- * @private
- */
- StackingAxis.onDestroy = function () {
- var stacking = this.stacking;
- if (!stacking) {
- return;
- }
- var stacks = stacking.stacks;
- // Destroy each stack total
- objectEach(stacks, function (stack, stackKey) {
- destroyObjectProperties(stack);
- stacks[stackKey] = null;
- });
- if (stacking &&
- stacking.stackTotalGroup) {
- stacking.stackTotalGroup.destroy();
- }
- };
- /**
- * @private
- */
- StackingAxis.onInit = function () {
- var axis = this;
- if (!axis.stacking) {
- axis.stacking = new StackingAxisAdditions(axis);
- }
- };
- return StackingAxis;
- }());
- return StackingAxis;
- });
- _registerModule(_modules, 'Mixins/LegendSymbol.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var merge = U.merge,
- pick = U.pick;
- /* eslint-disable valid-jsdoc */
- /**
- * Legend symbol mixin.
- *
- * @private
- * @mixin Highcharts.LegendSymbolMixin
- */
- var LegendSymbolMixin = H.LegendSymbolMixin = {
- /**
- * Get the series' symbol in the legend
- *
- * @private
- * @function Highcharts.LegendSymbolMixin.drawRectangle
- *
- * @param {Highcharts.Legend} legend
- * The legend object
- *
- * @param {Highcharts.Point|Highcharts.Series} item
- * The series (this) or point
- */
- drawRectangle: function (legend,
- item) {
- var options = legend.options,
- symbolHeight = legend.symbolHeight,
- square = options.squareSymbol,
- symbolWidth = square ? symbolHeight : legend.symbolWidth;
- item.legendSymbol = this.chart.renderer.rect(square ? (legend.symbolWidth - symbolHeight) / 2 : 0, legend.baseline - symbolHeight + 1, // #3988
- symbolWidth, symbolHeight, pick(legend.options.symbolRadius, symbolHeight / 2))
- .addClass('highcharts-point')
- .attr({
- zIndex: 3
- }).add(item.legendGroup);
- },
- /**
- * Get the series' symbol in the legend. This method should be overridable
- * to create custom symbols through
- * Highcharts.seriesTypes[type].prototype.drawLegendSymbols.
- *
- * @private
- * @function Highcharts.LegendSymbolMixin.drawLineMarker
- *
- * @param {Highcharts.Legend} legend
- * The legend object.
- */
- drawLineMarker: function (legend) {
- var options = this.options,
- markerOptions = options.marker,
- radius,
- legendSymbol,
- symbolWidth = legend.symbolWidth,
- symbolHeight = legend.symbolHeight,
- generalRadius = symbolHeight / 2,
- renderer = this.chart.renderer,
- legendItemGroup = this.legendGroup,
- verticalCenter = legend.baseline -
- Math.round(legend.fontMetrics.b * 0.3),
- attr = {};
- // Draw the line
- if (!this.chart.styledMode) {
- attr = {
- 'stroke-width': options.lineWidth || 0
- };
- if (options.dashStyle) {
- attr.dashstyle = options.dashStyle;
- }
- }
- this.legendLine = renderer
- .path([
- ['M', 0, verticalCenter],
- ['L', symbolWidth, verticalCenter]
- ])
- .addClass('highcharts-graph')
- .attr(attr)
- .add(legendItemGroup);
- // Draw the marker
- if (markerOptions && markerOptions.enabled !== false && symbolWidth) {
- // Do not allow the marker to be larger than the symbolHeight
- radius = Math.min(pick(markerOptions.radius, generalRadius), generalRadius);
- // Restrict symbol markers size
- if (this.symbol.indexOf('url') === 0) {
- markerOptions = merge(markerOptions, {
- width: symbolHeight,
- height: symbolHeight
- });
- radius = 0;
- }
- this.legendSymbol = legendSymbol = renderer.symbol(this.symbol, (symbolWidth / 2) - radius, verticalCenter - radius, 2 * radius, 2 * radius, markerOptions)
- .addClass('highcharts-point')
- .add(legendItemGroup);
- legendSymbol.isMarker = true;
- }
- }
- };
- return LegendSymbolMixin;
- });
- _registerModule(_modules, 'Core/Series/Point.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var animObject = U.animObject,
- defined = U.defined,
- erase = U.erase,
- extend = U.extend,
- fireEvent = U.fireEvent,
- format = U.format,
- getNestedProperty = U.getNestedProperty,
- isArray = U.isArray,
- isNumber = U.isNumber,
- isObject = U.isObject,
- syncTimeout = U.syncTimeout,
- pick = U.pick,
- removeEvent = U.removeEvent,
- uniqueKey = U.uniqueKey;
- /**
- * Function callback when a series point is clicked. Return false to cancel the
- * action.
- *
- * @callback Highcharts.PointClickCallbackFunction
- *
- * @param {Highcharts.Point} this
- * The point where the event occured.
- *
- * @param {Highcharts.PointClickEventObject} event
- * Event arguments.
- */
- /**
- * Common information for a click event on a series point.
- *
- * @interface Highcharts.PointClickEventObject
- * @extends Highcharts.PointerEventObject
- */ /**
- * Clicked point.
- * @name Highcharts.PointClickEventObject#point
- * @type {Highcharts.Point}
- */
- /**
- * Configuration hash for the data label and tooltip formatters.
- *
- * @interface Highcharts.PointLabelObject
- */ /**
- * The point's current color.
- * @name Highcharts.PointLabelObject#color
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined}
- */ /**
- * The point's current color index, used in styled mode instead of `color`. The
- * color index is inserted in class names used for styling.
- * @name Highcharts.PointLabelObject#colorIndex
- * @type {number}
- */ /**
- * The name of the related point.
- * @name Highcharts.PointLabelObject#key
- * @type {string|undefined}
- */ /**
- * The percentage for related points in a stacked series or pies.
- * @name Highcharts.PointLabelObject#percentage
- * @type {number}
- */ /**
- * The related point. The point name, if defined, is available through
- * `this.point.name`.
- * @name Highcharts.PointLabelObject#point
- * @type {Highcharts.Point}
- */ /**
- * The related series. The series name is available through `this.series.name`.
- * @name Highcharts.PointLabelObject#series
- * @type {Highcharts.Series}
- */ /**
- * The total of values in either a stack for stacked series, or a pie in a pie
- * series.
- * @name Highcharts.PointLabelObject#total
- * @type {number|undefined}
- */ /**
- * For categorized axes this property holds the category name for the point. For
- * other axes it holds the X value.
- * @name Highcharts.PointLabelObject#x
- * @type {number|string|undefined}
- */ /**
- * The y value of the point.
- * @name Highcharts.PointLabelObject#y
- * @type {number|undefined}
- */
- /**
- * Gets fired when the mouse leaves the area close to the point.
- *
- * @callback Highcharts.PointMouseOutCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {global.PointerEvent} event
- * Event that occured.
- */
- /**
- * Gets fired when the mouse enters the area close to the point.
- *
- * @callback Highcharts.PointMouseOverCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {global.Event} event
- * Event that occured.
- */
- /**
- * The generic point options for all series.
- *
- * In TypeScript you have to extend `PointOptionsObject` with an additional
- * declaration to allow custom data options:
- *
- * ```
- * declare interface PointOptionsObject {
- * customProperty: string;
- * }
- * ```
- *
- * @interface Highcharts.PointOptionsObject
- */
- /**
- * Possible option types for a data point.
- *
- * @typedef {number|string|Array<(number|string)>|Highcharts.PointOptionsObject|null} Highcharts.PointOptionsType
- */
- /**
- * Gets fired when the point is removed using the `.remove()` method.
- *
- * @callback Highcharts.PointRemoveCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {global.Event} event
- * Event that occured.
- */
- /**
- * Possible key values for the point state options.
- *
- * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.PointStateValue
- */
- /**
- * Gets fired when the point is updated programmatically through the `.update()`
- * method.
- *
- * @callback Highcharts.PointUpdateCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {Highcharts.PointUpdateEventObject} event
- * Event that occured.
- */
- /**
- * Information about the update event.
- *
- * @interface Highcharts.PointUpdateEventObject
- * @extends global.Event
- */ /**
- * Options data of the update event.
- * @name Highcharts.PointUpdateEventObject#options
- * @type {Highcharts.PointOptionsType}
- */
- ''; // detach doclet above
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Point object. The point objects are generated from the `series.data`
- * configuration objects or raw numbers. They can be accessed from the
- * `Series.points` array. Other ways to instantiate points are through {@link
- * Highcharts.Series#addPoint} or {@link Highcharts.Series#setData}.
- *
- * @class
- * @name Highcharts.Point
- */
- var Point = /** @class */ (function () {
- function Point() {
- /* *
- *
- * Properties
- *
- * */
- /**
- * For categorized axes this property holds the category name for the
- * point. For other axes it holds the X value.
- *
- * @name Highcharts.Point#category
- * @type {string}
- */
- this.category = void 0;
- /**
- * The point's current color index, used in styled mode instead of
- * `color`. The color index is inserted in class names used for styling.
- *
- * @name Highcharts.Point#colorIndex
- * @type {number}
- */
- this.colorIndex = void 0;
- this.formatPrefix = 'point';
- this.id = void 0;
- this.isNull = false;
- /**
- * The name of the point. The name can be given as the first position of the
- * point configuration array, or as a `name` property in the configuration:
- *
- * @example
- * // Array config
- * data: [
- * ['John', 1],
- * ['Jane', 2]
- * ]
- *
- * // Object config
- * data: [{
- * name: 'John',
- * y: 1
- * }, {
- * name: 'Jane',
- * y: 2
- * }]
- *
- * @name Highcharts.Point#name
- * @type {string}
- */
- this.name = void 0;
- /**
- * The point's options as applied in the initial configuration, or
- * extended through `Point.update`.
- *
- * In TypeScript you have to extend `PointOptionsObject` via an
- * additional interface to allow custom data options:
- *
- * ```
- * declare interface PointOptionsObject {
- * customProperty: string;
- * }
- * ```
- *
- * @name Highcharts.Point#options
- * @type {Highcharts.PointOptionsObject}
- */
- this.options = void 0;
- /**
- * The percentage for points in a stacked series or pies.
- *
- * @name Highcharts.Point#percentage
- * @type {number|undefined}
- */
- this.percentage = void 0;
- this.selected = false;
- /**
- * The series object associated with the point.
- *
- * @name Highcharts.Point#series
- * @type {Highcharts.Series}
- */
- this.series = void 0;
- /**
- * The total of values in either a stack for stacked series, or a pie in a
- * pie series.
- *
- * @name Highcharts.Point#total
- * @type {number|undefined}
- */
- this.total = void 0;
- /**
- * For certain series types, like pie charts, where individual points can
- * be shown or hidden.
- *
- * @name Highcharts.Point#visible
- * @type {boolean}
- * @default true
- */
- this.visible = true;
- this.x = void 0;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Animate SVG elements associated with the point.
- *
- * @private
- * @function Highcharts.Point#animateBeforeDestroy
- */
- Point.prototype.animateBeforeDestroy = function () {
- var point = this,
- animateParams = { x: point.startXPos,
- opacity: 0 },
- isDataLabel,
- graphicalProps = point.getGraphicalProps();
- graphicalProps.singular.forEach(function (prop) {
- isDataLabel = prop === 'dataLabel';
- point[prop] = point[prop].animate(isDataLabel ? {
- x: point[prop].startXPos,
- y: point[prop].startYPos,
- opacity: 0
- } : animateParams);
- });
- graphicalProps.plural.forEach(function (plural) {
- point[plural].forEach(function (item) {
- if (item.element) {
- item.animate(extend({ x: point.startXPos }, (item.startYPos ? {
- x: item.startXPos,
- y: item.startYPos
- } : {})));
- }
- });
- });
- };
- /**
- * Apply the options containing the x and y data and possible some extra
- * properties. Called on point init or from point.update.
- *
- * @private
- * @function Highcharts.Point#applyOptions
- *
- * @param {Highcharts.PointOptionsType} options
- * The point options as defined in series.data.
- *
- * @param {number} [x]
- * Optionally, the x value.
- *
- * @return {Highcharts.Point}
- * The Point instance.
- */
- Point.prototype.applyOptions = function (options, x) {
- var point = this,
- series = point.series,
- pointValKey = series.options.pointValKey || series.pointValKey;
- options = Point.prototype.optionsToObject.call(this, options);
- // copy options directly to point
- extend(point, options);
- point.options = point.options ? extend(point.options, options) : options;
- // Since options are copied into the Point instance, some accidental
- // options must be shielded (#5681)
- if (options.group) {
- delete point.group;
- }
- if (options.dataLabels) {
- delete point.dataLabels;
- }
- /**
- * The y value of the point.
- * @name Highcharts.Point#y
- * @type {number|undefined}
- */
- // For higher dimension series types. For instance, for ranges, point.y
- // is mapped to point.low.
- if (pointValKey) {
- point.y = Point.prototype.getNestedProperty.call(point, pointValKey);
- }
- point.isNull = pick(point.isValid && !point.isValid(), point.x === null || !isNumber(point.y)); // #3571, check for NaN
- point.formatPrefix = point.isNull ? 'null' : 'point'; // #9233, #10874
- // The point is initially selected by options (#5777)
- if (point.selected) {
- point.state = 'select';
- }
- /**
- * The x value of the point.
- * @name Highcharts.Point#x
- * @type {number}
- */
- // If no x is set by now, get auto incremented value. All points must
- // have an x value, however the y value can be null to create a gap in
- // the series
- if ('name' in point &&
- typeof x === 'undefined' &&
- series.xAxis &&
- series.xAxis.hasNames) {
- point.x = series.xAxis.nameToX(point);
- }
- if (typeof point.x === 'undefined' && series) {
- if (typeof x === 'undefined') {
- point.x = series.autoIncrement(point);
- }
- else {
- point.x = x;
- }
- }
- return point;
- };
- /**
- * Destroy a point to clear memory. Its reference still stays in
- * `series.data`.
- *
- * @private
- * @function Highcharts.Point#destroy
- */
- Point.prototype.destroy = function () {
- var point = this,
- series = point.series,
- chart = series.chart,
- dataSorting = series.options.dataSorting,
- hoverPoints = chart.hoverPoints,
- globalAnimation = point.series.chart.renderer.globalAnimation,
- animation = animObject(globalAnimation),
- prop;
- /**
- * Allow to call after animation.
- * @private
- */
- function destroyPoint() {
- // Remove all events and elements
- if (point.graphic || point.dataLabel || point.dataLabels) {
- removeEvent(point);
- point.destroyElements();
- }
- for (prop in point) { // eslint-disable-line guard-for-in
- point[prop] = null;
- }
- }
- if (point.legendItem) { // pies have legend items
- chart.legend.destroyItem(point);
- }
- if (hoverPoints) {
- point.setState();
- erase(hoverPoints, point);
- if (!hoverPoints.length) {
- chart.hoverPoints = null;
- }
- }
- if (point === chart.hoverPoint) {
- point.onMouseOut();
- }
- // Remove properties after animation
- if (!dataSorting || !dataSorting.enabled) {
- destroyPoint();
- }
- else {
- this.animateBeforeDestroy();
- syncTimeout(destroyPoint, animation.duration);
- }
- chart.pointCount--;
- };
- /**
- * Destroy SVG elements associated with the point.
- *
- * @private
- * @function Highcharts.Point#destroyElements
- * @param {Highcharts.Dictionary<number>} [kinds]
- */
- Point.prototype.destroyElements = function (kinds) {
- var point = this,
- props = point.getGraphicalProps(kinds);
- props.singular.forEach(function (prop) {
- point[prop] = point[prop].destroy();
- });
- props.plural.forEach(function (plural) {
- point[plural].forEach(function (item) {
- if (item.element) {
- item.destroy();
- }
- });
- delete point[plural];
- });
- };
- /**
- * Fire an event on the Point object.
- *
- * @private
- * @function Highcharts.Point#firePointEvent
- *
- * @param {string} eventType
- * Type of the event.
- *
- * @param {Highcharts.Dictionary<any>|Event} [eventArgs]
- * Additional event arguments.
- *
- * @param {Highcharts.EventCallbackFunction<Highcharts.Point>|Function} [defaultFunction]
- * Default event handler.
- *
- * @fires Highcharts.Point#event:*
- */
- Point.prototype.firePointEvent = function (eventType, eventArgs, defaultFunction) {
- var point = this,
- series = this.series,
- seriesOptions = series.options;
- // load event handlers on demand to save time on mouseover/out
- if (seriesOptions.point.events[eventType] ||
- (point.options &&
- point.options.events &&
- point.options.events[eventType])) {
- point.importEvents();
- }
- // add default handler if in selection mode
- if (eventType === 'click' && seriesOptions.allowPointSelect) {
- defaultFunction = function (event) {
- // Control key is for Windows, meta (= Cmd key) for Mac, Shift
- // for Opera.
- if (point.select) { // #2911
- point.select(null, event.ctrlKey || event.metaKey || event.shiftKey);
- }
- };
- }
- fireEvent(point, eventType, eventArgs, defaultFunction);
- };
- /**
- * Get the CSS class names for individual points. Used internally where the
- * returned value is set on every point.
- *
- * @function Highcharts.Point#getClassName
- *
- * @return {string}
- * The class names.
- */
- Point.prototype.getClassName = function () {
- var point = this;
- return 'highcharts-point' +
- (point.selected ? ' highcharts-point-select' : '') +
- (point.negative ? ' highcharts-negative' : '') +
- (point.isNull ? ' highcharts-null-point' : '') +
- (typeof point.colorIndex !== 'undefined' ?
- ' highcharts-color-' + point.colorIndex : '') +
- (point.options.className ? ' ' + point.options.className : '') +
- (point.zone && point.zone.className ? ' ' +
- point.zone.className.replace('highcharts-negative', '') : '');
- };
- /**
- * Get props of all existing graphical point elements.
- *
- * @private
- * @function Highcharts.Point#getGraphicalProps
- * @param {Highcharts.Dictionary<number>} [kinds]
- * @return {Highcharts.PointGraphicalProps}
- */
- Point.prototype.getGraphicalProps = function (kinds) {
- var point = this,
- props = [],
- prop,
- i,
- graphicalProps = { singular: [],
- plural: [] };
- kinds = kinds || { graphic: 1, dataLabel: 1 };
- if (kinds.graphic) {
- props.push('graphic', 'shadowGroup');
- }
- if (kinds.dataLabel) {
- props.push('dataLabel', 'dataLabelUpper', 'connector');
- }
- i = props.length;
- while (i--) {
- prop = props[i];
- if (point[prop]) {
- graphicalProps.singular.push(prop);
- }
- }
- ['dataLabel', 'connector'].forEach(function (prop) {
- var plural = prop + 's';
- if (kinds[prop] && point[plural]) {
- graphicalProps.plural.push(plural);
- }
- });
- return graphicalProps;
- };
- /**
- * Return the configuration hash needed for the data label and tooltip
- * formatters.
- *
- * @function Highcharts.Point#getLabelConfig
- *
- * @return {Highcharts.PointLabelObject}
- * Abstract object used in formatters and formats.
- */
- Point.prototype.getLabelConfig = function () {
- return {
- x: this.category,
- y: this.y,
- color: this.color,
- colorIndex: this.colorIndex,
- key: this.name || this.category,
- series: this.series,
- point: this,
- percentage: this.percentage,
- total: this.total || this.stackTotal
- };
- };
- /**
- * Returns the value of the point property for a given value.
- * @private
- */
- Point.prototype.getNestedProperty = function (key) {
- if (!key) {
- return;
- }
- if (key.indexOf('custom.') === 0) {
- return getNestedProperty(key, this.options);
- }
- return this[key];
- };
- /**
- * In a series with `zones`, return the zone that the point belongs to.
- *
- * @function Highcharts.Point#getZone
- *
- * @return {Highcharts.SeriesZonesOptionsObject}
- * The zone item.
- */
- Point.prototype.getZone = function () {
- var series = this.series,
- zones = series.zones,
- zoneAxis = series.zoneAxis || 'y',
- i = 0,
- zone;
- zone = zones[i];
- while (this[zoneAxis] >= zone.value) {
- zone = zones[++i];
- }
- // For resetting or reusing the point (#8100)
- if (!this.nonZonedColor) {
- this.nonZonedColor = this.color;
- }
- if (zone && zone.color && !this.options.color) {
- this.color = zone.color;
- }
- else {
- this.color = this.nonZonedColor;
- }
- return zone;
- };
- /**
- * Utility to check if point has new shape type. Used in column series and
- * all others that are based on column series.
- *
- * @return boolean|undefined
- */
- Point.prototype.hasNewShapeType = function () {
- var point = this;
- var oldShapeType = point.graphic &&
- (point.graphic.symbolName || point.graphic.element.nodeName);
- return oldShapeType !== this.shapeType;
- };
- /**
- * Initialize the point. Called internally based on the `series.data`
- * option.
- *
- * @function Highcharts.Point#init
- *
- * @param {Highcharts.Series} series
- * The series object containing this point.
- *
- * @param {Highcharts.PointOptionsType} options
- * The data in either number, array or object format.
- *
- * @param {number} [x]
- * Optionally, the X value of the point.
- *
- * @return {Highcharts.Point}
- * The Point instance.
- *
- * @fires Highcharts.Point#event:afterInit
- */
- Point.prototype.init = function (series, options, x) {
- this.series = series;
- this.applyOptions(options, x);
- // Add a unique ID to the point if none is assigned
- this.id = defined(this.id) ? this.id : uniqueKey();
- this.resolveColor();
- series.chart.pointCount++;
- fireEvent(this, 'afterInit');
- return this;
- };
- /**
- * Transform number or array configs into objects. Also called for object
- * configs. Used internally to unify the different configuration formats for
- * points. For example, a simple number `10` in a line series will be
- * transformed to `{ y: 10 }`, and an array config like `[1, 10]` in a
- * scatter series will be transformed to `{ x: 1, y: 10 }`.
- *
- * @function Highcharts.Point#optionsToObject
- *
- * @param {Highcharts.PointOptionsType} options
- * The input option.
- *
- * @return {Highcharts.Dictionary<*>}
- * Transformed options.
- */
- Point.prototype.optionsToObject = function (options) {
- var ret = {},
- series = this.series,
- keys = series.options.keys,
- pointArrayMap = keys || series.pointArrayMap || ['y'],
- valueCount = pointArrayMap.length,
- firstItemType,
- i = 0,
- j = 0;
- if (isNumber(options) || options === null) {
- ret[pointArrayMap[0]] = options;
- }
- else if (isArray(options)) {
- // with leading x value
- if (!keys && options.length > valueCount) {
- firstItemType = typeof options[0];
- if (firstItemType === 'string') {
- ret.name = options[0];
- }
- else if (firstItemType === 'number') {
- ret.x = options[0];
- }
- i++;
- }
- while (j < valueCount) {
- // Skip undefined positions for keys
- if (!keys || typeof options[i] !== 'undefined') {
- if (pointArrayMap[j].indexOf('.') > 0) {
- // Handle nested keys, e.g. ['color.pattern.image']
- // Avoid function call unless necessary.
- Point.prototype.setNestedProperty(ret, options[i], pointArrayMap[j]);
- }
- else {
- ret[pointArrayMap[j]] = options[i];
- }
- }
- i++;
- j++;
- }
- }
- else if (typeof options === 'object') {
- ret = options;
- // This is the fastest way to detect if there are individual point
- // dataLabels that need to be considered in drawDataLabels. These
- // can only occur in object configs.
- if (options.dataLabels) {
- series._hasPointLabels = true;
- }
- // Same approach as above for markers
- if (options.marker) {
- series._hasPointMarkers = true;
- }
- }
- return ret;
- };
- /**
- * @private
- * @function Highcharts.Point#resolveColor
- * @return {void}
- */
- Point.prototype.resolveColor = function () {
- var series = this.series,
- colors,
- optionsChart = series.chart.options.chart,
- colorCount = optionsChart.colorCount,
- styledMode = series.chart.styledMode,
- colorIndex;
- // remove points nonZonedColor for later recalculation
- delete this.nonZonedColor;
- /**
- * The point's current color.
- *
- * @name Highcharts.Point#color
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject|undefined}
- */
- if (!styledMode && !this.options.color) {
- this.color = series.color; // #3445
- }
- if (series.options.colorByPoint) {
- if (!styledMode) {
- colors = series.options.colors || series.chart.options.colors;
- this.color = this.color || colors[series.colorCounter];
- colorCount = colors.length;
- }
- colorIndex = series.colorCounter;
- series.colorCounter++;
- // loop back to zero
- if (series.colorCounter === colorCount) {
- series.colorCounter = 0;
- }
- }
- else {
- colorIndex = series.colorIndex;
- }
- this.colorIndex = pick(this.colorIndex, colorIndex);
- };
- /**
- * Set a value in an object, on the property defined by key. The key
- * supports nested properties using dot notation. The function modifies the
- * input object and does not make a copy.
- *
- * @function Highcharts.Point#setNestedProperty<T>
- *
- * @param {T} object
- * The object to set the value on.
- *
- * @param {*} value
- * The value to set.
- *
- * @param {string} key
- * Key to the property to set.
- *
- * @return {T}
- * The modified object.
- */
- Point.prototype.setNestedProperty = function (object, value, key) {
- var nestedKeys = key.split('.');
- nestedKeys.reduce(function (result, key, i, arr) {
- var isLastKey = arr.length - 1 === i;
- result[key] = (isLastKey ?
- value :
- isObject(result[key], true) ?
- result[key] :
- {});
- return result[key];
- }, object);
- return object;
- };
- /**
- * Extendable method for formatting each point's tooltip line.
- *
- * @function Highcharts.Point#tooltipFormatter
- *
- * @param {string} pointFormat
- * The point format.
- *
- * @return {string}
- * A string to be concatenated in to the common tooltip text.
- */
- Point.prototype.tooltipFormatter = function (pointFormat) {
- // Insert options for valueDecimals, valuePrefix, and valueSuffix
- var series = this.series, seriesTooltipOptions = series.tooltipOptions, valueDecimals = pick(seriesTooltipOptions.valueDecimals, ''), valuePrefix = seriesTooltipOptions.valuePrefix || '', valueSuffix = seriesTooltipOptions.valueSuffix || '';
- // Replace default point style with class name
- if (series.chart.styledMode) {
- pointFormat =
- series.chart.tooltip.styledModeFormat(pointFormat);
- }
- // Loop over the point array map and replace unformatted values with
- // sprintf formatting markup
- (series.pointArrayMap || ['y']).forEach(function (key) {
- key = '{point.' + key; // without the closing bracket
- if (valuePrefix || valueSuffix) {
- pointFormat = pointFormat.replace(RegExp(key + '}', 'g'), valuePrefix + key + '}' + valueSuffix);
- }
- pointFormat = pointFormat.replace(RegExp(key + '}', 'g'), key + ':,.' + valueDecimals + 'f}');
- });
- return format(pointFormat, {
- point: this,
- series: this.series
- }, series.chart);
- };
- return Point;
- }());
- H.Point = Point;
- return Point;
- });
- _registerModule(_modules, 'Core/Series/Series.js', [_modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (H, LegendSymbolMixin, O, Point, SVGElement, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- animObject = U.animObject,
- arrayMax = U.arrayMax,
- arrayMin = U.arrayMin,
- clamp = U.clamp,
- correctFloat = U.correctFloat,
- defined = U.defined,
- erase = U.erase,
- error = U.error,
- extend = U.extend,
- find = U.find,
- fireEvent = U.fireEvent,
- getNestedProperty = U.getNestedProperty,
- isArray = U.isArray,
- isFunction = U.isFunction,
- isNumber = U.isNumber,
- isString = U.isString,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- removeEvent = U.removeEvent,
- seriesType = U.seriesType,
- splat = U.splat,
- syncTimeout = U.syncTimeout;
- /**
- * This is a placeholder type of the possible series options for
- * [Highcharts](../highcharts/series), [Highstock](../highstock/series),
- * [Highmaps](../highmaps/series), and [Gantt](../gantt/series).
- *
- * In TypeScript is this dynamically generated to reference all possible types
- * of series options.
- *
- * @ignore-declaration
- * @typedef {Highcharts.SeriesOptions|Highcharts.Dictionary<*>} Highcharts.SeriesOptionsType
- */
- /**
- * Options for `dataSorting`.
- *
- * @interface Highcharts.DataSortingOptionsObject
- * @since 8.0.0
- */ /**
- * Enable or disable data sorting for the series.
- * @name Highcharts.DataSortingOptionsObject#enabled
- * @type {boolean|undefined}
- */ /**
- * Whether to allow matching points by name in an update.
- * @name Highcharts.DataSortingOptionsObject#matchByName
- * @type {boolean|undefined}
- */ /**
- * Determines what data value should be used to sort by.
- * @name Highcharts.DataSortingOptionsObject#sortKey
- * @type {string|undefined}
- */
- /**
- * Function callback when a series has been animated.
- *
- * @callback Highcharts.SeriesAfterAnimateCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {Highcharts.SeriesAfterAnimateEventObject} event
- * Event arguments.
- */
- /**
- * Event information regarding completed animation of a series.
- *
- * @interface Highcharts.SeriesAfterAnimateEventObject
- */ /**
- * Animated series.
- * @name Highcharts.SeriesAfterAnimateEventObject#target
- * @type {Highcharts.Series}
- */ /**
- * Event type.
- * @name Highcharts.SeriesAfterAnimateEventObject#type
- * @type {"afterAnimate"}
- */
- /**
- * Function callback when the checkbox next to the series' name in the legend is
- * clicked.
- *
- * @callback Highcharts.SeriesCheckboxClickCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {Highcharts.SeriesCheckboxClickEventObject} event
- * Event arguments.
- */
- /**
- * Event information regarding check of a series box.
- *
- * @interface Highcharts.SeriesCheckboxClickEventObject
- */ /**
- * Whether the box has been checked.
- * @name Highcharts.SeriesCheckboxClickEventObject#checked
- * @type {boolean}
- */ /**
- * Related series.
- * @name Highcharts.SeriesCheckboxClickEventObject#item
- * @type {Highcharts.Series}
- */ /**
- * Related series.
- * @name Highcharts.SeriesCheckboxClickEventObject#target
- * @type {Highcharts.Series}
- */ /**
- * Event type.
- * @name Highcharts.SeriesCheckboxClickEventObject#type
- * @type {"checkboxClick"}
- */
- /**
- * Function callback when a series is clicked. Return false to cancel toogle
- * actions.
- *
- * @callback Highcharts.SeriesClickCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {Highcharts.SeriesClickEventObject} event
- * Event arguments.
- */
- /**
- * Common information for a click event on a series.
- *
- * @interface Highcharts.SeriesClickEventObject
- * @extends global.Event
- */ /**
- * Nearest point on the graph.
- * @name Highcharts.SeriesClickEventObject#point
- * @type {Highcharts.Point}
- */
- /**
- * Gets fired when the series is hidden after chart generation time, either by
- * clicking the legend item or by calling `.hide()`.
- *
- * @callback Highcharts.SeriesHideCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {global.Event} event
- * The event that occured.
- */
- /**
- * The SVG value used for the `stroke-linecap` and `stroke-linejoin` of a line
- * graph.
- *
- * @typedef {"butt"|"round"|"square"|string} Highcharts.SeriesLinecapValue
- */
- /**
- * Gets fired when the legend item belonging to the series is clicked. The
- * default action is to toggle the visibility of the series. This can be
- * prevented by returning `false` or calling `event.preventDefault()`.
- *
- * @callback Highcharts.SeriesLegendItemClickCallbackFunction
- *
- * @param {Highcharts.Series} this
- * The series where the event occured.
- *
- * @param {Highcharts.SeriesLegendItemClickEventObject} event
- * The event that occured.
- */
- /**
- * Information about the event.
- *
- * @interface Highcharts.SeriesLegendItemClickEventObject
- */ /**
- * Related browser event.
- * @name Highcharts.SeriesLegendItemClickEventObject#browserEvent
- * @type {global.PointerEvent}
- */ /**
- * Prevent the default action of toggle the visibility of the series.
- * @name Highcharts.SeriesLegendItemClickEventObject#preventDefault
- * @type {Function}
- */ /**
- * Related series.
- * @name Highcharts.SeriesCheckboxClickEventObject#target
- * @type {Highcharts.Series}
- */ /**
- * Event type.
- * @name Highcharts.SeriesCheckboxClickEventObject#type
- * @type {"checkboxClick"}
- */
- /**
- * Gets fired when the mouse leaves the graph.
- *
- * @callback Highcharts.SeriesMouseOutCallbackFunction
- *
- * @param {Highcharts.Series} this
- * Series where the event occured.
- *
- * @param {global.PointerEvent} event
- * Event that occured.
- */
- /**
- * Gets fired when the mouse enters the graph.
- *
- * @callback Highcharts.SeriesMouseOverCallbackFunction
- *
- * @param {Highcharts.Series} this
- * Series where the event occured.
- *
- * @param {global.PointerEvent} event
- * Event that occured.
- */
- /**
- * Translation and scale for the plot area of a series.
- *
- * @interface Highcharts.SeriesPlotBoxObject
- */ /**
- * @name Highcharts.SeriesPlotBoxObject#scaleX
- * @type {number}
- */ /**
- * @name Highcharts.SeriesPlotBoxObject#scaleY
- * @type {number}
- */ /**
- * @name Highcharts.SeriesPlotBoxObject#translateX
- * @type {number}
- */ /**
- * @name Highcharts.SeriesPlotBoxObject#translateY
- * @type {number}
- */
- /**
- * Gets fired when the series is shown after chart generation time, either by
- * clicking the legend item or by calling `.show()`.
- *
- * @callback Highcharts.SeriesShowCallbackFunction
- *
- * @param {Highcharts.Series} this
- * Series where the event occured.
- *
- * @param {global.Event} event
- * Event that occured.
- */
- /**
- * Possible key values for the series state options.
- *
- * @typedef {"hover"|"inactive"|"normal"|"select"} Highcharts.SeriesStateValue
- */
- ''; // detach doclets above
- var seriesTypes = H.seriesTypes,
- win = H.win;
- /**
- * This is the base series prototype that all other series types inherit from.
- * A new series is initialized either through the
- * [series](https://api.highcharts.com/highcharts/series)
- * option structure, or after the chart is initialized, through
- * {@link Highcharts.Chart#addSeries}.
- *
- * The object can be accessed in a number of ways. All series and point event
- * handlers give a reference to the `series` object. The chart object has a
- * {@link Highcharts.Chart#series|series} property that is a collection of all
- * the chart's series. The point objects and axis objects also have the same
- * reference.
- *
- * Another way to reference the series programmatically is by `id`. Add an id
- * in the series configuration options, and get the series object by
- * {@link Highcharts.Chart#get}.
- *
- * Configuration options for the series are given in three levels. Options for
- * all series in a chart are given in the
- * [plotOptions.series](https://api.highcharts.com/highcharts/plotOptions.series)
- * object. Then options for all series of a specific type
- * are given in the plotOptions of that type, for example `plotOptions.line`.
- * Next, options for one single series are given in the series array, or as
- * arguments to `chart.addSeries`.
- *
- * The data in the series is stored in various arrays.
- *
- * - First, `series.options.data` contains all the original config options for
- * each point whether added by options or methods like `series.addPoint`.
- *
- * - Next, `series.data` contains those values converted to points, but in case
- * the series data length exceeds the `cropThreshold`, or if the data is
- * grouped, `series.data` doesn't contain all the points. It only contains the
- * points that have been created on demand.
- *
- * - Then there's `series.points` that contains all currently visible point
- * objects. In case of cropping, the cropped-away points are not part of this
- * array. The `series.points` array starts at `series.cropStart` compared to
- * `series.data` and `series.options.data`. If however the series data is
- * grouped, these can't be correlated one to one.
- *
- * - `series.xData` and `series.processedXData` contain clean x values,
- * equivalent to `series.data` and `series.points`.
- *
- * - `series.yData` and `series.processedYData` contain clean y values,
- * equivalent to `series.data` and `series.points`.
- *
- * @class
- * @name Highcharts.Series
- *
- * @param {Highcharts.Chart} chart
- * The chart instance.
- *
- * @param {Highcharts.SeriesOptionsType|object} options
- * The series options.
- */ /**
- * The line series is the base type and is therefor the series base prototype.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.line
- *
- * @augments Highcharts.Series
- */
- H.Series = seriesType('line',
- /**
- * Series options for specific data and the data itself. In TypeScript you
- * have to cast the series options to specific series types, to get all
- * possible options for a series.
- *
- * @example
- * // TypeScript example
- * Highcharts.chart('container', {
- * series: [{
- * color: '#06C',
- * data: [[0, 1], [2, 3]]
- * } as Highcharts.SeriesLineOptions ]
- * });
- *
- * @type {Array<*>}
- * @apioption series
- */
- /**
- * An id for the series. This can be used after render time to get a pointer
- * to the series object through `chart.get()`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-id/
- * Get series by id
- *
- * @type {string}
- * @since 1.2.0
- * @apioption series.id
- */
- /**
- * The index of the series in the chart, affecting the internal index in the
- * `chart.series` array, the visible Z index as well as the order in the
- * legend.
- *
- * @type {number}
- * @since 2.3.0
- * @apioption series.index
- */
- /**
- * The sequential index of the series in the legend.
- *
- * @see [legend.reversed](#legend.reversed),
- * [yAxis.reversedStacks](#yAxis.reversedStacks)
- *
- * @sample {highcharts|highstock} highcharts/series/legendindex/
- * Legend in opposite order
- *
- * @type {number}
- * @apioption series.legendIndex
- */
- /**
- * The name of the series as shown in the legend, tooltip etc.
- *
- * @sample {highcharts} highcharts/series/name/
- * Series name
- * @sample {highmaps} maps/demo/category-map/
- * Series name
- *
- * @type {string}
- * @apioption series.name
- */
- /**
- * This option allows grouping series in a stacked chart. The stack option
- * can be a string or anything else, as long as the grouped series' stack
- * options match each other after conversion into a string.
- *
- * @sample {highcharts} highcharts/series/stack/
- * Stacked and grouped columns
- *
- * @type {number|string}
- * @since 2.1
- * @product highcharts highstock
- * @apioption series.stack
- */
- /**
- * The type of series, for example `line` or `column`. By default, the
- * series type is inherited from [chart.type](#chart.type), so unless the
- * chart is a combination of series types, there is no need to set it on the
- * series level.
- *
- * @sample {highcharts} highcharts/series/type/
- * Line and column in the same chart
- * @sample highcharts/series/type-dynamic/
- * Dynamic types with button selector
- * @sample {highmaps} maps/demo/mapline-mappoint/
- * Multiple types in the same map
- *
- * @type {string}
- * @apioption series.type
- */
- /**
- * When using dual or multiple x axes, this number defines which xAxis the
- * particular series is connected to. It refers to either the
- * {@link #xAxis.id|axis id}
- * or the index of the axis in the xAxis array, with 0 being the first.
- *
- * @type {number|string}
- * @default 0
- * @product highcharts highstock
- * @apioption series.xAxis
- */
- /**
- * When using dual or multiple y axes, this number defines which yAxis the
- * particular series is connected to. It refers to either the
- * {@link #yAxis.id|axis id}
- * or the index of the axis in the yAxis array, with 0 being the first.
- *
- * @sample {highcharts} highcharts/series/yaxis/
- * Apply the column series to the secondary Y axis
- *
- * @type {number|string}
- * @default 0
- * @product highcharts highstock
- * @apioption series.yAxis
- */
- /**
- * Define the visual z index of the series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-zindex-default/
- * With no z index, the series defined last are on top
- * @sample {highcharts} highcharts/plotoptions/series-zindex/
- * With a z index, the series with the highest z index is on top
- * @sample {highstock} highcharts/plotoptions/series-zindex-default/
- * With no z index, the series defined last are on top
- * @sample {highstock} highcharts/plotoptions/series-zindex/
- * With a z index, the series with the highest z index is on top
- *
- * @type {number}
- * @product highcharts highstock
- * @apioption series.zIndex
- */
- null,
- /**
- * General options for all series types.
- *
- * @optionparent plotOptions.series
- */
- {
- /**
- * The SVG value used for the `stroke-linecap` and `stroke-linejoin`
- * of a line graph. Round means that lines are rounded in the ends and
- * bends.
- *
- * @type {Highcharts.SeriesLinecapValue}
- * @default round
- * @since 3.0.7
- * @apioption plotOptions.line.linecap
- */
- /**
- * Pixel width of the graph line.
- *
- * @see In styled mode, the line stroke-width can be set with the
- * `.highcharts-graph` class name.
- *
- * @sample {highcharts} highcharts/plotoptions/series-linewidth-general/
- * On all series
- * @sample {highcharts} highcharts/plotoptions/series-linewidth-specific/
- * On one single series
- *
- * @product highcharts highstock
- *
- * @private
- */
- lineWidth: 2,
- /**
- * For some series, there is a limit that shuts down initial animation
- * by default when the total number of points in the chart is too high.
- * For example, for a column chart and its derivatives, animation does
- * not run if there is more than 250 points totally. To disable this
- * cap, set `animationLimit` to `Infinity`.
- *
- * @type {number}
- * @apioption plotOptions.series.animationLimit
- */
- /**
- * Allow this series' points to be selected by clicking on the graphic
- * (columns, point markers, pie slices, map areas etc).
- *
- * The selected points can be handled by point select and unselect
- * events, or collectively by the [getSelectedPoints
- * ](/class-reference/Highcharts.Chart#getSelectedPoints) function.
- *
- * And alternative way of selecting points is through dragging.
- *
- * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-line/
- * Line
- * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-column/
- * Column
- * @sample {highcharts} highcharts/plotoptions/series-allowpointselect-pie/
- * Pie
- * @sample {highcharts} highcharts/chart/events-selection-points/
- * Select a range of points through a drag selection
- * @sample {highmaps} maps/plotoptions/series-allowpointselect/
- * Map area
- * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/
- * Map bubble
- *
- * @since 1.2.0
- *
- * @private
- */
- allowPointSelect: false,
- /**
- * When true, each point or column edge is rounded to its nearest pixel
- * in order to render sharp on screen. In some cases, when there are a
- * lot of densely packed columns, this leads to visible difference
- * in column widths or distance between columns. In these cases,
- * setting `crisp` to `false` may look better, even though each column
- * is rendered blurry.
- *
- * @sample {highcharts} highcharts/plotoptions/column-crisp-false/
- * Crisp is false
- *
- * @since 5.0.10
- * @product highcharts highstock gantt
- *
- * @private
- */
- crisp: true,
- /**
- * If true, a checkbox is displayed next to the legend item to allow
- * selecting the series. The state of the checkbox is determined by
- * the `selected` option.
- *
- * @productdesc {highmaps}
- * Note that if a `colorAxis` is defined, the color axis is represented
- * in the legend, not the series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-showcheckbox-true/
- * Show select box
- *
- * @since 1.2.0
- *
- * @private
- */
- showCheckbox: false,
- /**
- * Enable or disable the initial animation when a series is displayed.
- * The animation can also be set as a configuration object. Please
- * note that this option only applies to the initial animation of the
- * series itself. For other animations, see [chart.animation](
- * #chart.animation) and the animation parameter under the API methods.
- * The following properties are supported:
- *
- * - `defer`: The animation delay time in milliseconds.
- *
- * - `duration`: The duration of the animation in milliseconds.
- *
- * - `easing`: Can be a string reference to an easing function set on
- * the `Math` object or a function. See the _Custom easing function_
- * demo below.
- *
- * Due to poor performance, animation is disabled in old IE browsers
- * for several chart types.
- *
- * @sample {highcharts} highcharts/plotoptions/series-animation-disabled/
- * Animation disabled
- * @sample {highcharts} highcharts/plotoptions/series-animation-slower/
- * Slower animation
- * @sample {highcharts} highcharts/plotoptions/series-animation-easing/
- * Custom easing function
- * @sample {highstock} stock/plotoptions/animation-slower/
- * Slower animation
- * @sample {highstock} stock/plotoptions/animation-easing/
- * Custom easing function
- * @sample {highmaps} maps/plotoptions/series-animation-true/
- * Animation enabled on map series
- * @sample {highmaps} maps/plotoptions/mapbubble-animation-false/
- * Disabled on mapbubble series
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @default {highcharts} true
- * @default {highstock} true
- * @default {highmaps} false
- *
- * @private
- */
- animation: {
- /** @internal */
- duration: 1000
- },
- /**
- * @default 0
- * @type {number}
- * @since 8.2.0
- * @apioption plotOptions.series.animation.defer
- */
- /**
- * An additional class name to apply to the series' graphical elements.
- * This option does not replace default class names of the graphical
- * element.
- *
- * @type {string}
- * @since 5.0.0
- * @apioption plotOptions.series.className
- */
- /**
- * Disable this option to allow series rendering in the whole plotting
- * area.
- *
- * **Note:** Clipping should be always enabled when
- * [chart.zoomType](#chart.zoomType) is set
- *
- * @sample {highcharts} highcharts/plotoptions/series-clip/
- * Disabled clipping
- *
- * @default true
- * @type {boolean}
- * @since 3.0.0
- * @apioption plotOptions.series.clip
- */
- /**
- * The main color of the series. In line type series it applies to the
- * line and the point markers unless otherwise specified. In bar type
- * series it applies to the bars unless a color is specified per point.
- * The default value is pulled from the `options.colors` array.
- *
- * In styled mode, the color can be defined by the
- * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series
- * color can be set with the `.highcharts-series`,
- * `.highcharts-color-{n}`, `.highcharts-{type}-series` or
- * `.highcharts-series-{n}` class, or individual classes given by the
- * `className` option.
- *
- * @productdesc {highmaps}
- * In maps, the series color is rarely used, as most choropleth maps use
- * the color to denote the value of each point. The series color can
- * however be used in a map with multiple series holding categorized
- * data.
- *
- * @sample {highcharts} highcharts/plotoptions/series-color-general/
- * General plot option
- * @sample {highcharts} highcharts/plotoptions/series-color-specific/
- * One specific series
- * @sample {highcharts} highcharts/plotoptions/series-color-area/
- * Area color
- * @sample {highcharts} highcharts/series/infographic/
- * Pattern fill
- * @sample {highmaps} maps/demo/category-map/
- * Category map by multiple series
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption plotOptions.series.color
- */
- /**
- * Styled mode only. A specific color index to use for the series, so
- * its graphic representations are given the class name
- * `highcharts-color-{n}`.
- *
- * @type {number}
- * @since 5.0.0
- * @apioption plotOptions.series.colorIndex
- */
- /**
- * Whether to connect a graph line across null points, or render a gap
- * between the two points on either side of the null.
- *
- * @sample {highcharts} highcharts/plotoptions/series-connectnulls-false/
- * False by default
- * @sample {highcharts} highcharts/plotoptions/series-connectnulls-true/
- * True
- *
- * @type {boolean}
- * @default false
- * @product highcharts highstock
- * @apioption plotOptions.series.connectNulls
- */
- /**
- * You can set the cursor to "pointer" if you have click events attached
- * to the series, to signal to the user that the points and lines can
- * be clicked.
- *
- * In styled mode, the series cursor can be set with the same classes
- * as listed under [series.color](#plotOptions.series.color).
- *
- * @sample {highcharts} highcharts/plotoptions/series-cursor-line/
- * On line graph
- * @sample {highcharts} highcharts/plotoptions/series-cursor-column/
- * On columns
- * @sample {highcharts} highcharts/plotoptions/series-cursor-scatter/
- * On scatter markers
- * @sample {highstock} stock/plotoptions/cursor/
- * Pointer on a line graph
- * @sample {highmaps} maps/plotoptions/series-allowpointselect/
- * Map area
- * @sample {highmaps} maps/plotoptions/mapbubble-allowpointselect/
- * Map bubble
- *
- * @type {string|Highcharts.CursorValue}
- * @apioption plotOptions.series.cursor
- */
- /**
- * A reserved subspace to store options and values for customized
- * functionality. Here you can add additional data for your own event
- * callbacks and formatter callbacks.
- *
- * @sample {highcharts} highcharts/point/custom/
- * Point and series with custom data
- *
- * @type {Highcharts.Dictionary<*>}
- * @apioption plotOptions.series.custom
- */
- /**
- * Name of the dash style to use for the graph, or for some series types
- * the outline of each shape.
- *
- * In styled mode, the
- * [stroke dash-array](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/series-dashstyle/)
- * can be set with the same classes as listed under
- * [series.color](#plotOptions.series.color).
- *
- * @sample {highcharts} highcharts/plotoptions/series-dashstyle-all/
- * Possible values demonstrated
- * @sample {highcharts} highcharts/plotoptions/series-dashstyle/
- * Chart suitable for printing in black and white
- * @sample {highstock} highcharts/plotoptions/series-dashstyle-all/
- * Possible values demonstrated
- * @sample {highmaps} highcharts/plotoptions/series-dashstyle-all/
- * Possible values demonstrated
- * @sample {highmaps} maps/plotoptions/series-dashstyle/
- * Dotted borders on a map
- *
- * @type {Highcharts.DashStyleValue}
- * @default Solid
- * @since 2.1
- * @apioption plotOptions.series.dashStyle
- */
- /**
- * A description of the series to add to the screen reader information
- * about the series.
- *
- * @type {string}
- * @since 5.0.0
- * @requires modules/accessibility
- * @apioption plotOptions.series.description
- */
- /**
- * Options for the series data sorting.
- *
- * @type {Highcharts.DataSortingOptionsObject}
- * @since 8.0.0
- * @product highcharts highstock
- * @apioption plotOptions.series.dataSorting
- */
- /**
- * Enable or disable data sorting for the series. Use [xAxis.reversed](
- * #xAxis.reversed) to change the sorting order.
- *
- * @sample {highcharts} highcharts/datasorting/animation/
- * Data sorting in scatter-3d
- * @sample {highcharts} highcharts/datasorting/labels-animation/
- * Axis labels animation
- * @sample {highcharts} highcharts/datasorting/dependent-sorting/
- * Dependent series sorting
- * @sample {highcharts} highcharts/datasorting/independent-sorting/
- * Independent series sorting
- *
- * @type {boolean}
- * @since 8.0.0
- * @apioption plotOptions.series.dataSorting.enabled
- */
- /**
- * Whether to allow matching points by name in an update. If this option
- * is disabled, points will be matched by order.
- *
- * @sample {highcharts} highcharts/datasorting/match-by-name/
- * Enabled match by name
- *
- * @type {boolean}
- * @since 8.0.0
- * @apioption plotOptions.series.dataSorting.matchByName
- */
- /**
- * Determines what data value should be used to sort by.
- *
- * @sample {highcharts} highcharts/datasorting/sort-key/
- * Sort key as `z` value
- *
- * @type {string}
- * @since 8.0.0
- * @default y
- * @apioption plotOptions.series.dataSorting.sortKey
- */
- /**
- * Enable or disable the mouse tracking for a specific series. This
- * includes point tooltips and click events on graphs and points. For
- * large datasets it improves performance.
- *
- * @sample {highcharts} highcharts/plotoptions/series-enablemousetracking-false/
- * No mouse tracking
- * @sample {highmaps} maps/plotoptions/series-enablemousetracking-false/
- * No mouse tracking
- *
- * @type {boolean}
- * @default true
- * @apioption plotOptions.series.enableMouseTracking
- */
- /**
- * Whether to use the Y extremes of the total chart width or only the
- * zoomed area when zooming in on parts of the X axis. By default, the
- * Y axis adjusts to the min and max of the visible data. Cartesian
- * series only.
- *
- * @type {boolean}
- * @default false
- * @since 4.1.6
- * @product highcharts highstock gantt
- * @apioption plotOptions.series.getExtremesFromAll
- */
- /**
- * An array specifying which option maps to which key in the data point
- * array. This makes it convenient to work with unstructured data arrays
- * from different sources.
- *
- * @see [series.data](#series.line.data)
- *
- * @sample {highcharts|highstock} highcharts/series/data-keys/
- * An extended data array with keys
- * @sample {highcharts|highstock} highcharts/series/data-nested-keys/
- * Nested keys used to access object properties
- *
- * @type {Array<string>}
- * @since 4.1.6
- * @apioption plotOptions.series.keys
- */
- /**
- * The line cap used for line ends and line joins on the graph.
- *
- * @type {Highcharts.SeriesLinecapValue}
- * @default round
- * @product highcharts highstock
- * @apioption plotOptions.series.linecap
- */
- /**
- * The [id](#series.id) of another series to link to. Additionally,
- * the value can be ":previous" to link to the previous series. When
- * two series are linked, only the first one appears in the legend.
- * Toggling the visibility of this also toggles the linked series.
- *
- * If master series uses data sorting and linked series does not have
- * its own sorting definition, the linked series will be sorted in the
- * same order as the master one.
- *
- * @sample {highcharts|highstock} highcharts/demo/arearange-line/
- * Linked series
- *
- * @type {string}
- * @since 3.0
- * @product highcharts highstock gantt
- * @apioption plotOptions.series.linkedTo
- */
- /**
- * Options for the corresponding navigator series if `showInNavigator`
- * is `true` for this series. Available options are the same as any
- * series, documented at [plotOptions](#plotOptions.series) and
- * [series](#series).
- *
- * These options are merged with options in [navigator.series](
- * #navigator.series), and will take precedence if the same option is
- * defined both places.
- *
- * @see [navigator.series](#navigator.series)
- *
- * @type {Highcharts.PlotSeriesOptions}
- * @since 5.0.0
- * @product highstock
- * @apioption plotOptions.series.navigatorOptions
- */
- /**
- * The color for the parts of the graph or points that are below the
- * [threshold](#plotOptions.series.threshold). Note that `zones` takes
- * precedence over the negative color. Using `negativeColor` is
- * equivalent to applying a zone with value of 0.
- *
- * @see In styled mode, a negative color is applied by setting this option
- * to `true` combined with the `.highcharts-negative` class name.
- *
- * @sample {highcharts} highcharts/plotoptions/series-negative-color/
- * Spline, area and column
- * @sample {highcharts} highcharts/plotoptions/arearange-negativecolor/
- * Arearange
- * @sample {highcharts} highcharts/css/series-negative-color/
- * Styled mode
- * @sample {highstock} highcharts/plotoptions/series-negative-color/
- * Spline, area and column
- * @sample {highstock} highcharts/plotoptions/arearange-negativecolor/
- * Arearange
- * @sample {highmaps} highcharts/plotoptions/series-negative-color/
- * Spline, area and column
- * @sample {highmaps} highcharts/plotoptions/arearange-negativecolor/
- * Arearange
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 3.0
- * @apioption plotOptions.series.negativeColor
- */
- /**
- * Same as
- * [accessibility.pointDescriptionFormatter](#accessibility.pointDescriptionFormatter),
- * but for an individual series. Overrides the chart wide configuration.
- *
- * @type {Function}
- * @since 5.0.12
- * @apioption plotOptions.series.pointDescriptionFormatter
- */
- /**
- * If no x values are given for the points in a series, `pointInterval`
- * defines the interval of the x values. For example, if a series
- * contains one value every decade starting from year 0, set
- * `pointInterval` to `10`. In true `datetime` axes, the `pointInterval`
- * is set in milliseconds.
- *
- * It can be also be combined with `pointIntervalUnit` to draw irregular
- * time intervals.
- *
- * Please note that this options applies to the _series data_, not the
- * interval of the axis ticks, which is independent.
- *
- * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/
- * Datetime X axis
- * @sample {highstock} stock/plotoptions/pointinterval-pointstart/
- * Using pointStart and pointInterval
- *
- * @type {number}
- * @default 1
- * @product highcharts highstock gantt
- * @apioption plotOptions.series.pointInterval
- */
- /**
- * On datetime series, this allows for setting the
- * [pointInterval](#plotOptions.series.pointInterval) to irregular time
- * units, `day`, `month` and `year`. A day is usually the same as 24
- * hours, but `pointIntervalUnit` also takes the DST crossover into
- * consideration when dealing with local time. Combine this option with
- * `pointInterval` to draw weeks, quarters, 6 months, 10 years etc.
- *
- * Please note that this options applies to the _series data_, not the
- * interval of the axis ticks, which is independent.
- *
- * @sample {highcharts} highcharts/plotoptions/series-pointintervalunit/
- * One point a month
- * @sample {highstock} highcharts/plotoptions/series-pointintervalunit/
- * One point a month
- *
- * @type {string}
- * @since 4.1.0
- * @product highcharts highstock gantt
- * @validvalue ["day", "month", "year"]
- * @apioption plotOptions.series.pointIntervalUnit
- */
- /**
- * Possible values: `"on"`, `"between"`, `number`.
- *
- * In a column chart, when pointPlacement is `"on"`, the point will not
- * create any padding of the X axis. In a polar column chart this means
- * that the first column points directly north. If the pointPlacement is
- * `"between"`, the columns will be laid out between ticks. This is
- * useful for example for visualising an amount between two points in
- * time or in a certain sector of a polar chart.
- *
- * Since Highcharts 3.0.2, the point placement can also be numeric,
- * where 0 is on the axis value, -0.5 is between this value and the
- * previous, and 0.5 is between this value and the next. Unlike the
- * textual options, numeric point placement options won't affect axis
- * padding.
- *
- * Note that pointPlacement needs a [pointRange](
- * #plotOptions.series.pointRange) to work. For column series this is
- * computed, but for line-type series it needs to be set.
- *
- * For the `xrange` series type and gantt charts, if the Y axis is a
- * category axis, the `pointPlacement` applies to the Y axis rather than
- * the (typically datetime) X axis.
- *
- * Defaults to `undefined` in cartesian charts, `"between"` in polar
- * charts.
- *
- * @see [xAxis.tickmarkPlacement](#xAxis.tickmarkPlacement)
- *
- * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-between/
- * Between in a column chart
- * @sample {highcharts|highstock} highcharts/plotoptions/series-pointplacement-numeric/
- * Numeric placement for custom layout
- * @sample {highcharts|highstock} maps/plotoptions/heatmap-pointplacement/
- * Placement in heatmap
- *
- * @type {string|number}
- * @since 2.3.0
- * @product highcharts highstock gantt
- * @apioption plotOptions.series.pointPlacement
- */
- /**
- * If no x values are given for the points in a series, pointStart
- * defines on what value to start. For example, if a series contains one
- * yearly value starting from 1945, set pointStart to 1945.
- *
- * @sample {highcharts} highcharts/plotoptions/series-pointstart-linear/
- * Linear
- * @sample {highcharts} highcharts/plotoptions/series-pointstart-datetime/
- * Datetime
- * @sample {highstock} stock/plotoptions/pointinterval-pointstart/
- * Using pointStart and pointInterval
- *
- * @type {number}
- * @default 0
- * @product highcharts highstock gantt
- * @apioption plotOptions.series.pointStart
- */
- /**
- * Whether to select the series initially. If `showCheckbox` is true,
- * the checkbox next to the series name in the legend will be checked
- * for a selected series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-selected/
- * One out of two series selected
- *
- * @type {boolean}
- * @default false
- * @since 1.2.0
- * @apioption plotOptions.series.selected
- */
- /**
- * Whether to apply a drop shadow to the graph line. Since 2.3 the
- * shadow can be an object configuration containing `color`, `offsetX`,
- * `offsetY`, `opacity` and `width`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-shadow/
- * Shadow enabled
- *
- * @type {boolean|Highcharts.ShadowOptionsObject}
- * @default false
- * @apioption plotOptions.series.shadow
- */
- /**
- * Whether to display this particular series or series type in the
- * legend. Standalone series are shown in legend by default, and linked
- * series are not. Since v7.2.0 it is possible to show series that use
- * colorAxis by setting this option to `true`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-showinlegend/
- * One series in the legend, one hidden
- *
- * @type {boolean}
- * @apioption plotOptions.series.showInLegend
- */
- /**
- * Whether or not to show the series in the navigator. Takes precedence
- * over [navigator.baseSeries](#navigator.baseSeries) if defined.
- *
- * @type {boolean}
- * @since 5.0.0
- * @product highstock
- * @apioption plotOptions.series.showInNavigator
- */
- /**
- * If set to `true`, the accessibility module will skip past the points
- * in this series for keyboard navigation.
- *
- * @type {boolean}
- * @since 5.0.12
- * @apioption plotOptions.series.skipKeyboardNavigation
- */
- /**
- * Whether to stack the values of each series on top of each other.
- * Possible values are `undefined` to disable, `"normal"` to stack by
- * value or `"percent"`.
- *
- * When stacking is enabled, data must be sorted
- * in ascending X order.
- *
- * Some stacking options are related to specific series types. In the
- * streamgraph series type, the stacking option is set to `"stream"`.
- * The second one is `"overlap"`, which only applies to waterfall
- * series.
- *
- * @see [yAxis.reversedStacks](#yAxis.reversedStacks)
- *
- * @sample {highcharts} highcharts/plotoptions/series-stacking-line/
- * Line
- * @sample {highcharts} highcharts/plotoptions/series-stacking-column/
- * Column
- * @sample {highcharts} highcharts/plotoptions/series-stacking-bar/
- * Bar
- * @sample {highcharts} highcharts/plotoptions/series-stacking-area/
- * Area
- * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-line/
- * Line
- * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-column/
- * Column
- * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-bar/
- * Bar
- * @sample {highcharts} highcharts/plotoptions/series-stacking-percent-area/
- * Area
- * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-normal-stacking
- * Waterfall with normal stacking
- * @sample {highcharts} highcharts/plotoptions/series-waterfall-with-overlap-stacking
- * Waterfall with overlap stacking
- * @sample {highstock} stock/plotoptions/stacking/
- * Area
- *
- * @type {string}
- * @product highcharts highstock
- * @validvalue ["normal", "overlap", "percent", "stream"]
- * @apioption plotOptions.series.stacking
- */
- /**
- * Whether to apply steps to the line. Possible values are `left`,
- * `center` and `right`.
- *
- * @sample {highcharts} highcharts/plotoptions/line-step/
- * Different step line options
- * @sample {highcharts} highcharts/plotoptions/area-step/
- * Stepped, stacked area
- * @sample {highstock} stock/plotoptions/line-step/
- * Step line
- *
- * @type {string}
- * @since 1.2.5
- * @product highcharts highstock
- * @validvalue ["left", "center", "right"]
- * @apioption plotOptions.series.step
- */
- /**
- * The threshold, also called zero level or base level. For line type
- * series this is only used in conjunction with
- * [negativeColor](#plotOptions.series.negativeColor).
- *
- * @see [softThreshold](#plotOptions.series.softThreshold).
- *
- * @type {number}
- * @default 0
- * @since 3.0
- * @product highcharts highstock
- * @apioption plotOptions.series.threshold
- */
- /**
- * Set the initial visibility of the series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-visible/
- * Two series, one hidden and one visible
- * @sample {highstock} stock/plotoptions/series-visibility/
- * Hidden series
- *
- * @type {boolean}
- * @default true
- * @apioption plotOptions.series.visible
- */
- /**
- * Defines the Axis on which the zones are applied.
- *
- * @see [zones](#plotOptions.series.zones)
- *
- * @sample {highcharts} highcharts/series/color-zones-zoneaxis-x/
- * Zones on the X-Axis
- * @sample {highstock} highcharts/series/color-zones-zoneaxis-x/
- * Zones on the X-Axis
- *
- * @type {string}
- * @default y
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zoneAxis
- */
- /**
- * General event handlers for the series items. These event hooks can
- * also be attached to the series at run time using the
- * `Highcharts.addEvent` function.
- *
- * @declare Highcharts.SeriesEventsOptionsObject
- *
- * @private
- */
- events: {},
- /**
- * Fires after the series has finished its initial animation, or in case
- * animation is disabled, immediately as the series is displayed.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-afteranimate/
- * Show label after animate
- * @sample {highstock} highcharts/plotoptions/series-events-afteranimate/
- * Show label after animate
- *
- * @type {Highcharts.SeriesAfterAnimateCallbackFunction}
- * @since 4.0
- * @product highcharts highstock gantt
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.afterAnimate
- */
- /**
- * Fires when the checkbox next to the series' name in the legend is
- * clicked. One parameter, `event`, is passed to the function. The state
- * of the checkbox is found by `event.checked`. The checked item is
- * found by `event.item`. Return `false` to prevent the default action
- * which is to toggle the select state of the series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/
- * Alert checkbox status
- *
- * @type {Highcharts.SeriesCheckboxClickCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.checkboxClick
- */
- /**
- * Fires when the series is clicked. One parameter, `event`, is passed
- * to the function, containing common event information. Additionally,
- * `event.point` holds a pointer to the nearest point on the graph.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-click/
- * Alert click info
- * @sample {highstock} stock/plotoptions/series-events-click/
- * Alert click info
- * @sample {highmaps} maps/plotoptions/series-events-click/
- * Display click info in subtitle
- *
- * @type {Highcharts.SeriesClickCallbackFunction}
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.click
- */
- /**
- * Fires when the series is hidden after chart generation time, either
- * by clicking the legend item or by calling `.hide()`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-hide/
- * Alert when the series is hidden by clicking the legend item
- *
- * @type {Highcharts.SeriesHideCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.hide
- */
- /**
- * Fires when the legend item belonging to the series is clicked. One
- * parameter, `event`, is passed to the function. The default action
- * is to toggle the visibility of the series. This can be prevented
- * by returning `false` or calling `event.preventDefault()`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-legenditemclick/
- * Confirm hiding and showing
- *
- * @type {Highcharts.SeriesLegendItemClickCallbackFunction}
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.legendItemClick
- */
- /**
- * Fires when the mouse leaves the graph. One parameter, `event`, is
- * passed to the function, containing common event information. If the
- * [stickyTracking](#plotOptions.series) option is true, `mouseOut`
- * doesn't happen before the mouse enters another graph or leaves the
- * plot area.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/
- * With sticky tracking by default
- * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/
- * Without sticky tracking
- *
- * @type {Highcharts.SeriesMouseOutCallbackFunction}
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.mouseOut
- */
- /**
- * Fires when the mouse enters the graph. One parameter, `event`, is
- * passed to the function, containing common event information.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-sticky/
- * With sticky tracking by default
- * @sample {highcharts} highcharts/plotoptions/series-events-mouseover-no-sticky/
- * Without sticky tracking
- *
- * @type {Highcharts.SeriesMouseOverCallbackFunction}
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.mouseOver
- */
- /**
- * Fires when the series is shown after chart generation time, either
- * by clicking the legend item or by calling `.show()`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-show/
- * Alert when the series is shown by clicking the legend item.
- *
- * @type {Highcharts.SeriesShowCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Series
- * @apioption plotOptions.series.events.show
- */
- /**
- * Options for the point markers of line-like series. Properties like
- * `fillColor`, `lineColor` and `lineWidth` define the visual appearance
- * of the markers. Other series types, like column series, don't have
- * markers, but have visual options on the series level instead.
- *
- * In styled mode, the markers can be styled with the
- * `.highcharts-point`, `.highcharts-point-hover` and
- * `.highcharts-point-select` class names.
- *
- * @declare Highcharts.PointMarkerOptionsObject
- *
- * @private
- */
- marker: {
- /**
- * Enable or disable the point marker. If `undefined`, the markers
- * are hidden when the data is dense, and shown for more widespread
- * data points.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-enabled/
- * Disabled markers
- * @sample {highcharts} highcharts/plotoptions/series-marker-enabled-false/
- * Disabled in normal state but enabled on hover
- * @sample {highstock} stock/plotoptions/series-marker/
- * Enabled markers
- *
- * @type {boolean}
- * @default {highcharts} undefined
- * @default {highstock} false
- * @apioption plotOptions.series.marker.enabled
- */
- /**
- * The threshold for how dense the point markers should be before
- * they are hidden, given that `enabled` is not defined. The number
- * indicates the horizontal distance between the two closest points
- * in the series, as multiples of the `marker.radius`. In other
- * words, the default value of 2 means points are hidden if
- * overlapping horizontally.
- *
- * @sample highcharts/plotoptions/series-marker-enabledthreshold
- * A higher threshold
- *
- * @since 6.0.5
- */
- enabledThreshold: 2,
- /**
- * The fill color of the point marker. When `undefined`, the series'
- * or point's color is used.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/
- * White fill
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption plotOptions.series.marker.fillColor
- */
- /**
- * Image markers only. Set the image width explicitly. When using
- * this option, a `width` must also be set.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/
- * Fixed width and height
- * @sample {highstock} highcharts/plotoptions/series-marker-width-height/
- * Fixed width and height
- *
- * @type {number}
- * @since 4.0.4
- * @apioption plotOptions.series.marker.height
- */
- /**
- * The color of the point marker's outline. When `undefined`, the
- * series' or point's color is used.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/
- * Inherit from series color (undefined)
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- lineColor: '#ffffff',
- /**
- * The width of the point marker's outline.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-fillcolor/
- * 2px blue marker
- */
- lineWidth: 0,
- /**
- * The radius of the point marker.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-radius/
- * Bigger markers
- *
- * @default {highstock} 2
- */
- radius: 4,
- /**
- * A predefined shape or symbol for the marker. When undefined, the
- * symbol is pulled from options.symbols. Other possible values are
- * `'circle'`, `'square'`,`'diamond'`, `'triangle'` and
- * `'triangle-down'`.
- *
- * Additionally, the URL to a graphic can be given on this form:
- * `'url(graphic.png)'`. Note that for the image to be applied to
- * exported charts, its URL needs to be accessible by the export
- * server.
- *
- * Custom callbacks for symbol path generation can also be added to
- * `Highcharts.SVGRenderer.prototype.symbols`. The callback is then
- * used by its method name, as shown in the demo.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-symbol/
- * Predefined, graphic and custom markers
- * @sample {highstock} highcharts/plotoptions/series-marker-symbol/
- * Predefined, graphic and custom markers
- *
- * @type {string}
- * @apioption plotOptions.series.marker.symbol
- */
- /**
- * Image markers only. Set the image width explicitly. When using
- * this option, a `height` must also be set.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-width-height/
- * Fixed width and height
- * @sample {highstock} highcharts/plotoptions/series-marker-width-height/
- * Fixed width and height
- *
- * @type {number}
- * @since 4.0.4
- * @apioption plotOptions.series.marker.width
- */
- /**
- * States for a single point marker.
- *
- * @declare Highcharts.PointStatesOptionsObject
- */
- states: {
- /**
- * The normal state of a single point marker. Currently only
- * used for setting animation when returning to normal state
- * from hover.
- *
- * @declare Highcharts.PointStatesNormalOptionsObject
- */
- normal: {
- /**
- * Animation when returning to normal state after hovering.
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- */
- animation: true
- },
- /**
- * The hover state for a single point marker.
- *
- * @declare Highcharts.PointStatesHoverOptionsObject
- */
- hover: {
- /**
- * Animation when hovering over the marker.
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- */
- animation: {
- /** @internal */
- duration: 50
- },
- /**
- * Enable or disable the point marker.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-enabled/
- * Disabled hover state
- */
- enabled: true,
- /**
- * The fill color of the marker in hover state. When
- * `undefined`, the series' or point's fillColor for normal
- * state is used.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption plotOptions.series.marker.states.hover.fillColor
- */
- /**
- * The color of the point marker's outline. When
- * `undefined`, the series' or point's lineColor for normal
- * state is used.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linecolor/
- * White fill color, black line color
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption plotOptions.series.marker.states.hover.lineColor
- */
- /**
- * The width of the point marker's outline. When
- * `undefined`, the series' or point's lineWidth for normal
- * state is used.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-linewidth/
- * 3px line width
- *
- * @type {number}
- * @apioption plotOptions.series.marker.states.hover.lineWidth
- */
- /**
- * The radius of the point marker. In hover state, it
- * defaults to the normal state's radius + 2 as per the
- * [radiusPlus](#plotOptions.series.marker.states.hover.radiusPlus)
- * option.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-hover-radius/
- * 10px radius
- *
- * @type {number}
- * @apioption plotOptions.series.marker.states.hover.radius
- */
- /**
- * The number of pixels to increase the radius of the
- * hovered point.
- *
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 5 pixels greater radius on hover
- * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 5 pixels greater radius on hover
- *
- * @since 4.0.3
- */
- radiusPlus: 2,
- /**
- * The additional line width for a hovered point.
- *
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 2 pixels wider on hover
- * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 2 pixels wider on hover
- *
- * @since 4.0.3
- */
- lineWidthPlus: 1
- },
- /**
- * The appearance of the point marker when selected. In order to
- * allow a point to be selected, set the
- * `series.allowPointSelect` option to true.
- *
- * @declare Highcharts.PointStatesSelectOptionsObject
- */
- select: {
- /**
- * Enable or disable visible feedback for selection.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-enabled/
- * Disabled select state
- *
- * @type {boolean}
- * @default true
- * @apioption plotOptions.series.marker.states.select.enabled
- */
- /**
- * The radius of the point marker. In hover state, it
- * defaults to the normal state's radius + 2.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-radius/
- * 10px radius for selected points
- *
- * @type {number}
- * @apioption plotOptions.series.marker.states.select.radius
- */
- /**
- * The fill color of the point marker.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-fillcolor/
- * Solid red discs for selected points
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- fillColor: '#cccccc',
- /**
- * The color of the point marker's outline. When
- * `undefined`, the series' or point's color is used.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linecolor/
- * Red line color for selected points
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- lineColor: '#000000',
- /**
- * The width of the point marker's outline.
- *
- * @sample {highcharts} highcharts/plotoptions/series-marker-states-select-linewidth/
- * 3px line width for selected points
- */
- lineWidth: 2
- }
- }
- },
- /**
- * Properties for each single point.
- *
- * @declare Highcharts.PlotSeriesPointOptions
- *
- * @private
- */
- point: {
- /**
- * Fires when a point is clicked. One parameter, `event`, is passed
- * to the function, containing common event information.
- *
- * If the `series.allowPointSelect` option is true, the default
- * action for the point's click event is to toggle the point's
- * select state. Returning `false` cancels this action.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-click/
- * Click marker to alert values
- * @sample {highcharts} highcharts/plotoptions/series-point-events-click-column/
- * Click column
- * @sample {highcharts} highcharts/plotoptions/series-point-events-click-url/
- * Go to URL
- * @sample {highmaps} maps/plotoptions/series-point-events-click/
- * Click marker to display values
- * @sample {highmaps} maps/plotoptions/series-point-events-click-url/
- * Go to URL
- *
- * @type {Highcharts.PointClickCallbackFunction}
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.click
- */
- /**
- * Fires when the mouse leaves the area close to the point. One
- * parameter, `event`, is passed to the function, containing common
- * event information.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/
- * Show values in the chart's corner on mouse over
- *
- * @type {Highcharts.PointMouseOutCallbackFunction}
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.mouseOut
- */
- /**
- * Fires when the mouse enters the area close to the point. One
- * parameter, `event`, is passed to the function, containing common
- * event information.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-mouseover/
- * Show values in the chart's corner on mouse over
- *
- * @type {Highcharts.PointMouseOverCallbackFunction}
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.mouseOver
- */
- /**
- * Fires when the point is removed using the `.remove()` method. One
- * parameter, `event`, is passed to the function. Returning `false`
- * cancels the operation.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-remove/
- * Remove point and confirm
- *
- * @type {Highcharts.PointRemoveCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.remove
- */
- /**
- * Fires when the point is selected either programmatically or
- * following a click on the point. One parameter, `event`, is passed
- * to the function. Returning `false` cancels the operation.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-select/
- * Report the last selected point
- * @sample {highmaps} maps/plotoptions/series-allowpointselect/
- * Report select and unselect
- *
- * @type {Highcharts.PointSelectCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.select
- */
- /**
- * Fires when the point is unselected either programmatically or
- * following a click on the point. One parameter, `event`, is passed
- * to the function.
- * Returning `false` cancels the operation.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-unselect/
- * Report the last unselected point
- * @sample {highmaps} maps/plotoptions/series-allowpointselect/
- * Report select and unselect
- *
- * @type {Highcharts.PointUnselectCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.unselect
- */
- /**
- * Fires when the point is updated programmatically through the
- * `.update()` method. One parameter, `event`, is passed to the
- * function. The new point options can be accessed through
- * `event.options`. Returning `false` cancels the operation.
- *
- * @sample {highcharts} highcharts/plotoptions/series-point-events-update/
- * Confirm point updating
- *
- * @type {Highcharts.PointUpdateCallbackFunction}
- * @since 1.2.0
- * @context Highcharts.Point
- * @apioption plotOptions.series.point.events.update
- */
- /**
- * Events for each single point.
- *
- * @declare Highcharts.PointEventsOptionsObject
- */
- events: {}
- },
- /**
- * Options for the series data labels, appearing next to each data
- * point.
- *
- * Since v6.2.0, multiple data labels can be applied to each single
- * point by defining them as an array of configs.
- *
- * In styled mode, the data labels can be styled with the
- * `.highcharts-data-label-box` and `.highcharts-data-label` class names
- * ([see example](https://www.highcharts.com/samples/highcharts/css/series-datalabels)).
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled
- * Data labels enabled
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-multiple
- * Multiple data labels on a bar series
- * @sample {highcharts} highcharts/css/series-datalabels
- * Style mode example
- *
- * @type {*|Array<*>}
- * @product highcharts highstock highmaps gantt
- *
- * @private
- */
- dataLabels: {
- /**
- * Enable or disable the initial animation when a series is
- * displayed for the `dataLabels`. The animation can also be set as
- * a configuration object. Please note that this option only
- * applies to the initial animation.
- * For other animations, see [chart.animation](#chart.animation)
- * and the animation parameter under the API methods.
- * The following properties are supported:
- *
- * - `defer`: The animation delay time in milliseconds.
- *
- * @sample {highcharts} highcharts/plotoptions/animation-defer/
- * Animation defer settings
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @since 8.2.0
- * @apioption plotOptions.series.dataLabels.animation
- */
- animation: {},
- /**
- * The animation delay time in milliseconds.
- * Set to `0` renders dataLabel immediately.
- * As `undefined` inherits defer time from the [series.animation.defer](#plotOptions.series.animation.defer).
- *
- * @type {number}
- * @since 8.2.0
- * @apioption plotOptions.series.dataLabels.animation.defer
- */
- /**
- * The alignment of the data label compared to the point. If
- * `right`, the right side of the label should be touching the
- * point. For points with an extent, like columns, the alignments
- * also dictates how to align it inside the box, as given with the
- * [inside](#plotOptions.column.dataLabels.inside)
- * option. Can be one of `left`, `center` or `right`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-align-left/
- * Left aligned
- * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/
- * Data labels inside the bar
- *
- * @type {Highcharts.AlignValue|null}
- */
- align: 'center',
- /**
- * Whether to allow data labels to overlap. To make the labels less
- * sensitive for overlapping, the
- * [dataLabels.padding](#plotOptions.series.dataLabels.padding)
- * can be set to 0.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-allowoverlap-false/
- * Don't allow overlap
- *
- * @type {boolean}
- * @default false
- * @since 4.1.0
- * @apioption plotOptions.series.dataLabels.allowOverlap
- */
- /**
- * The background color or gradient for the data label.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- * @sample {highmaps} maps/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 2.2.1
- * @apioption plotOptions.series.dataLabels.backgroundColor
- */
- /**
- * The border color for the data label. Defaults to `undefined`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 2.2.1
- * @apioption plotOptions.series.dataLabels.borderColor
- */
- /**
- * The border radius in pixels for the data label.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- * @sample {highmaps} maps/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @type {number}
- * @default 0
- * @since 2.2.1
- * @apioption plotOptions.series.dataLabels.borderRadius
- */
- /**
- * The border width in pixels for the data label.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @type {number}
- * @default 0
- * @since 2.2.1
- * @apioption plotOptions.series.dataLabels.borderWidth
- */
- /**
- * A class name for the data label. Particularly in styled mode,
- * this can be used to give each series' or point's data label
- * unique styling. In addition to this option, a default color class
- * name is added so that we can give the labels a contrast text
- * shadow.
- *
- * @sample {highcharts} highcharts/css/data-label-contrast/
- * Contrast text shadow
- * @sample {highcharts} highcharts/css/series-datalabels/
- * Styling by CSS
- *
- * @type {string}
- * @since 5.0.0
- * @apioption plotOptions.series.dataLabels.className
- */
- /**
- * The text color for the data labels. Defaults to `undefined`. For
- * certain series types, like column or map, the data labels can be
- * drawn inside the points. In this case the data label will be
- * drawn with maximum contrast by default. Additionally, it will be
- * given a `text-outline` style with the opposite color, to further
- * increase the contrast. This can be overridden by setting the
- * `text-outline` style to `none` in the `dataLabels.style` option.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-color/
- * Red data labels
- * @sample {highmaps} maps/demo/color-axis/
- * White data labels
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption plotOptions.series.dataLabels.color
- */
- /**
- * Whether to hide data labels that are outside the plot area. By
- * default, the data label is moved inside the plot area according
- * to the
- * [overflow](#plotOptions.series.dataLabels.overflow)
- * option.
- *
- * @type {boolean}
- * @default true
- * @since 2.3.3
- * @apioption plotOptions.series.dataLabels.crop
- */
- /**
- * Whether to defer displaying the data labels until the initial
- * series animation has finished. Setting to `false` renders the
- * data label immediately. If set to `true` inherits the defer
- * time set in [plotOptions.series.animation](#plotOptions.series.animation).
- *
- * @sample highcharts/plotoptions/animation-defer
- * Set defer time
- *
- * @since 4.0.0
- * @product highcharts highstock gantt
- */
- defer: true,
- /**
- * Enable or disable the data labels.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-enabled/
- * Data labels enabled
- * @sample {highmaps} maps/demo/color-axis/
- * Data labels enabled
- *
- * @type {boolean}
- * @default false
- * @apioption plotOptions.series.dataLabels.enabled
- */
- /**
- * A declarative filter to control of which data labels to display.
- * The declarative filter is designed for use when callback
- * functions are not available, like when the chart options require
- * a pure JSON structure or for use with graphical editors. For
- * programmatic control, use the `formatter` instead, and return
- * `undefined` to disable a single data label.
- *
- * @example
- * filter: {
- * property: 'percentage',
- * operator: '>',
- * value: 4
- * }
- *
- * @sample {highcharts} highcharts/demo/pie-monochrome
- * Data labels filtered by percentage
- *
- * @declare Highcharts.DataLabelsFilterOptionsObject
- * @since 6.0.3
- * @apioption plotOptions.series.dataLabels.filter
- */
- /**
- * The operator to compare by. Can be one of `>`, `<`, `>=`, `<=`,
- * `==`, and `===`.
- *
- * @type {string}
- * @validvalue [">", "<", ">=", "<=", "==", "==="]
- * @apioption plotOptions.series.dataLabels.filter.operator
- */
- /**
- * The point property to filter by. Point options are passed
- * directly to properties, additionally there are `y` value,
- * `percentage` and others listed under {@link Highcharts.Point}
- * members.
- *
- * @type {string}
- * @apioption plotOptions.series.dataLabels.filter.property
- */
- /**
- * The value to compare against.
- *
- * @type {number}
- * @apioption plotOptions.series.dataLabels.filter.value
- */
- /**
- * A
- * [format string](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting)
- * for the data label. Available variables are the same as for
- * `formatter`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/
- * Add a unit
- * @sample {highmaps} maps/plotoptions/series-datalabels-format/
- * Formatted value in the data label
- *
- * @type {string}
- * @default y
- * @default point.value
- * @since 3.0
- * @apioption plotOptions.series.dataLabels.format
- */
- // eslint-disable-next-line valid-jsdoc
- /**
- * Callback JavaScript function to format the data label. Note that
- * if a `format` is defined, the format takes precedence and the
- * formatter is ignored.
- *
- * @sample {highmaps} maps/plotoptions/series-datalabels-format/
- * Formatted value
- *
- * @type {Highcharts.DataLabelsFormatterCallbackFunction}
- */
- formatter: function () {
- var numberFormatter = this.series.chart.numberFormatter;
- return typeof this.y !== 'number' ? '' : numberFormatter(this.y, -1);
- },
- /**
- * For points with an extent, like columns or map areas, whether to
- * align the data label inside the box or to the actual value point.
- * Defaults to `false` in most cases, `true` in stacked columns.
- *
- * @type {boolean}
- * @since 3.0
- * @apioption plotOptions.series.dataLabels.inside
- */
- /**
- * Format for points with the value of null. Works analogously to
- * [format](#plotOptions.series.dataLabels.format). `nullFormat` can
- * be applied only to series which support displaying null points.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/
- * Format data label and tooltip for null point.
- *
- * @type {boolean|string}
- * @since 7.1.0
- * @apioption plotOptions.series.dataLabels.nullFormat
- */
- /**
- * Callback JavaScript function that defines formatting for points
- * with the value of null. Works analogously to
- * [formatter](#plotOptions.series.dataLabels.formatter).
- * `nullPointFormatter` can be applied only to series which support
- * displaying null points.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-format/
- * Format data label and tooltip for null point.
- *
- * @type {Highcharts.DataLabelsFormatterCallbackFunction}
- * @since 7.1.0
- * @apioption plotOptions.series.dataLabels.nullFormatter
- */
- /**
- * How to handle data labels that flow outside the plot area. The
- * default is `"justify"`, which aligns them inside the plot area.
- * For columns and bars, this means it will be moved inside the bar.
- * To display data labels outside the plot area, set `crop` to
- * `false` and `overflow` to `"allow"`.
- *
- * @type {Highcharts.DataLabelsOverflowValue}
- * @default justify
- * @since 3.0.6
- * @apioption plotOptions.series.dataLabels.overflow
- */
- /**
- * When either the `borderWidth` or the `backgroundColor` is set,
- * this is the padding within the box.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- * @sample {highmaps} maps/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @since 2.2.1
- */
- padding: 5,
- /**
- * Aligns data labels relative to points. If `center` alignment is
- * not possible, it defaults to `right`.
- *
- * @type {Highcharts.AlignValue}
- * @default center
- * @apioption plotOptions.series.dataLabels.position
- */
- /**
- * Text rotation in degrees. Note that due to a more complex
- * structure, backgrounds, borders and padding will be lost on a
- * rotated data label.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/
- * Vertical labels
- *
- * @type {number}
- * @default 0
- * @apioption plotOptions.series.dataLabels.rotation
- */
- /**
- * The shadow of the box. Works best with `borderWidth` or
- * `backgroundColor`. Since 2.3 the shadow can be an object
- * configuration containing `color`, `offsetX`, `offsetY`, `opacity`
- * and `width`.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-box/
- * Data labels box options
- *
- * @type {boolean|Highcharts.ShadowOptionsObject}
- * @default false
- * @since 2.2.1
- * @apioption plotOptions.series.dataLabels.shadow
- */
- /**
- * The name of a symbol to use for the border around the label.
- * Symbols are predefined functions on the Renderer object.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-shape/
- * A callout for annotations
- *
- * @type {string}
- * @default square
- * @since 4.1.2
- * @apioption plotOptions.series.dataLabels.shape
- */
- /**
- * Styles for the label. The default `color` setting is
- * `"contrast"`, which is a pseudo color that Highcharts picks up
- * and applies the maximum contrast to the underlying point item,
- * for example the bar in a bar chart.
- *
- * The `textOutline` is a pseudo property that applies an outline of
- * the given width with the given color, which by default is the
- * maximum contrast to the text. So a bright text color will result
- * in a black text outline for maximum readability on a mixed
- * background. In some cases, especially with grayscale text, the
- * text outline doesn't work well, in which cases it can be disabled
- * by setting it to `"none"`. When `useHTML` is true, the
- * `textOutline` will not be picked up. In this, case, the same
- * effect can be acheived through the `text-shadow` CSS property.
- *
- * For some series types, where each point has an extent, like for
- * example tree maps, the data label may overflow the point. There
- * are two strategies for handling overflow. By default, the text
- * will wrap to multiple lines. The other strategy is to set
- * `style.textOverflow` to `ellipsis`, which will keep the text on
- * one line plus it will break inside long words.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-style/
- * Bold labels
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow/
- * Long labels truncated with an ellipsis in a pie
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap/
- * Long labels are wrapped in a pie
- * @sample {highmaps} maps/demo/color-axis/
- * Bold labels
- *
- * @type {Highcharts.CSSObject}
- * @since 4.1.0
- * @apioption plotOptions.series.dataLabels.style
- */
- style: {
- /** @internal */
- fontSize: '11px',
- /** @internal */
- fontWeight: 'bold',
- /** @internal */
- color: 'contrast',
- /** @internal */
- textOutline: '1px contrast'
- },
- /**
- * Options for a label text which should follow marker's shape.
- * Border and background are disabled for a label that follows a
- * path.
- *
- * **Note:** Only SVG-based renderer supports this option. Setting
- * `useHTML` to true will disable this option.
- *
- * @declare Highcharts.DataLabelsTextPathOptionsObject
- * @since 7.1.0
- * @apioption plotOptions.series.dataLabels.textPath
- */
- /**
- * Presentation attributes for the text path.
- *
- * @type {Highcharts.SVGAttributes}
- * @since 7.1.0
- * @apioption plotOptions.series.dataLabels.textPath.attributes
- */
- /**
- * Enable or disable `textPath` option for link's or marker's data
- * labels.
- *
- * @type {boolean}
- * @since 7.1.0
- * @apioption plotOptions.series.dataLabels.textPath.enabled
- */
- /**
- * Whether to
- * [use HTML](https://www.highcharts.com/docs/chart-concepts/labels-and-string-formatting#html)
- * to render the labels.
- *
- * @type {boolean}
- * @default false
- * @apioption plotOptions.series.dataLabels.useHTML
- */
- /**
- * The vertical alignment of a data label. Can be one of `top`,
- * `middle` or `bottom`. The default value depends on the data, for
- * instance in a column chart, the label is above positive values
- * and below negative values.
- *
- * @type {Highcharts.VerticalAlignValue|null}
- * @since 2.3.3
- */
- verticalAlign: 'bottom',
- /**
- * The x position offset of the label relative to the point in
- * pixels.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/
- * Vertical and positioned
- * @sample {highcharts} highcharts/plotoptions/bar-datalabels-align-inside-bar/
- * Data labels inside the bar
- */
- x: 0,
- /**
- * The Z index of the data labels. The default Z index puts it above
- * the series. Use a Z index of 2 to display it behind the series.
- *
- * @type {number}
- * @default 6
- * @since 2.3.5
- * @apioption plotOptions.series.dataLabels.z
- */
- /**
- * The y position offset of the label relative to the point in
- * pixels.
- *
- * @sample {highcharts} highcharts/plotoptions/series-datalabels-rotation/
- * Vertical and positioned
- */
- y: 0
- },
- /**
- * When the series contains less points than the crop threshold, all
- * points are drawn, even if the points fall outside the visible plot
- * area at the current zoom. The advantage of drawing all points
- * (including markers and columns), is that animation is performed on
- * updates. On the other hand, when the series contains more points than
- * the crop threshold, the series data is cropped to only contain points
- * that fall within the plot area. The advantage of cropping away
- * invisible points is to increase performance on large series.
- *
- * @since 2.2
- * @product highcharts highstock
- *
- * @private
- */
- cropThreshold: 300,
- /**
- * Opacity of a series parts: line, fill (e.g. area) and dataLabels.
- *
- * @see [states.inactive.opacity](#plotOptions.series.states.inactive.opacity)
- *
- * @since 7.1.0
- *
- * @private
- */
- opacity: 1,
- /**
- * The width of each point on the x axis. For example in a column chart
- * with one value each day, the pointRange would be 1 day (= 24 * 3600
- * * 1000 milliseconds). This is normally computed automatically, but
- * this option can be used to override the automatic value.
- *
- * @product highstock
- *
- * @private
- */
- pointRange: 0,
- /**
- * When this is true, the series will not cause the Y axis to cross
- * the zero plane (or [threshold](#plotOptions.series.threshold) option)
- * unless the data actually crosses the plane.
- *
- * For example, if `softThreshold` is `false`, a series of 0, 1, 2,
- * 3 will make the Y axis show negative values according to the
- * `minPadding` option. If `softThreshold` is `true`, the Y axis starts
- * at 0.
- *
- * @since 4.1.9
- * @product highcharts highstock
- *
- * @private
- */
- softThreshold: true,
- /**
- * @declare Highcharts.SeriesStatesOptionsObject
- *
- * @private
- */
- states: {
- /**
- * The normal state of a series, or for point items in column, pie
- * and similar series. Currently only used for setting animation
- * when returning to normal state from hover.
- *
- * @declare Highcharts.SeriesStatesNormalOptionsObject
- */
- normal: {
- /**
- * Animation when returning to normal state after hovering.
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- */
- animation: true
- },
- /**
- * Options for the hovered series. These settings override the
- * normal state options when a series is moused over or touched.
- *
- * @declare Highcharts.SeriesStatesHoverOptionsObject
- */
- hover: {
- /**
- * Enable separate styles for the hovered series to visualize
- * that the user hovers either the series itself or the legend.
- *
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled/
- * Line
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-column/
- * Column
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-enabled-pie/
- * Pie
- *
- * @type {boolean}
- * @default true
- * @since 1.2
- * @apioption plotOptions.series.states.hover.enabled
- */
- /**
- * Animation setting for hovering the graph in line-type series.
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * @since 5.0.8
- * @product highcharts highstock
- */
- animation: {
- /**
- * The duration of the hover animation in milliseconds. By
- * default the hover state animates quickly in, and slowly
- * back to normal.
- *
- * @internal
- */
- duration: 50
- },
- /**
- * Pixel width of the graph line. By default this property is
- * undefined, and the `lineWidthPlus` property dictates how much
- * to increase the linewidth from normal state.
- *
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidth/
- * 5px line on hover
- *
- * @type {number}
- * @product highcharts highstock
- * @apioption plotOptions.series.states.hover.lineWidth
- */
- /**
- * The additional line width for the graph of a hovered series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 5 pixels wider
- * @sample {highstock} highcharts/plotoptions/series-states-hover-linewidthplus/
- * 5 pixels wider
- *
- * @since 4.0.3
- * @product highcharts highstock
- */
- lineWidthPlus: 1,
- /**
- * In Highcharts 1.0, the appearance of all markers belonging
- * to the hovered series. For settings on the hover state of the
- * individual point, see
- * [marker.states.hover](#plotOptions.series.marker.states.hover).
- *
- * @deprecated
- *
- * @extends plotOptions.series.marker
- * @excluding states
- * @product highcharts highstock
- */
- marker: {
- // lineWidth: base + 1,
- // radius: base + 1
- },
- /**
- * Options for the halo appearing around the hovered point in
- * line-type series as well as outside the hovered slice in pie
- * charts. By default the halo is filled by the current point or
- * series color with an opacity of 0.25\. The halo can be
- * disabled by setting the `halo` option to `null`.
- *
- * In styled mode, the halo is styled with the
- * `.highcharts-halo` class, with colors inherited from
- * `.highcharts-color-{n}`.
- *
- * @sample {highcharts} highcharts/plotoptions/halo/
- * Halo options
- * @sample {highstock} highcharts/plotoptions/halo/
- * Halo options
- *
- * @declare Highcharts.SeriesStatesHoverHaloOptionsObject
- * @type {null|*}
- * @since 4.0
- * @product highcharts highstock
- */
- halo: {
- /**
- * A collection of SVG attributes to override the appearance
- * of the halo, for example `fill`, `stroke` and
- * `stroke-width`.
- *
- * @type {Highcharts.SVGAttributes}
- * @since 4.0
- * @product highcharts highstock
- * @apioption plotOptions.series.states.hover.halo.attributes
- */
- /**
- * The pixel size of the halo. For point markers this is the
- * radius of the halo. For pie slices it is the width of the
- * halo outside the slice. For bubbles it defaults to 5 and
- * is the width of the halo outside the bubble.
- *
- * @since 4.0
- * @product highcharts highstock
- */
- size: 10,
- /**
- * Opacity for the halo unless a specific fill is overridden
- * using the `attributes` setting. Note that Highcharts is
- * only able to apply opacity to colors of hex or rgb(a)
- * formats.
- *
- * @since 4.0
- * @product highcharts highstock
- */
- opacity: 0.25
- }
- },
- /**
- * Specific options for point in selected states, after being
- * selected by
- * [allowPointSelect](#plotOptions.series.allowPointSelect)
- * or programmatically.
- *
- * @sample maps/plotoptions/series-allowpointselect/
- * Allow point select demo
- *
- * @declare Highcharts.SeriesStatesSelectOptionsObject
- * @extends plotOptions.series.states.hover
- * @excluding brightness
- */
- select: {
- animation: {
- /** @internal */
- duration: 0
- }
- },
- /**
- * The opposite state of a hover for series.
- *
- * @sample highcharts/plotoptions/series-states-inactive-disabled
- * Disabled inactive state
- *
- * @declare Highcharts.SeriesStatesInactiveOptionsObject
- */
- inactive: {
- /**
- * Enable or disable the inactive state for a series
- *
- * @sample highcharts/plotoptions/series-states-inactive-disabled
- * Disabled inactive state
- *
- * @type {boolean}
- * @default true
- * @apioption plotOptions.series.states.inactive.enabled
- */
- /**
- * The animation for entering the inactive state.
- *
- * @type {boolean|Partial<Highcharts.AnimationOptionsObject>}
- */
- animation: {
- /** @internal */
- duration: 50
- },
- /**
- * Opacity of series elements (dataLabels, line, area).
- *
- * @type {number}
- */
- opacity: 0.2
- }
- },
- /**
- * Sticky tracking of mouse events. When true, the `mouseOut` event on a
- * series isn't triggered until the mouse moves over another series, or
- * out of the plot area. When false, the `mouseOut` event on a series is
- * triggered when the mouse leaves the area around the series' graph or
- * markers. This also implies the tooltip when not shared. When
- * `stickyTracking` is false and `tooltip.shared` is false, the tooltip
- * will be hidden when moving the mouse between series. Defaults to true
- * for line and area type series, but to false for columns, pies etc.
- *
- * **Note:** The boost module will force this option because of
- * technical limitations.
- *
- * @sample {highcharts} highcharts/plotoptions/series-stickytracking-true/
- * True by default
- * @sample {highcharts} highcharts/plotoptions/series-stickytracking-false/
- * False
- *
- * @default {highcharts} true
- * @default {highstock} true
- * @default {highmaps} false
- * @since 2.0
- *
- * @private
- */
- stickyTracking: true,
- /**
- * A configuration object for the tooltip rendering of each single
- * series. Properties are inherited from [tooltip](#tooltip), but only
- * the following properties can be defined on a series level.
- *
- * @declare Highcharts.SeriesTooltipOptionsObject
- * @since 2.3
- * @extends tooltip
- * @excluding animation, backgroundColor, borderColor, borderRadius,
- * borderWidth, className, crosshairs, enabled, formatter,
- * headerShape, hideDelay, outside, padding, positioner,
- * shadow, shape, shared, snap, split, style, useHTML
- * @apioption plotOptions.series.tooltip
- */
- /**
- * When a series contains a data array that is longer than this, only
- * one dimensional arrays of numbers, or two dimensional arrays with
- * x and y values are allowed. Also, only the first point is tested,
- * and the rest are assumed to be the same format. This saves expensive
- * data checking and indexing in long series. Set it to `0` disable.
- *
- * Note:
- * In boost mode turbo threshold is forced. Only array of numbers or
- * two dimensional arrays are allowed.
- *
- * @since 2.2
- * @product highcharts highstock gantt
- *
- * @private
- */
- turboThreshold: 1000,
- /**
- * An array defining zones within a series. Zones can be applied to the
- * X axis, Y axis or Z axis for bubbles, according to the `zoneAxis`
- * option. The zone definitions have to be in ascending order regarding
- * to the value.
- *
- * In styled mode, the color zones are styled with the
- * `.highcharts-zone-{n}` class, or custom classed from the `className`
- * option
- * ([view live demo](https://jsfiddle.net/gh/get/library/pure/highcharts/highcharts/tree/master/samples/highcharts/css/color-zones/)).
- *
- * @see [zoneAxis](#plotOptions.series.zoneAxis)
- *
- * @sample {highcharts} highcharts/series/color-zones-simple/
- * Color zones
- * @sample {highstock} highcharts/series/color-zones-simple/
- * Color zones
- *
- * @declare Highcharts.SeriesZonesOptionsObject
- * @type {Array<*>}
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zones
- */
- /**
- * Styled mode only. A custom class name for the zone.
- *
- * @sample highcharts/css/color-zones/
- * Zones styled by class name
- *
- * @type {string}
- * @since 5.0.0
- * @apioption plotOptions.series.zones.className
- */
- /**
- * Defines the color of the series.
- *
- * @see [series color](#plotOptions.series.color)
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zones.color
- */
- /**
- * A name for the dash style to use for the graph.
- *
- * @see [plotOptions.series.dashStyle](#plotOptions.series.dashStyle)
- *
- * @sample {highcharts|highstock} highcharts/series/color-zones-dashstyle-dot/
- * Dashed line indicates prognosis
- *
- * @type {Highcharts.DashStyleValue}
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zones.dashStyle
- */
- /**
- * Defines the fill color for the series (in area type series)
- *
- * @see [fillColor](#plotOptions.area.fillColor)
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zones.fillColor
- */
- /**
- * The value up to where the zone extends, if undefined the zones
- * stretches to the last value in the series.
- *
- * @type {number}
- * @since 4.1.0
- * @product highcharts highstock
- * @apioption plotOptions.series.zones.value
- */
- /**
- * When using dual or multiple color axes, this number defines which
- * colorAxis the particular series is connected to. It refers to
- * either the
- * {@link #colorAxis.id|axis id}
- * or the index of the axis in the colorAxis array, with 0 being the
- * first. Set this option to false to prevent a series from connecting
- * to the default color axis.
- *
- * Since v7.2.0 the option can also be an axis id or an axis index
- * instead of a boolean flag.
- *
- * @sample highcharts/coloraxis/coloraxis-with-pie/
- * Color axis with pie series
- * @sample highcharts/coloraxis/multiple-coloraxis/
- * Multiple color axis
- *
- * @type {number|string|boolean}
- * @default 0
- * @product highcharts highstock highmaps
- * @apioption plotOptions.series.colorAxis
- */
- /**
- * Determines what data value should be used to calculate point color
- * if `colorAxis` is used. Requires to set `min` and `max` if some
- * custom point property is used or if approximation for data grouping
- * is set to `'sum'`.
- *
- * @sample highcharts/coloraxis/custom-color-key/
- * Custom color key
- * @sample highcharts/coloraxis/changed-default-color-key/
- * Changed default color key
- *
- * @type {string}
- * @default y
- * @since 7.2.0
- * @product highcharts highstock highmaps
- * @apioption plotOptions.series.colorKey
- */
- /**
- * Determines whether the series should look for the nearest point
- * in both dimensions or just the x-dimension when hovering the series.
- * Defaults to `'xy'` for scatter series and `'x'` for most other
- * series. If the data has duplicate x-values, it is recommended to
- * set this to `'xy'` to allow hovering over all points.
- *
- * Applies only to series types using nearest neighbor search (not
- * direct hover) for tooltip.
- *
- * @sample {highcharts} highcharts/series/findnearestpointby/
- * Different hover behaviors
- * @sample {highstock} highcharts/series/findnearestpointby/
- * Different hover behaviors
- * @sample {highmaps} highcharts/series/findnearestpointby/
- * Different hover behaviors
- *
- * @since 5.0.10
- * @validvalue ["x", "xy"]
- *
- * @private
- */
- findNearestPointBy: 'x'
- },
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /** @lends Highcharts.Series.prototype */
- {
- axisTypes: ['xAxis', 'yAxis'],
- coll: 'series',
- colorCounter: 0,
- cropShoulder: 1,
- directTouch: false,
- isCartesian: true,
- // each point's x and y values are stored in this.xData and this.yData
- parallelArrays: ['x', 'y'],
- pointClass: Point,
- requireSorting: true,
- sorted: true,
- init: function (chart, options) {
- fireEvent(this, 'init', { options: options });
- var series = this,
- events,
- chartSeries = chart.series,
- lastSeries;
- // A lookup over those events that are added by _options_ (not
- // programmatically). These are updated through Series.update()
- // (#10861).
- this.eventOptions = this.eventOptions || {};
- // The 'eventsToUnbind' property moved from prototype into the
- // Series init to avoid reference to the same array between
- // the different series and charts. #12959, #13937
- this.eventsToUnbind = [];
- /**
- * Read only. The chart that the series belongs to.
- *
- * @name Highcharts.Series#chart
- * @type {Highcharts.Chart}
- */
- series.chart = chart;
- /**
- * Read only. The series' type, like "line", "area", "column" etc.
- * The type in the series options anc can be altered using
- * {@link Series#update}.
- *
- * @name Highcharts.Series#type
- * @type {string}
- */
- /**
- * Read only. The series' current options. To update, use
- * {@link Series#update}.
- *
- * @name Highcharts.Series#options
- * @type {Highcharts.SeriesOptionsType}
- */
- series.options = options = series.setOptions(options);
- series.linkedSeries = [];
- // bind the axes
- series.bindAxes();
- // set some variables
- extend(series, {
- /**
- * The series name as given in the options. Defaults to
- * "Series {n}".
- *
- * @name Highcharts.Series#name
- * @type {string}
- */
- name: options.name,
- state: '',
- /**
- * Read only. The series' visibility state as set by {@link
- * Series#show}, {@link Series#hide}, or in the initial
- * configuration.
- *
- * @name Highcharts.Series#visible
- * @type {boolean}
- */
- visible: options.visible !== false,
- /**
- * Read only. The series' selected state as set by {@link
- * Highcharts.Series#select}.
- *
- * @name Highcharts.Series#selected
- * @type {boolean}
- */
- selected: options.selected === true // false by default
- });
- // Register event listeners
- events = options.events;
- objectEach(events, function (event, eventType) {
- if (isFunction(event)) {
- // If event does not exist, or is changed by Series.update
- if (series.eventOptions[eventType] !== event) {
- // Remove existing if set by option
- if (isFunction(series.eventOptions[eventType])) {
- removeEvent(series, eventType, series.eventOptions[eventType]);
- }
- series.eventOptions[eventType] = event;
- addEvent(series, eventType, event);
- }
- }
- });
- if ((events && events.click) ||
- (options.point &&
- options.point.events &&
- options.point.events.click) ||
- options.allowPointSelect) {
- chart.runTrackerClick = true;
- }
- series.getColor();
- series.getSymbol();
- // Initialize the parallel data arrays
- series.parallelArrays.forEach(function (key) {
- if (!series[key + 'Data']) {
- series[key + 'Data'] = [];
- }
- });
- // Mark cartesian
- if (series.isCartesian) {
- chart.hasCartesianSeries = true;
- }
- // Get the index and register the series in the chart. The index is
- // one more than the current latest series index (#5960).
- if (chartSeries.length) {
- lastSeries = chartSeries[chartSeries.length - 1];
- }
- series._i = pick(lastSeries && lastSeries._i, -1) + 1;
- series.opacity = series.options.opacity;
- // Insert the series and re-order all series above the insertion
- // point.
- chart.orderSeries(this.insert(chartSeries));
- // Set options for series with sorting and set data later.
- if (options.dataSorting && options.dataSorting.enabled) {
- series.setDataSortingOptions();
- }
- else if (!series.points && !series.data) {
- series.setData(options.data, false);
- }
- fireEvent(this, 'afterInit');
- },
- /**
- * Check whether the series item is itself or inherits from a certain
- * series type.
- *
- * @function Highcharts.Series#is
- * @param {string} type The type of series to check for, can be either
- * featured or custom series types. For example `column`, `pie`,
- * `ohlc` etc.
- *
- * @return {boolean}
- * True if this item is or inherits from the given type.
- */
- is: function (type) {
- return seriesTypes[type] && this instanceof seriesTypes[type];
- },
- /**
- * Insert the series in a collection with other series, either the chart
- * series or yAxis series, in the correct order according to the index
- * option. Used internally when adding series.
- *
- * @private
- * @function Highcharts.Series#insert
- * @param {Array<Highcharts.Series>} collection
- * A collection of series, like `chart.series` or `xAxis.series`.
- * @return {number}
- * The index of the series in the collection.
- */
- insert: function (collection) {
- var indexOption = this.options.index,
- i;
- // Insert by index option
- if (isNumber(indexOption)) {
- i = collection.length;
- while (i--) {
- // Loop down until the interted element has higher index
- if (indexOption >=
- pick(collection[i].options.index, collection[i]._i)) {
- collection.splice(i + 1, 0, this);
- break;
- }
- }
- if (i === -1) {
- collection.unshift(this);
- }
- i = i + 1;
- // Or just push it to the end
- }
- else {
- collection.push(this);
- }
- return pick(i, collection.length - 1);
- },
- /**
- * Set the xAxis and yAxis properties of cartesian series, and register
- * the series in the `axis.series` array.
- *
- * @private
- * @function Highcharts.Series#bindAxes
- * @return {void}
- * @exception 18
- */
- bindAxes: function () {
- var series = this,
- seriesOptions = series.options,
- chart = series.chart,
- axisOptions;
- fireEvent(this, 'bindAxes', null, function () {
- // repeat for xAxis and yAxis
- (series.axisTypes || []).forEach(function (AXIS) {
- // loop through the chart's axis objects
- chart[AXIS].forEach(function (axis) {
- axisOptions = axis.options;
- // apply if the series xAxis or yAxis option mathches
- // the number of the axis, or if undefined, use the
- // first axis
- if (seriesOptions[AXIS] ===
- axisOptions.index ||
- (typeof seriesOptions[AXIS] !==
- 'undefined' &&
- seriesOptions[AXIS] === axisOptions.id) ||
- (typeof seriesOptions[AXIS] ===
- 'undefined' &&
- axisOptions.index === 0)) {
- // register this series in the axis.series lookup
- series.insert(axis.series);
- // set this series.xAxis or series.yAxis reference
- /**
- * Read only. The unique xAxis object associated
- * with the series.
- *
- * @name Highcharts.Series#xAxis
- * @type {Highcharts.Axis}
- */
- /**
- * Read only. The unique yAxis object associated
- * with the series.
- *
- * @name Highcharts.Series#yAxis
- * @type {Highcharts.Axis}
- */
- series[AXIS] = axis;
- // mark dirty for redraw
- axis.isDirty = true;
- }
- });
- // The series needs an X and an Y axis
- if (!series[AXIS] &&
- series.optionalAxis !== AXIS) {
- error(18, true, chart);
- }
- });
- });
- fireEvent(this, 'afterBindAxes');
- },
- /**
- * For simple series types like line and column, the data values are
- * held in arrays like xData and yData for quick lookup to find extremes
- * and more. For multidimensional series like bubble and map, this can
- * be extended with arrays like zData and valueData by adding to the
- * `series.parallelArrays` array.
- *
- * @private
- * @function Highcharts.Series#updateParallelArrays
- * @param {Highcharts.Point} point
- * @param {number|string} i
- * @return {void}
- */
- updateParallelArrays: function (point, i) {
- var series = point.series,
- args = arguments,
- fn = isNumber(i) ?
- // Insert the value in the given position
- function (key) {
- var val = key === 'y' && series.toYData ?
- series.toYData(point) :
- point[key];
- series[key + 'Data'][i] = val;
- } :
- // Apply the method specified in i with the following
- // arguments as arguments
- function (key) {
- Array.prototype[i].apply(series[key + 'Data'], Array.prototype.slice.call(args, 2));
- };
- series.parallelArrays.forEach(fn);
- },
- /**
- * Define hasData functions for series. These return true if there
- * are data points on this series within the plot area.
- *
- * @private
- * @function Highcharts.Series#hasData
- * @return {boolean}
- */
- hasData: function () {
- return ((this.visible &&
- typeof this.dataMax !== 'undefined' &&
- typeof this.dataMin !== 'undefined') || ( // #3703
- this.visible &&
- this.yData &&
- this.yData.length > 0) // #9758
- );
- },
- /**
- * Return an auto incremented x value based on the pointStart and
- * pointInterval options. This is only used if an x value is not given
- * for the point that calls autoIncrement.
- *
- * @private
- * @function Highcharts.Series#autoIncrement
- * @return {number}
- */
- autoIncrement: function () {
- var options = this.options,
- xIncrement = this.xIncrement,
- date,
- pointInterval,
- pointIntervalUnit = options.pointIntervalUnit,
- time = this.chart.time;
- xIncrement = pick(xIncrement, options.pointStart, 0);
- this.pointInterval = pointInterval = pick(this.pointInterval, options.pointInterval, 1);
- // Added code for pointInterval strings
- if (pointIntervalUnit) {
- date = new time.Date(xIncrement);
- if (pointIntervalUnit === 'day') {
- time.set('Date', date, time.get('Date', date) + pointInterval);
- }
- else if (pointIntervalUnit === 'month') {
- time.set('Month', date, time.get('Month', date) + pointInterval);
- }
- else if (pointIntervalUnit === 'year') {
- time.set('FullYear', date, time.get('FullYear', date) + pointInterval);
- }
- pointInterval = date.getTime() - xIncrement;
- }
- this.xIncrement = xIncrement + pointInterval;
- return xIncrement;
- },
- /**
- * Internal function to set properties for series if data sorting is
- * enabled.
- *
- * @private
- * @function Highcharts.Series#setDataSortingOptions
- * @return {void}
- */
- setDataSortingOptions: function () {
- var options = this.options;
- extend(this, {
- requireSorting: false,
- sorted: false,
- enabledDataSorting: true,
- allowDG: false
- });
- // To allow unsorted data for column series.
- if (!defined(options.pointRange)) {
- options.pointRange = 1;
- }
- },
- /**
- * Set the series options by merging from the options tree. Called
- * internally on initializing and updating series. This function will
- * not redraw the series. For API usage, use {@link Series#update}.
- * @private
- * @function Highcharts.Series#setOptions
- * @param {Highcharts.SeriesOptionsType} itemOptions
- * The series options.
- * @return {Highcharts.SeriesOptionsType}
- * @fires Highcharts.Series#event:afterSetOptions
- */
- setOptions: function (itemOptions) {
- var chart = this.chart,
- chartOptions = chart.options,
- plotOptions = chartOptions.plotOptions,
- userOptions = chart.userOptions || {},
- seriesUserOptions = merge(itemOptions),
- options,
- zones,
- zone,
- styledMode = chart.styledMode,
- e = {
- plotOptions: plotOptions,
- userOptions: seriesUserOptions
- };
- fireEvent(this, 'setOptions', e);
- // These may be modified by the event
- var typeOptions = e.plotOptions[this.type],
- userPlotOptions = (userOptions.plotOptions || {});
- // use copy to prevent undetected changes (#9762)
- /**
- * Contains series options by the user without defaults.
- * @name Highcharts.Series#userOptions
- * @type {Highcharts.SeriesOptionsType}
- */
- this.userOptions = e.userOptions;
- options = merge(typeOptions, plotOptions.series,
- // #3881, chart instance plotOptions[type] should trump
- // plotOptions.series
- userOptions.plotOptions &&
- userOptions.plotOptions[this.type], seriesUserOptions);
- // The tooltip options are merged between global and series specific
- // options. Importance order asscendingly:
- // globals: (1)tooltip, (2)plotOptions.series,
- // (3)plotOptions[this.type]
- // init userOptions with possible later updates: 4-6 like 1-3 and
- // (7)this series options
- this.tooltipOptions = merge(defaultOptions.tooltip, // 1
- defaultOptions.plotOptions.series &&
- defaultOptions.plotOptions.series.tooltip, // 2
- defaultOptions.plotOptions[this.type].tooltip, // 3
- chartOptions.tooltip.userOptions, // 4
- plotOptions.series &&
- plotOptions.series.tooltip, // 5
- plotOptions[this.type].tooltip, // 6
- seriesUserOptions.tooltip // 7
- );
- // When shared tooltip, stickyTracking is true by default,
- // unless user says otherwise.
- this.stickyTracking = pick(seriesUserOptions.stickyTracking, userPlotOptions[this.type] &&
- userPlotOptions[this.type].stickyTracking, userPlotOptions.series && userPlotOptions.series.stickyTracking, (this.tooltipOptions.shared && !this.noSharedTooltip ?
- true :
- options.stickyTracking));
- // Delete marker object if not allowed (#1125)
- if (typeOptions.marker === null) {
- delete options.marker;
- }
- // Handle color zones
- this.zoneAxis = options.zoneAxis;
- zones = this.zones = (options.zones || []).slice();
- if ((options.negativeColor || options.negativeFillColor) &&
- !options.zones) {
- zone = {
- value: options[this.zoneAxis + 'Threshold'] ||
- options.threshold ||
- 0,
- className: 'highcharts-negative'
- };
- if (!styledMode) {
- zone.color = options.negativeColor;
- zone.fillColor = options.negativeFillColor;
- }
- zones.push(zone);
- }
- if (zones.length) { // Push one extra zone for the rest
- if (defined(zones[zones.length - 1].value)) {
- zones.push(styledMode ? {} : {
- color: this.color,
- fillColor: this.fillColor
- });
- }
- }
- fireEvent(this, 'afterSetOptions', { options: options });
- return options;
- },
- /**
- * Return series name in "Series {Number}" format or the one defined by
- * a user. This method can be simply overridden as series name format
- * can vary (e.g. technical indicators).
- *
- * @function Highcharts.Series#getName
- * @return {string}
- * The series name.
- */
- getName: function () {
- // #4119
- return pick(this.options.name, 'Series ' + (this.index + 1));
- },
- /**
- * @private
- * @function Highcharts.Series#getCyclic
- * @param {string} prop
- * @param {*} [value]
- * @param {Highcharts.Dictionary<any>} [defaults]
- * @return {void}
- */
- getCyclic: function (prop, value, defaults) {
- var i, chart = this.chart, userOptions = this.userOptions, indexName = prop + 'Index', counterName = prop + 'Counter', len = defaults ? defaults.length : pick(chart.options.chart[prop + 'Count'], chart[prop + 'Count']), setting;
- if (!value) {
- // Pick up either the colorIndex option, or the _colorIndex
- // after Series.update()
- setting = pick(userOptions[indexName], userOptions['_' + indexName]);
- if (defined(setting)) { // after Series.update()
- i = setting;
- }
- else {
- // #6138
- if (!chart.series.length) {
- chart[counterName] = 0;
- }
- userOptions['_' + indexName] = i =
- chart[counterName] % len;
- chart[counterName] += 1;
- }
- if (defaults) {
- value = defaults[i];
- }
- }
- // Set the colorIndex
- if (typeof i !== 'undefined') {
- this[indexName] = i;
- }
- this[prop] = value;
- },
- /**
- * Get the series' color based on either the options or pulled from
- * global options.
- *
- * @private
- * @function Highcharts.Series#getColor
- * @return {void}
- */
- getColor: function () {
- if (this.chart.styledMode) {
- this.getCyclic('color');
- }
- else if (this.options.colorByPoint) {
- // #4359, selected slice got series.color even when colorByPoint
- // was set.
- this.options.color = null;
- }
- else {
- this.getCyclic('color', this.options.color ||
- defaultOptions.plotOptions[this.type].color, this.chart.options.colors);
- }
- },
- /**
- * Get all points' instances created for this series.
- *
- * @private
- * @function Highcharts.Series#getPointsCollection
- * @return {Array<Highcharts.Point>}
- */
- getPointsCollection: function () {
- return (this.hasGroupedData ? this.points : this.data) || [];
- },
- /**
- * Get the series' symbol based on either the options or pulled from
- * global options.
- *
- * @private
- * @function Highcharts.Series#getSymbol
- * @return {void}
- */
- getSymbol: function () {
- var seriesMarkerOption = this.options.marker;
- this.getCyclic('symbol', seriesMarkerOption.symbol, this.chart.options.symbols);
- },
- /**
- * Finds the index of an existing point that matches the given point
- * options.
- *
- * @private
- * @function Highcharts.Series#findPointIndex
- * @param {Highcharts.PointOptionsObject} optionsObject
- * The options of the point.
- * @param {number} fromIndex
- * The index to start searching from, used for optimizing
- * series with required sorting.
- * @returns {number|undefined}
- * Returns the index of a matching point, or undefined if no
- * match is found.
- */
- findPointIndex: function (optionsObject, fromIndex) {
- var id = optionsObject.id,
- x = optionsObject.x,
- oldData = this.points,
- matchingPoint,
- matchedById,
- pointIndex,
- matchKey,
- dataSorting = this.options.dataSorting;
- if (id) {
- matchingPoint = this.chart.get(id);
- }
- else if (this.linkedParent || this.enabledDataSorting) {
- matchKey = (dataSorting && dataSorting.matchByName) ?
- 'name' : 'index';
- matchingPoint = find(oldData, function (oldPoint) {
- return !oldPoint.touched && oldPoint[matchKey] ===
- optionsObject[matchKey];
- });
- // Add unmatched point as a new point
- if (!matchingPoint) {
- return void 0;
- }
- }
- if (matchingPoint) {
- pointIndex = matchingPoint && matchingPoint.index;
- if (typeof pointIndex !== 'undefined') {
- matchedById = true;
- }
- }
- // Search for the same X in the existing data set
- if (typeof pointIndex === 'undefined' && isNumber(x)) {
- pointIndex = this.xData.indexOf(x, fromIndex);
- }
- // Reduce pointIndex if data is cropped
- if (pointIndex !== -1 &&
- typeof pointIndex !== 'undefined' &&
- this.cropped) {
- pointIndex = (pointIndex >= this.cropStart) ?
- pointIndex - this.cropStart : pointIndex;
- }
- if (!matchedById &&
- oldData[pointIndex] && oldData[pointIndex].touched) {
- pointIndex = void 0;
- }
- return pointIndex;
- },
- /**
- * @private
- * @borrows LegendSymbolMixin.drawLineMarker as Highcharts.Series#drawLegendSymbol
- */
- drawLegendSymbol: LegendSymbolMixin.drawLineMarker,
- /**
- * Internal function called from setData. If the point count is the same
- * as is was, or if there are overlapping X values, just run
- * Point.update which is cheaper, allows animation, and keeps references
- * to points. This also allows adding or removing points if the X-es
- * don't match.
- *
- * @private
- * @function Highcharts.Series#updateData
- *
- * @param {Array<Highcharts.PointOptionsType>} data
- *
- * @return {boolean}
- */
- updateData: function (data, animation) {
- var options = this.options,
- dataSorting = options.dataSorting,
- oldData = this.points,
- pointsToAdd = [],
- hasUpdatedByKey,
- i,
- point,
- lastIndex,
- requireSorting = this.requireSorting,
- equalLength = data.length === oldData.length,
- succeeded = true;
- this.xIncrement = null;
- // Iterate the new data
- data.forEach(function (pointOptions, i) {
- var id,
- x,
- pointIndex,
- optionsObject = (defined(pointOptions) &&
- this.pointClass.prototype.optionsToObject.call({ series: this },
- pointOptions)) || {};
- // Get the x of the new data point
- x = optionsObject.x;
- id = optionsObject.id;
- if (id || isNumber(x)) {
- pointIndex = this.findPointIndex(optionsObject, lastIndex);
- // Matching X not found
- // or used already due to ununique x values (#8995),
- // add point (but later)
- if (pointIndex === -1 ||
- typeof pointIndex === 'undefined') {
- pointsToAdd.push(pointOptions);
- // Matching X found, update
- }
- else if (oldData[pointIndex] &&
- pointOptions !== options.data[pointIndex]) {
- oldData[pointIndex].update(pointOptions, false, null, false);
- // Mark it touched, below we will remove all points that
- // are not touched.
- oldData[pointIndex].touched = true;
- // Speed optimize by only searching after last known
- // index. Performs ~20% bettor on large data sets.
- if (requireSorting) {
- lastIndex = pointIndex + 1;
- }
- // Point exists, no changes, don't remove it
- }
- else if (oldData[pointIndex]) {
- oldData[pointIndex].touched = true;
- }
- // If the length is equal and some of the nodes had a
- // match in the same position, we don't want to remove
- // non-matches.
- if (!equalLength ||
- i !== pointIndex ||
- (dataSorting && dataSorting.enabled) ||
- this.hasDerivedData) {
- hasUpdatedByKey = true;
- }
- }
- else {
- // Gather all points that are not matched
- pointsToAdd.push(pointOptions);
- }
- }, this);
- // Remove points that don't exist in the updated data set
- if (hasUpdatedByKey) {
- i = oldData.length;
- while (i--) {
- point = oldData[i];
- if (point && !point.touched && point.remove) {
- point.remove(false, animation);
- }
- }
- // If we did not find keys (ids or x-values), and the length is the
- // same, update one-to-one
- }
- else if (equalLength && (!dataSorting || !dataSorting.enabled)) {
- data.forEach(function (point, i) {
- // .update doesn't exist on a linked, hidden series (#3709)
- // (#10187)
- if (oldData[i].update && point !== oldData[i].y) {
- oldData[i].update(point, false, null, false);
- }
- });
- // Don't add new points since those configs are used above
- pointsToAdd.length = 0;
- // Did not succeed in updating data
- }
- else {
- succeeded = false;
- }
- oldData.forEach(function (point) {
- if (point) {
- point.touched = false;
- }
- });
- if (!succeeded) {
- return false;
- }
- // Add new points
- pointsToAdd.forEach(function (point) {
- this.addPoint(point, false, null, null, false);
- }, this);
- if (this.xIncrement === null &&
- this.xData &&
- this.xData.length) {
- this.xIncrement = arrayMax(this.xData);
- this.autoIncrement();
- }
- return true;
- },
- /**
- * Apply a new set of data to the series and optionally redraw it. The
- * new data array is passed by reference (except in case of
- * `updatePoints`), and may later be mutated when updating the chart
- * data.
- *
- * Note the difference in behaviour when setting the same amount of
- * points, or a different amount of points, as handled by the
- * `updatePoints` parameter.
- *
- * @sample highcharts/members/series-setdata/
- * Set new data from a button
- * @sample highcharts/members/series-setdata-pie/
- * Set data in a pie
- * @sample stock/members/series-setdata/
- * Set new data in Highstock
- * @sample maps/members/series-setdata/
- * Set new data in Highmaps
- *
- * @function Highcharts.Series#setData
- *
- * @param {Array<Highcharts.PointOptionsType>} data
- * Takes an array of data in the same format as described under
- * `series.{type}.data` for the given series type, for example a
- * line series would take data in the form described under
- * [series.line.data](https://api.highcharts.com/highcharts/series.line.data).
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the series is altered. If
- * doing more operations on the chart, it is a good idea to set
- * redraw to false and call {@link Chart#redraw} after.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * When the updated data is the same length as the existing data,
- * points will be updated by default, and animation visualizes
- * how the points are changed. Set false to disable animation, or
- * a configuration object to set duration or easing.
- *
- * @param {boolean} [updatePoints=true]
- * When this is true, points will be updated instead of replaced
- * whenever possible. This occurs a) when the updated data is the
- * same length as the existing data, b) when points are matched
- * by their id's, or c) when points can be matched by X values.
- * This allows updating with animation and performs better. In
- * this case, the original array is not passed by reference. Set
- * `false` to prevent.
- *
- * @return {void}
- */
- setData: function (data, redraw, animation, updatePoints) {
- var series = this,
- oldData = series.points,
- oldDataLength = (oldData && oldData.length) || 0,
- dataLength,
- options = series.options,
- chart = series.chart,
- dataSorting = options.dataSorting,
- firstPoint = null,
- xAxis = series.xAxis,
- i,
- turboThreshold = options.turboThreshold,
- pt,
- xData = this.xData,
- yData = this.yData,
- pointArrayMap = series.pointArrayMap,
- valueCount = pointArrayMap && pointArrayMap.length,
- keys = options.keys,
- indexOfX = 0,
- indexOfY = 1,
- updatedData;
- data = data || [];
- dataLength = data.length;
- redraw = pick(redraw, true);
- if (dataSorting && dataSorting.enabled) {
- data = this.sortData(data);
- }
- // First try to run Point.update which is cheaper, allows animation,
- // and keeps references to points.
- if (updatePoints !== false &&
- dataLength &&
- oldDataLength &&
- !series.cropped &&
- !series.hasGroupedData &&
- series.visible &&
- // Soft updating has no benefit in boost, and causes JS error
- // (#8355)
- !series.isSeriesBoosting) {
- updatedData = this.updateData(data, animation);
- }
- if (!updatedData) {
- // Reset properties
- series.xIncrement = null;
- series.colorCounter = 0; // for series with colorByPoint (#1547)
- // Update parallel arrays
- this.parallelArrays.forEach(function (key) {
- series[key + 'Data'].length = 0;
- });
- // In turbo mode, only one- or twodimensional arrays of numbers
- // are allowed. The first value is tested, and we assume that
- // all the rest are defined the same way. Although the 'for'
- // loops are similar, they are repeated inside each if-else
- // conditional for max performance.
- if (turboThreshold && dataLength > turboThreshold) {
- firstPoint = series.getFirstValidPoint(data);
- if (isNumber(firstPoint)) { // assume all points are numbers
- for (i = 0; i < dataLength; i++) {
- xData[i] = this.autoIncrement();
- yData[i] = data[i];
- }
- // Assume all points are arrays when first point is
- }
- else if (isArray(firstPoint)) {
- if (valueCount) { // [x, low, high] or [x, o, h, l, c]
- for (i = 0; i < dataLength; i++) {
- pt = data[i];
- xData[i] = pt[0];
- yData[i] =
- pt.slice(1, valueCount + 1);
- }
- }
- else { // [x, y]
- if (keys) {
- indexOfX = keys.indexOf('x');
- indexOfY = keys.indexOf('y');
- indexOfX = indexOfX >= 0 ? indexOfX : 0;
- indexOfY = indexOfY >= 0 ? indexOfY : 1;
- }
- for (i = 0; i < dataLength; i++) {
- pt = data[i];
- xData[i] = pt[indexOfX];
- yData[i] = pt[indexOfY];
- }
- }
- }
- else {
- // Highcharts expects configs to be numbers or arrays in
- // turbo mode
- error(12, false, chart);
- }
- }
- else {
- for (i = 0; i < dataLength; i++) {
- // stray commas in oldIE:
- if (typeof data[i] !== 'undefined') {
- pt = { series: series };
- series.pointClass.prototype.applyOptions.apply(pt, [data[i]]);
- series.updateParallelArrays(pt, i);
- }
- }
- }
- // Forgetting to cast strings to numbers is a common caveat when
- // handling CSV or JSON
- if (yData && isString(yData[0])) {
- error(14, true, chart);
- }
- series.data = [];
- series.options.data = series.userOptions.data = data;
- // destroy old points
- i = oldDataLength;
- while (i--) {
- if (oldData[i] && oldData[i].destroy) {
- oldData[i].destroy();
- }
- }
- // reset minRange (#878)
- if (xAxis) {
- xAxis.minRange = xAxis.userMinRange;
- }
- // redraw
- series.isDirty = chart.isDirtyBox = true;
- series.isDirtyData = !!oldData;
- animation = false;
- }
- // Typically for pie series, points need to be processed and
- // generated prior to rendering the legend
- if (options.legendType === 'point') {
- this.processData();
- this.generatePoints();
- }
- if (redraw) {
- chart.redraw(animation);
- }
- },
- /**
- * Internal function to sort series data
- *
- * @private
- * @function Highcharts.Series#sortData
- * @param {Array<Highcharts.PointOptionsType>} data
- * Force data grouping.
- * @return {Array<Highcharts.PointOptionsObject>}
- */
- sortData: function (data) {
- var series = this,
- options = series.options,
- dataSorting = options.dataSorting,
- sortKey = dataSorting.sortKey || 'y',
- sortedData,
- getPointOptionsObject = function (series,
- pointOptions) {
- return (defined(pointOptions) &&
- series.pointClass.prototype.optionsToObject.call({
- series: series
- },
- pointOptions)) || {};
- };
- data.forEach(function (pointOptions, i) {
- data[i] = getPointOptionsObject(series, pointOptions);
- data[i].index = i;
- }, this);
- // Sorting
- sortedData = data.concat().sort(function (a, b) {
- var aValue = getNestedProperty(sortKey,
- a);
- var bValue = getNestedProperty(sortKey,
- b);
- return bValue < aValue ? -1 : bValue > aValue ? 1 : 0;
- });
- // Set x value depending on the position in the array
- sortedData.forEach(function (point, i) {
- point.x = i;
- }, this);
- // Set the same x for linked series points if they don't have their
- // own sorting
- if (series.linkedSeries) {
- series.linkedSeries.forEach(function (linkedSeries) {
- var options = linkedSeries.options,
- seriesData = options.data;
- if ((!options.dataSorting ||
- !options.dataSorting.enabled) &&
- seriesData) {
- seriesData.forEach(function (pointOptions, i) {
- seriesData[i] = getPointOptionsObject(linkedSeries, pointOptions);
- if (data[i]) {
- seriesData[i].x = data[i].x;
- seriesData[i].index = i;
- }
- });
- linkedSeries.setData(seriesData, false);
- }
- });
- }
- return data;
- },
- /**
- * Internal function to process the data by cropping away unused data
- * points if the series is longer than the crop threshold. This saves
- * computing time for large series.
- *
- * @private
- * @function Highcharts.Series#getProcessedData
- * @param {boolean} [forceExtremesFromAll]
- * Force getting extremes of a total series data range.
- * @return {Highcharts.SeriesProcessedDataObject}
- */
- getProcessedData: function (forceExtremesFromAll) {
- var series = this,
- // copied during slice operation:
- processedXData = series.xData,
- processedYData = series.yData,
- dataLength = processedXData.length,
- croppedData,
- cropStart = 0,
- cropped,
- distance,
- closestPointRange,
- xAxis = series.xAxis,
- i, // loop variable
- options = series.options,
- cropThreshold = options.cropThreshold,
- getExtremesFromAll = forceExtremesFromAll ||
- series.getExtremesFromAll ||
- options.getExtremesFromAll, // #4599
- isCartesian = series.isCartesian,
- xExtremes,
- val2lin = xAxis && xAxis.val2lin,
- isLog = !!(xAxis && xAxis.logarithmic),
- throwOnUnsorted = series.requireSorting,
- min,
- max;
- if (xAxis) {
- // corrected for log axis (#3053)
- xExtremes = xAxis.getExtremes();
- min = xExtremes.min;
- max = xExtremes.max;
- }
- // optionally filter out points outside the plot area
- if (isCartesian &&
- series.sorted &&
- !getExtremesFromAll &&
- (!cropThreshold ||
- dataLength > cropThreshold ||
- series.forceCrop)) {
- // it's outside current extremes
- if (processedXData[dataLength - 1] < min ||
- processedXData[0] > max) {
- processedXData = [];
- processedYData = [];
- // only crop if it's actually spilling out
- }
- else if (series.yData && (processedXData[0] < min ||
- processedXData[dataLength - 1] > max)) {
- croppedData = this.cropData(series.xData, series.yData, min, max);
- processedXData = croppedData.xData;
- processedYData = croppedData.yData;
- cropStart = croppedData.start;
- cropped = true;
- }
- }
- // Find the closest distance between processed points
- i = processedXData.length || 1;
- while (--i) {
- distance = (isLog ?
- (val2lin(processedXData[i]) -
- val2lin(processedXData[i - 1])) :
- (processedXData[i] -
- processedXData[i - 1]));
- if (distance > 0 &&
- (typeof closestPointRange === 'undefined' ||
- distance < closestPointRange)) {
- closestPointRange = distance;
- // Unsorted data is not supported by the line tooltip, as well
- // as data grouping and navigation in Stock charts (#725) and
- // width calculation of columns (#1900)
- }
- else if (distance < 0 && throwOnUnsorted) {
- error(15, false, series.chart);
- throwOnUnsorted = false; // Only once
- }
- }
- return {
- xData: processedXData,
- yData: processedYData,
- cropped: cropped,
- cropStart: cropStart,
- closestPointRange: closestPointRange
- };
- },
- /**
- * Internal function to apply processed data.
- * In Highstock, this function is extended to provide data grouping.
- *
- * @private
- * @function Highcharts.Series#processData
- * @param {boolean} [force]
- * Force data grouping.
- * @return {boolean|undefined}
- */
- processData: function (force) {
- var series = this,
- xAxis = series.xAxis,
- processedData;
- // If the series data or axes haven't changed, don't go through
- // this. Return false to pass the message on to override methods
- // like in data grouping.
- if (series.isCartesian &&
- !series.isDirty &&
- !xAxis.isDirty &&
- !series.yAxis.isDirty &&
- !force) {
- return false;
- }
- processedData = series.getProcessedData();
- // Record the properties
- series.cropped = processedData.cropped; // undefined or true
- series.cropStart = processedData.cropStart;
- series.processedXData = processedData.xData;
- series.processedYData = processedData.yData;
- series.closestPointRange =
- series.basePointRange = processedData.closestPointRange;
- },
- /**
- * Iterate over xData and crop values between min and max. Returns
- * object containing crop start/end cropped xData with corresponding
- * part of yData, dataMin and dataMax within the cropped range.
- *
- * @private
- * @function Highcharts.Series#cropData
- * @param {Array<number>} xData
- * @param {Array<number>} yData
- * @param {number} min
- * @param {number} max
- * @param {number} [cropShoulder]
- * @return {Highcharts.SeriesCropDataObject}
- */
- cropData: function (xData, yData, min, max, cropShoulder) {
- var dataLength = xData.length,
- cropStart = 0,
- cropEnd = dataLength,
- i,
- j;
- // line-type series need one point outside
- cropShoulder = pick(cropShoulder, this.cropShoulder);
- // iterate up to find slice start
- for (i = 0; i < dataLength; i++) {
- if (xData[i] >= min) {
- cropStart = Math.max(0, i - cropShoulder);
- break;
- }
- }
- // proceed to find slice end
- for (j = i; j < dataLength; j++) {
- if (xData[j] > max) {
- cropEnd = j + cropShoulder;
- break;
- }
- }
- return {
- xData: xData.slice(cropStart, cropEnd),
- yData: yData.slice(cropStart, cropEnd),
- start: cropStart,
- end: cropEnd
- };
- },
- /**
- * Generate the data point after the data has been processed by cropping
- * away unused points and optionally grouped in Highcharts Stock.
- *
- * @private
- * @function Highcharts.Series#generatePoints
- */
- generatePoints: function () {
- var series = this,
- options = series.options,
- dataOptions = options.data,
- data = series.data,
- dataLength,
- processedXData = series.processedXData,
- processedYData = series.processedYData,
- PointClass = series.pointClass,
- processedDataLength = processedXData.length,
- cropStart = series.cropStart || 0,
- cursor,
- hasGroupedData = series.hasGroupedData,
- keys = options.keys,
- point,
- points = [],
- i;
- if (!data && !hasGroupedData) {
- var arr = [];
- arr.length = dataOptions.length;
- data = series.data = arr;
- }
- if (keys && hasGroupedData) {
- // grouped data has already applied keys (#6590)
- series.options.keys = false;
- }
- for (i = 0; i < processedDataLength; i++) {
- cursor = cropStart + i;
- if (!hasGroupedData) {
- point = data[cursor];
- // #970:
- if (!point &&
- typeof dataOptions[cursor] !== 'undefined') {
- data[cursor] = point = (new PointClass()).init(series, dataOptions[cursor], processedXData[i]);
- }
- }
- else {
- // splat the y data in case of ohlc data array
- point = (new PointClass()).init(series, [processedXData[i]].concat(splat(processedYData[i])));
- /**
- * Highstock only. If a point object is created by data
- * grouping, it doesn't reflect actual points in the raw
- * data. In this case, the `dataGroup` property holds
- * information that points back to the raw data.
- *
- * - `dataGroup.start` is the index of the first raw data
- * point in the group.
- *
- * - `dataGroup.length` is the amount of points in the
- * group.
- *
- * @product highstock
- *
- * @name Highcharts.Point#dataGroup
- * @type {Highcharts.DataGroupingInfoObject|undefined}
- */
- point.dataGroup = series.groupMap[i];
- if (point.dataGroup.options) {
- point.options = point.dataGroup.options;
- extend(point, point.dataGroup.options);
- // Collision of props and options (#9770)
- delete point.dataLabels;
- }
- }
- if (point) { // #6279
- /**
- * Contains the point's index in the `Series.points` array.
- *
- * @name Highcharts.Point#index
- * @type {number}
- * @readonly
- */
- point.index = cursor; // For faster access in Point.update
- points[i] = point;
- }
- }
- // restore keys options (#6590)
- series.options.keys = keys;
- // Hide cropped-away points - this only runs when the number of
- // points is above cropThreshold, or when swithching view from
- // non-grouped data to grouped data (#637)
- if (data &&
- (processedDataLength !== (dataLength = data.length) ||
- hasGroupedData)) {
- for (i = 0; i < dataLength; i++) {
- // when has grouped data, clear all points
- if (i === cropStart && !hasGroupedData) {
- i += processedDataLength;
- }
- if (data[i]) {
- data[i].destroyElements();
- data[i].plotX = void 0; // #1003
- }
- }
- }
- /**
- * Read only. An array containing those values converted to points.
- * In case the series data length exceeds the `cropThreshold`, or if
- * the data is grouped, `series.data` doesn't contain all the
- * points. Also, in case a series is hidden, the `data` array may be
- * empty. To access raw values, `series.options.data` will always be
- * up to date. `Series.data` only contains the points that have been
- * created on demand. To modify the data, use
- * {@link Highcharts.Series#setData} or
- * {@link Highcharts.Point#update}.
- *
- * @see Series.points
- *
- * @name Highcharts.Series#data
- * @type {Array<Highcharts.Point>}
- */
- series.data = data;
- /**
- * An array containing all currently visible point objects. In case
- * of cropping, the cropped-away points are not part of this array.
- * The `series.points` array starts at `series.cropStart` compared
- * to `series.data` and `series.options.data`. If however the series
- * data is grouped, these can't be correlated one to one. To modify
- * the data, use {@link Highcharts.Series#setData} or
- * {@link Highcharts.Point#update}.
- *
- * @name Highcharts.Series#points
- * @type {Array<Highcharts.Point>}
- */
- series.points = points;
- fireEvent(this, 'afterGeneratePoints');
- },
- /**
- * Get current X extremes for the visible data.
- *
- * @private
- * @function Highcharts.Series#getXExtremes
- *
- * @param {Array<number>} xData
- * The data to inspect. Defaults to the current data within the
- * visible range.
- * @return {Highcharts.RangeObject}
- */
- getXExtremes: function (xData) {
- return {
- min: arrayMin(xData),
- max: arrayMax(xData)
- };
- },
- /**
- * Calculate Y extremes for the visible data. The result is returned
- * as an object with `dataMin` and `dataMax` properties.
- *
- * @private
- * @function Highcharts.Series#getExtremes
- * @param {Array<number>} [yData]
- * The data to inspect. Defaults to the current data within the
- * visible range.
- * @param {boolean} [forceExtremesFromAll]
- * Force getting extremes of a total series data range.
- * @return {Highcharts.DataExtremesObject}
- */
- getExtremes: function (yData, forceExtremesFromAll) {
- var xAxis = this.xAxis,
- yAxis = this.yAxis,
- xData = this.processedXData || this.xData,
- yDataLength,
- activeYData = [],
- activeCounter = 0,
- // #2117, need to compensate for log X axis
- xExtremes,
- xMin = 0,
- xMax = 0,
- validValue,
- withinRange,
- // Handle X outside the viewed area. This does not work with
- // non-sorted data like scatter (#7639).
- shoulder = this.requireSorting ? this.cropShoulder : 0,
- positiveValuesOnly = yAxis ? yAxis.positiveValuesOnly : false,
- x,
- y,
- i,
- j;
- yData = yData || this.stackedYData || this.processedYData || [];
- yDataLength = yData.length;
- if (xAxis) {
- xExtremes = xAxis.getExtremes();
- xMin = xExtremes.min;
- xMax = xExtremes.max;
- }
- for (i = 0; i < yDataLength; i++) {
- x = xData[i];
- y = yData[i];
- // For points within the visible range, including the first
- // point outside the visible range (#7061), consider y extremes.
- validValue = ((isNumber(y) || isArray(y)) &&
- ((y.length || y > 0) || !positiveValuesOnly));
- withinRange = (forceExtremesFromAll ||
- this.getExtremesFromAll ||
- this.options.getExtremesFromAll ||
- this.cropped ||
- !xAxis || // for colorAxis support
- ((xData[i + shoulder] || x) >= xMin &&
- (xData[i - shoulder] || x) <= xMax));
- if (validValue && withinRange) {
- j = y.length;
- if (j) { // array, like ohlc or range data
- while (j--) {
- if (isNumber(y[j])) { // #7380, #11513
- activeYData[activeCounter++] = y[j];
- }
- }
- }
- else {
- activeYData[activeCounter++] = y;
- }
- }
- }
- var dataExtremes = {
- dataMin: arrayMin(activeYData),
- dataMax: arrayMax(activeYData)
- };
- fireEvent(this, 'afterGetExtremes', { dataExtremes: dataExtremes });
- return dataExtremes;
- },
- /**
- * Set the current data extremes as `dataMin` and `dataMax` on the
- * Series item. Use this only when the series properties should be
- * updated.
- *
- * @private
- * @function Highcharts.Series#applyExtremes
- * @return {void}
- */
- applyExtremes: function () {
- var dataExtremes = this.getExtremes();
- /**
- * Contains the minimum value of the series' data point. Some series
- * types like `networkgraph` do not support this property as they
- * lack a `y`-value.
- * @name Highcharts.Series#dataMin
- * @type {number|undefined}
- * @readonly
- */
- this.dataMin = dataExtremes.dataMin;
- /**
- * Contains the maximum value of the series' data point. Some series
- * types like `networkgraph` do not support this property as they
- * lack a `y`-value.
- * @name Highcharts.Series#dataMax
- * @type {number|undefined}
- * @readonly
- */
- this.dataMax = dataExtremes.dataMax;
- return dataExtremes;
- },
- /**
- * Find and return the first non null point in the data
- *
- * @private
- * @function Highcharts.Series.getFirstValidPoint
- * @param {Array<Highcharts.PointOptionsType>} data
- * Array of options for points
- *
- * @return {Highcharts.PointOptionsType}
- */
- getFirstValidPoint: function (data) {
- var firstPoint = null,
- dataLength = data.length,
- i = 0;
- while (firstPoint === null && i < dataLength) {
- firstPoint = data[i];
- i++;
- }
- return firstPoint;
- },
- /**
- * Translate data points from raw data values to chart specific
- * positioning data needed later in the `drawPoints` and `drawGraph`
- * functions. This function can be overridden in plugins and custom
- * series type implementations.
- *
- * @function Highcharts.Series#translate
- * @return {void}
- * @fires Highcharts.Series#events:translate
- */
- translate: function () {
- if (!this.processedXData) { // hidden series
- this.processData();
- }
- this.generatePoints();
- var series = this,
- options = series.options,
- stacking = options.stacking,
- xAxis = series.xAxis,
- categories = xAxis.categories,
- enabledDataSorting = series.enabledDataSorting,
- yAxis = series.yAxis,
- points = series.points,
- dataLength = points.length,
- hasModifyValue = !!series.modifyValue,
- i,
- pointPlacement = series.pointPlacementToXValue(), // #7860
- dynamicallyPlaced = Boolean(pointPlacement),
- threshold = options.threshold,
- stackThreshold = options.startFromThreshold ? threshold : 0,
- plotX,
- lastPlotX,
- stackIndicator,
- zoneAxis = this.zoneAxis || 'y',
- closestPointRangePx = Number.MAX_VALUE;
- /**
- * Plotted coordinates need to be within a limited range. Drawing
- * too far outside the viewport causes various rendering issues
- * (#3201, #3923, #7555).
- * @private
- */
- function limitedRange(val) {
- return clamp(val, -1e5, 1e5);
- }
- // Translate each point
- for (i = 0; i < dataLength; i++) {
- var point = points[i],
- xValue = point.x,
- yValue = point.y,
- yBottom = point.low,
- stack = stacking && yAxis.stacking && yAxis.stacking.stacks[(series.negStacks &&
- yValue <
- (stackThreshold ? 0 : threshold) ?
- '-' :
- '') + series.stackKey],
- pointStack,
- stackValues;
- if (yAxis.positiveValuesOnly && !yAxis.validatePositiveValue(yValue) ||
- xAxis.positiveValuesOnly && !xAxis.validatePositiveValue(xValue)) {
- point.isNull = true;
- }
- // Get the plotX translation
- point.plotX = plotX = correctFloat(// #5236
- limitedRange(xAxis.translate(// #3923
- xValue, 0, 0, 0, 1, pointPlacement, this.type === 'flags')) // #3923
- );
- // Calculate the bottom y value for stacked series
- if (stacking &&
- series.visible &&
- stack &&
- stack[xValue]) {
- stackIndicator = series.getStackIndicator(stackIndicator, xValue, series.index);
- if (!point.isNull) {
- pointStack = stack[xValue];
- stackValues =
- pointStack.points[stackIndicator.key];
- }
- }
- if (isArray(stackValues)) {
- yBottom = stackValues[0];
- yValue = stackValues[1];
- if (yBottom === stackThreshold &&
- stackIndicator.key ===
- stack[xValue].base) {
- yBottom = pick((isNumber(threshold) && threshold), yAxis.min);
- }
- // #1200, #1232
- if (yAxis.positiveValuesOnly && yBottom <= 0) {
- yBottom = null;
- }
- point.total = point.stackTotal = pointStack.total;
- point.percentage =
- pointStack.total &&
- (point.y / pointStack.total * 100);
- point.stackY = yValue;
- // Place the stack label
- // in case of variwide series (where widths of points are
- // different in most cases), stack labels are positioned
- // wrongly, so the call of the setOffset is omited here and
- // labels are correctly positioned later, at the end of the
- // variwide's translate function (#10962)
- if (!series.irregularWidths) {
- pointStack.setOffset(series.pointXOffset || 0, series.barW || 0);
- }
- }
- // Set translated yBottom or remove it
- point.yBottom = defined(yBottom) ?
- limitedRange(yAxis.translate(yBottom, 0, 1, 0, 1)) :
- null;
- // general hook, used for Highstock compare mode
- if (hasModifyValue) {
- yValue = series.modifyValue(yValue, point);
- }
- // Set the the plotY value, reset it for redraws
- // #3201
- point.plotY = ((typeof yValue === 'number' && yValue !== Infinity) ?
- limitedRange(yAxis.translate(yValue, 0, 1, 0, 1)) :
- void 0);
- point.isInside = this.isPointInside(point);
- // Set client related positions for mouse tracking
- point.clientX = dynamicallyPlaced ?
- correctFloat(xAxis.translate(xValue, 0, 0, 0, 1, pointPlacement)) :
- plotX; // #1514, #5383, #5518
- // Negative points. For bubble charts, this means negative z
- // values (#9728)
- point.negative = point[zoneAxis] < (options[zoneAxis + 'Threshold'] ||
- threshold ||
- 0);
- // some API data
- point.category = (categories &&
- typeof categories[point.x] !== 'undefined' ?
- categories[point.x] :
- point.x);
- // Determine auto enabling of markers (#3635, #5099)
- if (!point.isNull && point.visible !== false) {
- if (typeof lastPlotX !== 'undefined') {
- closestPointRangePx = Math.min(closestPointRangePx, Math.abs(plotX - lastPlotX));
- }
- lastPlotX = plotX;
- }
- // Find point zone
- point.zone = (this.zones.length && point.getZone());
- // Animate new points with data sorting
- if (!point.graphic && series.group && enabledDataSorting) {
- point.isNew = true;
- }
- }
- series.closestPointRangePx = closestPointRangePx;
- fireEvent(this, 'afterTranslate');
- },
- /**
- * Return the series points with null points filtered out.
- *
- * @function Highcharts.Series#getValidPoints
- *
- * @param {Array<Highcharts.Point>} [points]
- * The points to inspect, defaults to {@link Series.points}.
- *
- * @param {boolean} [insideOnly=false]
- * Whether to inspect only the points that are inside the visible
- * view.
- *
- * @param {boolean} [allowNull=false]
- * Whether to allow null points to pass as valid points.
- *
- * @return {Array<Highcharts.Point>}
- * The valid points.
- */
- getValidPoints: function (points, insideOnly, allowNull) {
- var chart = this.chart;
- // #3916, #5029, #5085
- return (points || this.points || []).filter(function isValidPoint(point) {
- if (insideOnly && !chart.isInsidePlot(point.plotX, point.plotY, chart.inverted)) {
- return false;
- }
- return point.visible !== false &&
- (allowNull || !point.isNull);
- });
- },
- /**
- * Get the clipping for the series. Could be called for a series to
- * initiate animating the clip or to set the final clip (only width
- * and x).
- *
- * @private
- * @function Highcharts.Series#getClip
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Initialize the animation.
- * @param {boolean} [finalBox]
- * Final size for the clip - end state for the animation.
- * @return {Highcharts.Dictionary<number>}
- */
- getClipBox: function (animation, finalBox) {
- var series = this,
- options = series.options,
- chart = series.chart,
- inverted = chart.inverted,
- xAxis = series.xAxis,
- yAxis = xAxis && series.yAxis,
- clipBox,
- scrollablePlotAreaOptions = chart.options.chart.scrollablePlotArea || {};
- if (animation && options.clip === false && yAxis) {
- // support for not clipped series animation (#10450)
- clipBox = inverted ? {
- y: -chart.chartWidth + yAxis.len + yAxis.pos,
- height: chart.chartWidth,
- width: chart.chartHeight,
- x: -chart.chartHeight + xAxis.len + xAxis.pos
- } : {
- y: -yAxis.pos,
- height: chart.chartHeight,
- width: chart.chartWidth,
- x: -xAxis.pos
- };
- // x and width will be changed later when setting for animation
- // initial state in Series.setClip
- }
- else {
- clipBox = series.clipBox || chart.clipBox;
- if (finalBox) {
- clipBox.width = chart.plotSizeX;
- clipBox.x = (chart.scrollablePixelsX || 0) *
- (scrollablePlotAreaOptions.scrollPositionX || 0);
- }
- }
- return !finalBox ? clipBox : {
- width: clipBox.width,
- x: clipBox.x
- };
- },
- /**
- * Set the clipping for the series. For animated series it is called
- * twice, first to initiate animating the clip then the second time
- * without the animation to set the final clip.
- *
- * @private
- * @function Highcharts.Series#setClip
- * @param {boolean|Highcharts.AnimationOptionsObject} [animation]
- */
- setClip: function (animation) {
- var chart = this.chart, options = this.options, renderer = chart.renderer, inverted = chart.inverted, seriesClipBox = this.clipBox, clipBox = this.getClipBox(animation), sharedClipKey = this.sharedClipKey ||
- [
- '_sharedClip',
- animation && animation.duration,
- animation && animation.easing,
- clipBox.height,
- options.xAxis,
- options.yAxis
- ].join(','), // #4526
- clipRect = chart[sharedClipKey], markerClipRect = chart[sharedClipKey + 'm'];
- if (animation) {
- clipBox.width = 0;
- if (inverted) {
- clipBox.x = chart.plotHeight +
- (options.clip !== false ? 0 : chart.plotTop);
- }
- }
- // If a clipping rectangle with the same properties is currently
- // present in the chart, use that.
- if (!clipRect) {
- // When animation is set, prepare the initial positions
- if (animation) {
- chart[sharedClipKey + 'm'] = markerClipRect =
- renderer.clipRect(
- // include the width of the first marker
- inverted ? chart.plotSizeX + 99 : -99, inverted ? -chart.plotLeft : -chart.plotTop, 99, inverted ? chart.chartWidth : chart.chartHeight);
- }
- chart[sharedClipKey] = clipRect = renderer.clipRect(clipBox);
- // Create hashmap for series indexes
- clipRect.count = { length: 0 };
- // When the series is rendered again before starting animating, in
- // compliance to a responsive rule
- }
- else if (!chart.hasLoaded) {
- clipRect.attr(clipBox);
- }
- if (animation) {
- if (!clipRect.count[this.index]) {
- clipRect.count[this.index] = true;
- clipRect.count.length += 1;
- }
- }
- if (options.clip !== false || animation) {
- this.group.clip(animation || seriesClipBox ? clipRect : chart.clipRect);
- this.markerGroup.clip(markerClipRect);
- this.sharedClipKey = sharedClipKey;
- }
- // Remove the shared clipping rectangle when all series are shown
- if (!animation) {
- if (clipRect.count[this.index]) {
- delete clipRect.count[this.index];
- clipRect.count.length -= 1;
- }
- if (clipRect.count.length === 0 &&
- sharedClipKey &&
- chart[sharedClipKey]) {
- if (!seriesClipBox) {
- chart[sharedClipKey] =
- chart[sharedClipKey].destroy();
- }
- if (chart[sharedClipKey + 'm']) {
- chart[sharedClipKey + 'm'] =
- chart[sharedClipKey + 'm'].destroy();
- }
- }
- }
- },
- /**
- * Animate in the series. Called internally twice. First with the `init`
- * parameter set to true, which sets up the initial state of the
- * animation. Then when ready, it is called with the `init` parameter
- * undefined, in order to perform the actual animation. After the
- * second run, the function is removed.
- *
- * @function Highcharts.Series#animate
- *
- * @param {boolean} [init]
- * Initialize the animation.
- */
- animate: function (init) {
- var series = this,
- chart = series.chart,
- animation = animObject(series.options.animation),
- clipRect,
- sharedClipKey,
- finalBox;
- // Initialize the animation. Set up the clipping rectangle.
- if (!chart.hasRendered) {
- if (init) {
- series.setClip(animation);
- // Run the animation
- }
- else {
- sharedClipKey = this.sharedClipKey;
- clipRect = chart[sharedClipKey];
- finalBox = series.getClipBox(animation, true);
- if (clipRect) {
- clipRect.animate(finalBox, animation);
- }
- if (chart[sharedClipKey + 'm']) {
- chart[sharedClipKey + 'm'].animate({
- width: finalBox.width + 99,
- x: finalBox.x - (chart.inverted ? 0 : 99)
- }, animation);
- }
- }
- }
- },
- /**
- * This runs after animation to land on the final plot clipping.
- *
- * @private
- * @function Highcharts.Series#afterAnimate
- * @fires Highcharts.Series#event:afterAnimate
- */
- afterAnimate: function () {
- this.setClip();
- fireEvent(this, 'afterAnimate');
- this.finishedAnimating = true;
- },
- /**
- * Draw the markers for line-like series types, and columns or other
- * graphical representation for {@link Point} objects for other series
- * types. The resulting element is typically stored as
- * {@link Point.graphic}, and is created on the first call and updated
- * and moved on subsequent calls.
- *
- * @function Highcharts.Series#drawPoints
- */
- drawPoints: function () {
- var series = this,
- points = series.points,
- chart = series.chart,
- i,
- point,
- graphic,
- verb,
- options = series.options,
- seriesMarkerOptions = options.marker,
- pointMarkerOptions,
- hasPointMarker,
- markerGroup = (series[series.specialGroup] ||
- series.markerGroup),
- xAxis = series.xAxis,
- markerAttribs,
- globallyEnabled = pick(seriesMarkerOptions.enabled, !xAxis || xAxis.isRadial ? true : null,
- // Use larger or equal as radius is null in bubbles (#6321)
- series.closestPointRangePx >= (seriesMarkerOptions.enabledThreshold *
- seriesMarkerOptions.radius));
- if (seriesMarkerOptions.enabled !== false ||
- series._hasPointMarkers) {
- for (i = 0; i < points.length; i++) {
- point = points[i];
- graphic = point.graphic;
- verb = graphic ? 'animate' : 'attr';
- pointMarkerOptions = point.marker || {};
- hasPointMarker = !!point.marker;
- var shouldDrawMarker = ((globallyEnabled &&
- typeof pointMarkerOptions.enabled === 'undefined') || pointMarkerOptions.enabled) && !point.isNull && point.visible !== false;
- // only draw the point if y is defined
- if (shouldDrawMarker) {
- // Shortcuts
- var symbol = pick(pointMarkerOptions.symbol,
- series.symbol);
- markerAttribs = series.markerAttribs(point, (point.selected && 'select'));
- // Set starting position for point sliding animation.
- if (series.enabledDataSorting) {
- point.startXPos = xAxis.reversed ?
- -markerAttribs.width :
- xAxis.width;
- }
- var isInside = point.isInside !== false;
- if (graphic) { // update
- // Since the marker group isn't clipped, each
- // individual marker must be toggled
- graphic[isInside ? 'show' : 'hide'](isInside)
- .animate(markerAttribs);
- }
- else if (isInside &&
- (markerAttribs.width > 0 || point.hasImage)) {
- /**
- * The graphic representation of the point.
- * Typically this is a simple shape, like a `rect`
- * for column charts or `path` for line markers, but
- * for some complex series types like boxplot or 3D
- * charts, the graphic may be a `g` element
- * containing other shapes. The graphic is generated
- * the first time {@link Series#drawPoints} runs,
- * and updated and moved on subsequent runs.
- *
- * @name Point#graphic
- * @type {SVGElement}
- */
- point.graphic = graphic = chart.renderer
- .symbol(symbol, markerAttribs.x, markerAttribs.y, markerAttribs.width, markerAttribs.height, hasPointMarker ?
- pointMarkerOptions :
- seriesMarkerOptions)
- .add(markerGroup);
- // Sliding animation for new points
- if (series.enabledDataSorting &&
- chart.hasRendered) {
- graphic.attr({
- x: point.startXPos
- });
- verb = 'animate';
- }
- }
- if (graphic && verb === 'animate') { // update
- // Since the marker group isn't clipped, each
- // individual marker must be toggled
- graphic[isInside ? 'show' : 'hide'](isInside)
- .animate(markerAttribs);
- }
- // Presentational attributes
- if (graphic && !chart.styledMode) {
- graphic[verb](series.pointAttribs(point, (point.selected && 'select')));
- }
- if (graphic) {
- graphic.addClass(point.getClassName(), true);
- }
- }
- else if (graphic) {
- point.graphic = graphic.destroy(); // #1269
- }
- }
- }
- },
- /**
- * Get non-presentational attributes for a point. Used internally for
- * both styled mode and classic. Can be overridden for different series
- * types.
- *
- * @see Series#pointAttribs
- *
- * @function Highcharts.Series#markerAttribs
- *
- * @param {Highcharts.Point} point
- * The Point to inspect.
- *
- * @param {string} [state]
- * The state, can be either `hover`, `select` or undefined.
- *
- * @return {Highcharts.SVGAttributes}
- * A hash containing those attributes that are not settable from
- * CSS.
- */
- markerAttribs: function (point, state) {
- var seriesOptions = this.options,
- seriesMarkerOptions = seriesOptions.marker,
- seriesStateOptions,
- pointMarkerOptions = point.marker || {},
- symbol = (pointMarkerOptions.symbol ||
- seriesMarkerOptions.symbol),
- pointStateOptions,
- radius = pick(pointMarkerOptions.radius,
- seriesMarkerOptions.radius),
- attribs;
- // Handle hover and select states
- if (state) {
- seriesStateOptions = seriesMarkerOptions.states[state];
- pointStateOptions = pointMarkerOptions.states &&
- pointMarkerOptions.states[state];
- radius = pick(pointStateOptions && pointStateOptions.radius, seriesStateOptions && seriesStateOptions.radius, radius + (seriesStateOptions && seriesStateOptions.radiusPlus ||
- 0));
- }
- point.hasImage = symbol && symbol.indexOf('url') === 0;
- if (point.hasImage) {
- radius = 0; // and subsequently width and height is not set
- }
- attribs = {
- // Math.floor for #1843:
- x: seriesOptions.crisp ?
- Math.floor(point.plotX) - radius :
- point.plotX - radius,
- y: point.plotY - radius
- };
- if (radius) {
- attribs.width = attribs.height = 2 * radius;
- }
- return attribs;
- },
- /**
- * Internal function to get presentational attributes for each point.
- * Unlike {@link Series#markerAttribs}, this function should return
- * those attributes that can also be set in CSS. In styled mode,
- * `pointAttribs` won't be called.
- *
- * @private
- * @function Highcharts.Series#pointAttribs
- *
- * @param {Highcharts.Point} [point]
- * The point instance to inspect.
- *
- * @param {string} [state]
- * The point state, can be either `hover`, `select` or 'normal'.
- * If undefined, normal state is assumed.
- *
- * @return {Highcharts.SVGAttributes}
- * The presentational attributes to be set on the point.
- */
- pointAttribs: function (point, state) {
- var seriesMarkerOptions = this.options.marker,
- seriesStateOptions,
- pointOptions = point && point.options,
- pointMarkerOptions = ((pointOptions && pointOptions.marker) || {}),
- pointStateOptions,
- color = this.color,
- pointColorOption = pointOptions && pointOptions.color,
- pointColor = point && point.color,
- strokeWidth = pick(pointMarkerOptions.lineWidth,
- seriesMarkerOptions.lineWidth),
- zoneColor = point && point.zone && point.zone.color,
- fill,
- stroke,
- opacity = 1;
- color = (pointColorOption ||
- zoneColor ||
- pointColor ||
- color);
- fill = (pointMarkerOptions.fillColor ||
- seriesMarkerOptions.fillColor ||
- color);
- stroke = (pointMarkerOptions.lineColor ||
- seriesMarkerOptions.lineColor ||
- color);
- // Handle hover and select states
- state = state || 'normal';
- if (state) {
- seriesStateOptions = seriesMarkerOptions.states[state];
- pointStateOptions = (pointMarkerOptions.states &&
- pointMarkerOptions.states[state]) || {};
- strokeWidth = pick(pointStateOptions.lineWidth, seriesStateOptions.lineWidth, strokeWidth + pick(pointStateOptions.lineWidthPlus, seriesStateOptions.lineWidthPlus, 0));
- fill = (pointStateOptions.fillColor ||
- seriesStateOptions.fillColor ||
- fill);
- stroke = (pointStateOptions.lineColor ||
- seriesStateOptions.lineColor ||
- stroke);
- opacity = pick(pointStateOptions.opacity, seriesStateOptions.opacity, opacity);
- }
- return {
- 'stroke': stroke,
- 'stroke-width': strokeWidth,
- 'fill': fill,
- 'opacity': opacity
- };
- },
- /**
- * Clear DOM objects and free up memory.
- *
- * @private
- * @function Highcharts.Series#destroy
- * @param {boolean} [keepEventsForUpdate]
- * @return {void}
- * @fires Highcharts.Series#event:destroy
- */
- destroy: function (keepEventsForUpdate) {
- var series = this,
- chart = series.chart,
- issue134 = /AppleWebKit\/533/.test(win.navigator.userAgent),
- destroy,
- i,
- data = series.data || [],
- point,
- axis;
- // add event hook
- fireEvent(series, 'destroy');
- // remove events
- this.removeEvents(keepEventsForUpdate);
- // erase from axes
- (series.axisTypes || []).forEach(function (AXIS) {
- axis = series[AXIS];
- if (axis && axis.series) {
- erase(axis.series, series);
- axis.isDirty = axis.forceRedraw = true;
- }
- });
- // remove legend items
- if (series.legendItem) {
- series.chart.legend.destroyItem(series);
- }
- // destroy all points with their elements
- i = data.length;
- while (i--) {
- point = data[i];
- if (point && point.destroy) {
- point.destroy();
- }
- }
- series.points = null;
- // Clear the animation timeout if we are destroying the series
- // during initial animation
- U.clearTimeout(series.animationTimeout);
- // Destroy all SVGElements associated to the series
- objectEach(series, function (val, prop) {
- // Survive provides a hook for not destroying
- if (val instanceof SVGElement && !val.survive) {
- // issue 134 workaround
- destroy = issue134 && prop === 'group' ?
- 'hide' :
- 'destroy';
- val[destroy]();
- }
- });
- // remove from hoverSeries
- if (chart.hoverSeries === series) {
- chart.hoverSeries = null;
- }
- erase(chart.series, series);
- chart.orderSeries();
- // clear all members
- objectEach(series, function (val, prop) {
- if (!keepEventsForUpdate || prop !== 'hcEvents') {
- delete series[prop];
- }
- });
- },
- /**
- * Get the graph path.
- *
- * @private
- * @function Highcharts.Series#getGraphPath
- * @param {Array<Highcharts.Point>} points
- * @param {boolean} [nullsAsZeroes]
- * @param {boolean} [connectCliffs]
- * @return {Highcharts.SVGPathArray}
- */
- getGraphPath: function (points, nullsAsZeroes, connectCliffs) {
- var series = this,
- options = series.options,
- step = options.step,
- reversed,
- graphPath = [],
- xMap = [],
- gap;
- points = points || series.points;
- // Bottom of a stack is reversed
- reversed = points.reversed;
- if (reversed) {
- points.reverse();
- }
- // Reverse the steps (#5004)
- step = {
- right: 1,
- center: 2
- }[step] || (step && 3);
- if (step && reversed) {
- step = 4 - step;
- }
- // Remove invalid points, especially in spline (#5015)
- points = this.getValidPoints(points, false, !(options.connectNulls && !nullsAsZeroes && !connectCliffs));
- // Build the line
- points.forEach(function (point, i) {
- var plotX = point.plotX,
- plotY = point.plotY,
- lastPoint = points[i - 1],
- // the path to this point from the previous
- pathToPoint;
- if ((point.leftCliff || (lastPoint && lastPoint.rightCliff)) &&
- !connectCliffs) {
- gap = true; // ... and continue
- }
- // Line series, nullsAsZeroes is not handled
- if (point.isNull && !defined(nullsAsZeroes) && i > 0) {
- gap = !options.connectNulls;
- // Area series, nullsAsZeroes is set
- }
- else if (point.isNull && !nullsAsZeroes) {
- gap = true;
- }
- else {
- if (i === 0 || gap) {
- pathToPoint = [[
- 'M',
- point.plotX,
- point.plotY
- ]];
- // Generate the spline as defined in the SplineSeries object
- }
- else if (series.getPointSpline) {
- pathToPoint = [series.getPointSpline(points, point, i)];
- }
- else if (step) {
- if (step === 1) { // right
- pathToPoint = [[
- 'L',
- lastPoint.plotX,
- plotY
- ]];
- }
- else if (step === 2) { // center
- pathToPoint = [[
- 'L',
- (lastPoint.plotX + plotX) / 2,
- lastPoint.plotY
- ], [
- 'L',
- (lastPoint.plotX + plotX) / 2,
- plotY
- ]];
- }
- else {
- pathToPoint = [[
- 'L',
- plotX,
- lastPoint.plotY
- ]];
- }
- pathToPoint.push([
- 'L',
- plotX,
- plotY
- ]);
- }
- else {
- // normal line to next point
- pathToPoint = [[
- 'L',
- plotX,
- plotY
- ]];
- }
- // Prepare for animation. When step is enabled, there are
- // two path nodes for each x value.
- xMap.push(point.x);
- if (step) {
- xMap.push(point.x);
- if (step === 2) { // step = center (#8073)
- xMap.push(point.x);
- }
- }
- graphPath.push.apply(graphPath, pathToPoint);
- gap = false;
- }
- });
- graphPath.xMap = xMap;
- series.graphPath = graphPath;
- return graphPath;
- },
- /**
- * Draw the graph. Called internally when rendering line-like series
- * types. The first time it generates the `series.graph` item and
- * optionally other series-wide items like `series.area` for area
- * charts. On subsequent calls these items are updated with new
- * positions and attributes.
- *
- * @function Highcharts.Series#drawGraph
- */
- drawGraph: function () {
- var series = this,
- options = this.options,
- graphPath = (this.gappedPath || this.getGraphPath).call(this),
- styledMode = this.chart.styledMode,
- props = [[
- 'graph',
- 'highcharts-graph'
- ]];
- // Presentational properties
- if (!styledMode) {
- props[0].push((options.lineColor ||
- this.color ||
- '#cccccc' // when colorByPoint = true
- ), options.dashStyle);
- }
- props = series.getZonesGraphs(props);
- // Draw the graph
- props.forEach(function (prop, i) {
- var graphKey = prop[0],
- graph = series[graphKey],
- verb = graph ? 'animate' : 'attr',
- attribs;
- if (graph) {
- graph.endX = series.preventGraphAnimation ?
- null :
- graphPath.xMap;
- graph.animate({ d: graphPath });
- }
- else if (graphPath.length) { // #1487
- /**
- * SVG element of area-based charts. Can be used for styling
- * purposes. If zones are configured, this element will be
- * hidden and replaced by multiple zone areas, accessible
- * via `series['zone-area-x']` (where x is a number,
- * starting with 0).
- *
- * @name Highcharts.Series#area
- * @type {Highcharts.SVGElement|undefined}
- */
- /**
- * SVG element of line-based charts. Can be used for styling
- * purposes. If zones are configured, this element will be
- * hidden and replaced by multiple zone lines, accessible
- * via `series['zone-graph-x']` (where x is a number,
- * starting with 0).
- *
- * @name Highcharts.Series#graph
- * @type {Highcharts.SVGElement|undefined}
- */
- series[graphKey] = graph = series.chart.renderer
- .path(graphPath)
- .addClass(prop[1])
- .attr({ zIndex: 1 }) // #1069
- .add(series.group);
- }
- if (graph && !styledMode) {
- attribs = {
- 'stroke': prop[2],
- 'stroke-width': options.lineWidth,
- // Polygon series use filled graph
- 'fill': (series.fillGraph && series.color) || 'none'
- };
- if (prop[3]) {
- attribs.dashstyle = prop[3];
- }
- else if (options.linecap !== 'square') {
- attribs['stroke-linecap'] =
- attribs['stroke-linejoin'] = 'round';
- }
- graph[verb](attribs)
- // Add shadow to normal series (0) or to first
- // zone (1) #3932
- .shadow((i < 2) && options.shadow);
- }
- // Helpers for animation
- if (graph) {
- graph.startX = graphPath.xMap;
- graph.isArea = graphPath.isArea; // For arearange animation
- }
- });
- },
- /**
- * Get zones properties for building graphs. Extendable by series with
- * multiple lines within one series.
- *
- * @private
- * @function Highcharts.Series#getZonesGraphs
- *
- * @param {Array<Array<string>>} props
- *
- * @return {Array<Array<string>>}
- */
- getZonesGraphs: function (props) {
- // Add the zone properties if any
- this.zones.forEach(function (zone, i) {
- var propset = [
- 'zone-graph-' + i,
- 'highcharts-graph highcharts-zone-graph-' + i + ' ' +
- (zone.className || '')
- ];
- if (!this.chart.styledMode) {
- propset.push((zone.color || this.color), (zone.dashStyle || this.options.dashStyle));
- }
- props.push(propset);
- }, this);
- return props;
- },
- /**
- * Clip the graphs into zones for colors and styling.
- *
- * @private
- * @function Highcharts.Series#applyZones
- * @return {void}
- */
- applyZones: function () {
- var series = this,
- chart = this.chart,
- renderer = chart.renderer,
- zones = this.zones,
- translatedFrom,
- translatedTo,
- clips = (this.clips || []),
- clipAttr,
- graph = this.graph,
- area = this.area,
- chartSizeMax = Math.max(chart.chartWidth,
- chart.chartHeight),
- axis = this[(this.zoneAxis || 'y') + 'Axis'],
- extremes,
- reversed,
- inverted = chart.inverted,
- horiz,
- pxRange,
- pxPosMin,
- pxPosMax,
- ignoreZones = false,
- zoneArea,
- zoneGraph;
- if (zones.length &&
- (graph || area) &&
- axis &&
- typeof axis.min !== 'undefined') {
- reversed = axis.reversed;
- horiz = axis.horiz;
- // The use of the Color Threshold assumes there are no gaps
- // so it is safe to hide the original graph and area
- // unless it is not waterfall series, then use showLine property
- // to set lines between columns to be visible (#7862)
- if (graph && !this.showLine) {
- graph.hide();
- }
- if (area) {
- area.hide();
- }
- // Create the clips
- extremes = axis.getExtremes();
- zones.forEach(function (threshold, i) {
- translatedFrom = reversed ?
- (horiz ? chart.plotWidth : 0) :
- (horiz ? 0 : (axis.toPixels(extremes.min) || 0));
- translatedFrom = clamp(pick(translatedTo, translatedFrom), 0, chartSizeMax);
- translatedTo = clamp(Math.round(axis.toPixels(pick(threshold.value, extremes.max), true) || 0), 0, chartSizeMax);
- if (ignoreZones) {
- translatedFrom = translatedTo =
- axis.toPixels(extremes.max);
- }
- pxRange = Math.abs(translatedFrom - translatedTo);
- pxPosMin = Math.min(translatedFrom, translatedTo);
- pxPosMax = Math.max(translatedFrom, translatedTo);
- if (axis.isXAxis) {
- clipAttr = {
- x: inverted ? pxPosMax : pxPosMin,
- y: 0,
- width: pxRange,
- height: chartSizeMax
- };
- if (!horiz) {
- clipAttr.x = chart.plotHeight - clipAttr.x;
- }
- }
- else {
- clipAttr = {
- x: 0,
- y: inverted ? pxPosMax : pxPosMin,
- width: chartSizeMax,
- height: pxRange
- };
- if (horiz) {
- clipAttr.y = chart.plotWidth - clipAttr.y;
- }
- }
- // VML SUPPPORT
- if (inverted && renderer.isVML) {
- if (axis.isXAxis) {
- clipAttr = {
- x: 0,
- y: reversed ? pxPosMin : pxPosMax,
- height: clipAttr.width,
- width: chart.chartWidth
- };
- }
- else {
- clipAttr = {
- x: (clipAttr.y -
- chart.plotLeft -
- chart.spacingBox.x),
- y: 0,
- width: clipAttr.height,
- height: chart.chartHeight
- };
- }
- }
- // END OF VML SUPPORT
- if (clips[i]) {
- clips[i].animate(clipAttr);
- }
- else {
- clips[i] = renderer.clipRect(clipAttr);
- }
- // when no data, graph zone is not applied and after setData
- // clip was ignored. As a result, it should be applied each
- // time.
- zoneArea = series['zone-area-' + i];
- zoneGraph = series['zone-graph-' + i];
- if (graph && zoneGraph) {
- zoneGraph.clip(clips[i]);
- }
- if (area && zoneArea) {
- zoneArea.clip(clips[i]);
- }
- // if this zone extends out of the axis, ignore the others
- ignoreZones = threshold.value > extremes.max;
- // Clear translatedTo for indicators
- if (series.resetZones && translatedTo === 0) {
- translatedTo = void 0;
- }
- });
- this.clips = clips;
- }
- else if (series.visible) {
- // If zones were removed, restore graph and area
- if (graph) {
- graph.show(true);
- }
- if (area) {
- area.show(true);
- }
- }
- },
- /**
- * Initialize and perform group inversion on series.group and
- * series.markerGroup.
- *
- * @private
- * @function Highcharts.Series#invertGroups
- * @param {boolean} [inverted]
- * @return {void}
- */
- invertGroups: function (inverted) {
- var series = this,
- chart = series.chart;
- /**
- * @private
- */
- function setInvert() {
- ['group', 'markerGroup'].forEach(function (groupName) {
- if (series[groupName]) {
- // VML/HTML needs explicit attributes for flipping
- if (chart.renderer.isVML) {
- series[groupName].attr({
- width: series.yAxis.len,
- height: series.xAxis.len
- });
- }
- series[groupName].width = series.yAxis.len;
- series[groupName].height = series.xAxis.len;
- // If inverted polar, don't invert series group
- series[groupName].invert(series.isRadialSeries ? false : inverted);
- }
- });
- }
- // Pie, go away (#1736)
- if (!series.xAxis) {
- return;
- }
- // A fixed size is needed for inversion to work
- series.eventsToUnbind.push(addEvent(chart, 'resize', setInvert));
- // Do it now
- setInvert();
- // On subsequent render and redraw, just do setInvert without
- // setting up events again
- series.invertGroups = setInvert;
- },
- /**
- * General abstraction for creating plot groups like series.group,
- * series.dataLabelsGroup and series.markerGroup. On subsequent calls,
- * the group will only be adjusted to the updated plot size.
- *
- * @private
- * @function Highcharts.Series#plotGroup
- * @param {string} prop
- * @param {string} name
- * @param {string} visibility
- * @param {number} [zIndex]
- * @param {Highcharts.SVGElement} [parent]
- * @return {Highcharts.SVGElement}
- */
- plotGroup: function (prop, name, visibility, zIndex, parent) {
- var group = this[prop],
- isNew = !group,
- attrs = {
- visibility: visibility,
- zIndex: zIndex || 0.1 // IE8 and pointer logic use this
- };
- // Avoid setting undefined opacity, or in styled mode
- if (typeof this.opacity !== 'undefined' &&
- !this.chart.styledMode && this.state !== 'inactive' // #13719
- ) {
- attrs.opacity = this.opacity;
- }
- // Generate it on first call
- if (isNew) {
- this[prop] = group = this.chart.renderer
- .g()
- .add(parent);
- }
- // Add the class names, and replace existing ones as response to
- // Series.update (#6660)
- group.addClass(('highcharts-' + name +
- ' highcharts-series-' + this.index +
- ' highcharts-' + this.type + '-series ' +
- (defined(this.colorIndex) ?
- 'highcharts-color-' + this.colorIndex + ' ' :
- '') +
- (this.options.className || '') +
- (group.hasClass('highcharts-tracker') ?
- ' highcharts-tracker' :
- '')), true);
- // Place it on first and subsequent (redraw) calls
- group.attr(attrs)[isNew ? 'attr' : 'animate'](this.getPlotBox());
- return group;
- },
- /**
- * Get the translation and scale for the plot area of this series.
- *
- * @function Highcharts.Series#getPlotBox
- *
- * @return {Highcharts.SeriesPlotBoxObject}
- */
- getPlotBox: function () {
- var chart = this.chart,
- xAxis = this.xAxis,
- yAxis = this.yAxis;
- // Swap axes for inverted (#2339)
- if (chart.inverted) {
- xAxis = yAxis;
- yAxis = this.xAxis;
- }
- return {
- translateX: xAxis ? xAxis.left : chart.plotLeft,
- translateY: yAxis ? yAxis.top : chart.plotTop,
- scaleX: 1,
- scaleY: 1
- };
- },
- /**
- * Removes the event handlers attached previously with addEvents.
- *
- * @private
- * @function Highcharts.Series#removeEvents
- * @param {boolean} [keepEventsForUpdate]
- * @return {void}
- */
- removeEvents: function (keepEventsForUpdate) {
- var series = this;
- if (!keepEventsForUpdate) {
- // remove all events
- removeEvent(series);
- }
- else if (series.eventsToUnbind.length) {
- // remove only internal events for proper update
- // #12355 - solves problem with multiple destroy events
- series.eventsToUnbind.forEach(function (unbind) {
- unbind();
- });
- series.eventsToUnbind.length = 0;
- }
- },
- /**
- * Render the graph and markers. Called internally when first rendering
- * and later when redrawing the chart. This function can be extended in
- * plugins, but normally shouldn't be called directly.
- *
- * @function Highcharts.Series#render
- *
- * @return {void}
- *
- * @fires Highcharts.Series#event:afterRender
- */
- render: function () {
- var series = this,
- chart = series.chart,
- group,
- options = series.options,
- animOptions = animObject(options.animation),
- // Animation doesn't work in IE8 quirks when the group div is
- // hidden, and looks bad in other oldIE
- animDuration = (!series.finishedAnimating &&
- chart.renderer.isSVG &&
- animOptions.duration),
- visibility = series.visible ? 'inherit' : 'hidden', // #2597
- zIndex = options.zIndex,
- hasRendered = series.hasRendered,
- chartSeriesGroup = chart.seriesGroup,
- inverted = chart.inverted;
- fireEvent(this, 'render');
- // the group
- group = series.plotGroup('group', 'series', visibility, zIndex, chartSeriesGroup);
- series.markerGroup = series.plotGroup('markerGroup', 'markers', visibility, zIndex, chartSeriesGroup);
- // initiate the animation
- if (animDuration && series.animate) {
- series.animate(true);
- }
- // SVGRenderer needs to know this before drawing elements (#1089,
- // #1795)
- group.inverted = series.isCartesian || series.invertable ?
- inverted : false;
- // Draw the graph if any
- if (series.drawGraph) {
- series.drawGraph();
- series.applyZones();
- }
- // Draw the points
- if (series.visible) {
- series.drawPoints();
- }
- /* series.points.forEach(function (point) {
- if (point.redraw) {
- point.redraw();
- }
- }); */
- // Draw the data labels
- if (series.drawDataLabels) {
- series.drawDataLabels();
- }
- // In pie charts, slices are added to the DOM, but actual rendering
- // is postponed until labels reserved their space
- if (series.redrawPoints) {
- series.redrawPoints();
- }
- // draw the mouse tracking area
- if (series.drawTracker &&
- series.options.enableMouseTracking !== false) {
- series.drawTracker();
- }
- // Handle inverted series and tracker groups
- series.invertGroups(inverted);
- // Initial clipping, must be defined after inverting groups for VML.
- // Applies to columns etc. (#3839).
- if (options.clip !== false &&
- !series.sharedClipKey &&
- !hasRendered) {
- group.clip(chart.clipRect);
- }
- // Run the animation
- if (animDuration && series.animate) {
- series.animate();
- }
- // Call the afterAnimate function on animation complete (but don't
- // overwrite the animation.complete option which should be available
- // to the user).
- if (!hasRendered) {
- // Additional time if defer is defined before afterAnimate
- // will be triggered
- if (animDuration && animOptions.defer) {
- animDuration += animOptions.defer;
- }
- series.animationTimeout = syncTimeout(function () {
- series.afterAnimate();
- }, animDuration || 0);
- }
- // Means data is in accordance with what you see
- series.isDirty = false;
- // (See #322) series.isDirty = series.isDirtyData = false; // means
- // data is in accordance with what you see
- series.hasRendered = true;
- fireEvent(series, 'afterRender');
- },
- /**
- * Redraw the series. This function is called internally from
- * `chart.redraw` and normally shouldn't be called directly.
- *
- * @private
- * @function Highcharts.Series#redraw
- * @return {void}
- */
- redraw: function () {
- var series = this,
- chart = series.chart,
- // cache it here as it is set to false in render, but used after
- wasDirty = series.isDirty || series.isDirtyData,
- group = series.group,
- xAxis = series.xAxis,
- yAxis = series.yAxis;
- // reposition on resize
- if (group) {
- if (chart.inverted) {
- group.attr({
- width: chart.plotWidth,
- height: chart.plotHeight
- });
- }
- group.animate({
- translateX: pick(xAxis && xAxis.left, chart.plotLeft),
- translateY: pick(yAxis && yAxis.top, chart.plotTop)
- });
- }
- series.translate();
- series.render();
- if (wasDirty) { // #3868, #3945
- delete this.kdTree;
- }
- },
- kdAxisArray: ['clientX', 'plotY'],
- /**
- * @private
- * @function Highcharts.Series#searchPoint
- * @param {Highcharts.PointerEventObject} e
- * @param {boolean} [compareX]
- * @return {Highcharts.Point}
- */
- searchPoint: function (e, compareX) {
- var series = this,
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- inverted = series.chart.inverted;
- return this.searchKDTree({
- clientX: inverted ?
- xAxis.len - e.chartY + xAxis.pos :
- e.chartX - xAxis.pos,
- plotY: inverted ?
- yAxis.len - e.chartX + yAxis.pos :
- e.chartY - yAxis.pos
- }, compareX, e);
- },
- /**
- * Build the k-d-tree that is used by mouse and touch interaction to get
- * the closest point. Line-like series typically have a one-dimensional
- * tree where points are searched along the X axis, while scatter-like
- * series typically search in two dimensions, X and Y.
- *
- * @private
- * @function Highcharts.Series#buildKDTree
- * @param {Highcharts.PointerEventObject} [e]
- * @return {void}
- */
- buildKDTree: function (e) {
- // Prevent multiple k-d-trees from being built simultaneously
- // (#6235)
- this.buildingKdTree = true;
- var series = this,
- dimensions = series.options.findNearestPointBy
- .indexOf('y') > -1 ? 2 : 1;
- /**
- * Internal function
- * @private
- */
- function _kdtree(points, depth, dimensions) {
- var axis,
- median,
- length = points && points.length;
- if (length) {
- // alternate between the axis
- axis = series.kdAxisArray[depth % dimensions];
- // sort point array
- points.sort(function (a, b) {
- return a[axis] - b[axis];
- });
- median = Math.floor(length / 2);
- // build and return nod
- return {
- point: points[median],
- left: _kdtree(points.slice(0, median), depth + 1, dimensions),
- right: _kdtree(points.slice(median + 1), depth + 1, dimensions)
- };
- }
- }
- /**
- * Start the recursive build process with a clone of the points
- * array and null points filtered out. (#3873)
- * @private
- */
- function startRecursive() {
- series.kdTree = _kdtree(series.getValidPoints(null,
- // For line-type series restrict to plot area, but
- // column-type series not (#3916, #4511)
- !series.directTouch), dimensions, dimensions);
- series.buildingKdTree = false;
- }
- delete series.kdTree;
- // For testing tooltips, don't build async. Also if touchstart, we
- // may be dealing with click events on mobile, so don't delay
- // (#6817).
- syncTimeout(startRecursive, series.options.kdNow || (e && e.type === 'touchstart') ? 0 : 1);
- },
- /**
- * @private
- * @function Highcharts.Series#searchKDTree
- * @param {Highcharts.KDPointSearchObject} point
- * @param {boolean} [compareX]
- * @param {Highcharts.PointerEventObject} [e]
- * @return {Highcharts.Point|undefined}
- */
- searchKDTree: function (point, compareX, e) {
- var series = this,
- kdX = this.kdAxisArray[0],
- kdY = this.kdAxisArray[1],
- kdComparer = compareX ? 'distX' : 'dist',
- kdDimensions = series.options.findNearestPointBy
- .indexOf('y') > -1 ? 2 : 1;
- /**
- * Set the one and two dimensional distance on the point object.
- * @private
- */
- function setDistance(p1, p2) {
- var x = (defined(p1[kdX]) &&
- defined(p2[kdX])) ?
- Math.pow(p1[kdX] - p2[kdX], 2) :
- null,
- y = (defined(p1[kdY]) &&
- defined(p2[kdY])) ?
- Math.pow(p1[kdY] - p2[kdY], 2) :
- null,
- r = (x || 0) + (y || 0);
- p2.dist = defined(r) ? Math.sqrt(r) : Number.MAX_VALUE;
- p2.distX = defined(x) ? Math.sqrt(x) : Number.MAX_VALUE;
- }
- /**
- * @private
- */
- function _search(search, tree, depth, dimensions) {
- var point = tree.point,
- axis = series.kdAxisArray[depth % dimensions],
- tdist,
- sideA,
- sideB,
- ret = point,
- nPoint1,
- nPoint2;
- setDistance(search, point);
- // Pick side based on distance to splitting point
- tdist = search[axis] - point[axis];
- sideA = tdist < 0 ? 'left' : 'right';
- sideB = tdist < 0 ? 'right' : 'left';
- // End of tree
- if (tree[sideA]) {
- nPoint1 = _search(search, tree[sideA], depth + 1, dimensions);
- ret = (nPoint1[kdComparer] <
- ret[kdComparer] ?
- nPoint1 :
- point);
- }
- if (tree[sideB]) {
- // compare distance to current best to splitting point to
- // decide wether to check side B or not
- if (Math.sqrt(tdist * tdist) < ret[kdComparer]) {
- nPoint2 = _search(search, tree[sideB], depth + 1, dimensions);
- ret = (nPoint2[kdComparer] <
- ret[kdComparer] ?
- nPoint2 :
- ret);
- }
- }
- return ret;
- }
- if (!this.kdTree && !this.buildingKdTree) {
- this.buildKDTree(e);
- }
- if (this.kdTree) {
- return _search(point, this.kdTree, kdDimensions, kdDimensions);
- }
- },
- /**
- * @private
- * @function Highcharts.Series#pointPlacementToXValue
- * @return {number}
- */
- pointPlacementToXValue: function () {
- var _a = this,
- _b = _a.options,
- pointPlacement = _b.pointPlacement,
- pointRange = _b.pointRange,
- axis = _a.xAxis;
- var factor = pointPlacement;
- // Point placement is relative to each series pointRange (#5889)
- if (factor === 'between') {
- factor = axis.reversed ? -0.5 : 0.5; // #11955
- }
- return isNumber(factor) ?
- factor * pick(pointRange, axis.pointRange) :
- 0;
- },
- /**
- * @private
- * @function Highcharts.Series#isPointInside
- * @param {Highcharts.Point} point
- * @return {boolean}
- */
- isPointInside: function (point) {
- var isInside = typeof point.plotY !== 'undefined' &&
- typeof point.plotX !== 'undefined' &&
- point.plotY >= 0 &&
- point.plotY <= this.yAxis.len && // #3519
- point.plotX >= 0 &&
- point.plotX <= this.xAxis.len;
- return isInside;
- }
- }); // end Series prototype
- /**
- * A line series displays information as a series of data points connected by
- * straight line segments.
- *
- * @sample {highcharts} highcharts/demo/line-basic/
- * Line chart
- * @sample {highstock} stock/demo/basic-line/
- * Line chart
- *
- * @extends plotOptions.series
- * @product highcharts highstock
- * @apioption plotOptions.line
- */
- /**
- * The SVG value used for the `stroke-linecap` and `stroke-linejoin`
- * of a line graph. Round means that lines are rounded in the ends and
- * bends.
- *
- * @type {Highcharts.SeriesLinecapValue}
- * @default round
- * @since 3.0.7
- * @apioption plotOptions.line.linecap
- */
- /**
- * A `line` series. If the [type](#series.line.type) option is not
- * specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.line
- * @excluding dataParser,dataURL
- * @product highcharts highstock
- * @apioption series.line
- */
- /**
- * An array of data points for the series. For the `line` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 1],
- * [1, 2],
- * [2, 8]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.line.turboThreshold),
- * this option is not available.
- * ```js
- * data: [{
- * x: 1,
- * y: 9,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 6,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * **Note:** In TypeScript you have to extend `PointOptionsObject` with an
- * additional declaration to allow custom data types:
- * ```ts
- * declare module `highcharts` {
- * interface PointOptionsObject {
- * custom: Record<string, (boolean|number|string)>;
- * }
- * }
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @declare Highcharts.PointOptionsObject
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @apioption series.line.data
- */
- /**
- * An additional, individual class name for the data point's graphic
- * representation.
- *
- * @type {string}
- * @since 5.0.0
- * @product highcharts gantt
- * @apioption series.line.data.className
- */
- /**
- * Individual color for the point. By default the color is pulled from
- * the global `colors` array.
- *
- * In styled mode, the `color` option doesn't take effect. Instead, use
- * `colorIndex`.
- *
- * @sample {highcharts} highcharts/point/color/
- * Mark the highest point
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock gantt
- * @apioption series.line.data.color
- */
- /**
- * A specific color index to use for the point, so its graphic representations
- * are given the class name `highcharts-color-{n}`. In styled mode this will
- * change the color of the graphic. In non-styled mode, the color by is set by
- * the `fill` attribute, so the change in class name won't have a visual effect
- * by default.
- *
- * @type {number}
- * @since 5.0.0
- * @product highcharts gantt
- * @apioption series.line.data.colorIndex
- */
- /**
- * A reserved subspace to store options and values for customized functionality.
- * Here you can add additional data for your own event callbacks and formatter
- * callbacks.
- *
- * @sample {highcharts} highcharts/point/custom/
- * Point and series with custom data
- *
- * @type {Highcharts.Dictionary<*>}
- * @apioption series.line.data.custom
- */
- /**
- * Individual data label for each point. The options are the same as
- * the ones for [plotOptions.series.dataLabels](
- * #plotOptions.series.dataLabels).
- *
- * @sample highcharts/point/datalabels/
- * Show a label for the last value
- *
- * @declare Highcharts.DataLabelsOptions
- * @extends plotOptions.line.dataLabels
- * @product highcharts highstock gantt
- * @apioption series.line.data.dataLabels
- */
- /**
- * A description of the point to add to the screen reader information
- * about the point.
- *
- * @type {string}
- * @since 5.0.0
- * @requires modules/accessibility
- * @apioption series.line.data.description
- */
- /**
- * An id for the point. This can be used after render time to get a
- * pointer to the point object through `chart.get()`.
- *
- * @sample {highcharts} highcharts/point/id/
- * Remove an id'd point
- *
- * @type {string}
- * @since 1.2.0
- * @product highcharts highstock gantt
- * @apioption series.line.data.id
- */
- /**
- * The rank for this point's data label in case of collision. If two
- * data labels are about to overlap, only the one with the highest `labelrank`
- * will be drawn.
- *
- * @type {number}
- * @apioption series.line.data.labelrank
- */
- /**
- * The name of the point as shown in the legend, tooltip, dataLabels, etc.
- *
- * @see [xAxis.uniqueNames](#xAxis.uniqueNames)
- *
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Point names
- *
- * @type {string}
- * @apioption series.line.data.name
- */
- /**
- * Whether the data point is selected initially.
- *
- * @type {boolean}
- * @default false
- * @product highcharts highstock gantt
- * @apioption series.line.data.selected
- */
- /**
- * The x value of the point. For datetime axes, the X value is the timestamp
- * in milliseconds since 1970.
- *
- * @type {number}
- * @product highcharts highstock
- * @apioption series.line.data.x
- */
- /**
- * The y value of the point.
- *
- * @type {number|null}
- * @product highcharts highstock
- * @apioption series.line.data.y
- */
- /**
- * The individual point events.
- *
- * @extends plotOptions.series.point.events
- * @product highcharts highstock gantt
- * @apioption series.line.data.events
- */
- /**
- * Options for the point markers of line-like series.
- *
- * @declare Highcharts.PointMarkerOptionsObject
- * @extends plotOptions.series.marker
- * @product highcharts highstock
- * @apioption series.line.data.marker
- */
- ''; // include precedent doclets in transpilat
- });
- _registerModule(_modules, 'Extensions/Stacking.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Axis/StackingAxis.js'], _modules['Core/Utilities.js']], function (Axis, Chart, H, StackingAxis, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var correctFloat = U.correctFloat,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- format = U.format,
- isNumber = U.isNumber,
- pick = U.pick;
- /**
- * Stack of data points
- *
- * @product highcharts
- *
- * @interface Highcharts.StackItemObject
- */ /**
- * Alignment settings
- * @name Highcharts.StackItemObject#alignOptions
- * @type {Highcharts.AlignObject}
- */ /**
- * Related axis
- * @name Highcharts.StackItemObject#axis
- * @type {Highcharts.Axis}
- */ /**
- * Cumulative value of the stacked data points
- * @name Highcharts.StackItemObject#cumulative
- * @type {number}
- */ /**
- * True if on the negative side
- * @name Highcharts.StackItemObject#isNegative
- * @type {boolean}
- */ /**
- * Related SVG element
- * @name Highcharts.StackItemObject#label
- * @type {Highcharts.SVGElement}
- */ /**
- * Related stack options
- * @name Highcharts.StackItemObject#options
- * @type {Highcharts.YAxisStackLabelsOptions}
- */ /**
- * Total value of the stacked data points
- * @name Highcharts.StackItemObject#total
- * @type {number}
- */ /**
- * Shared x value of the stack
- * @name Highcharts.StackItemObject#x
- * @type {number}
- */
- ''; // detached doclets above
- var Series = H.Series;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The class for stacks. Each stack, on a specific X value and either negative
- * or positive, has its own stack item.
- *
- * @private
- * @class
- * @name Highcharts.StackItem
- * @param {Highcharts.Axis} axis
- * @param {Highcharts.YAxisStackLabelsOptions} options
- * @param {boolean} isNegative
- * @param {number} x
- * @param {Highcharts.OptionsStackingValue} [stackOption]
- */
- var StackItem = /** @class */ (function () {
- function StackItem(axis, options, isNegative, x, stackOption) {
- var inverted = axis.chart.inverted;
- this.axis = axis;
- // Tells if the stack is negative
- this.isNegative = isNegative;
- // Save the options to be able to style the label
- this.options = options = options || {};
- // Save the x value to be able to position the label later
- this.x = x;
- // Initialize total value
- this.total = null;
- // This will keep each points' extremes stored by series.index and point
- // index
- this.points = {};
- this.hasValidPoints = false;
- // Save the stack option on the series configuration object,
- // and whether to treat it as percent
- this.stack = stackOption;
- this.leftCliff = 0;
- this.rightCliff = 0;
- // The align options and text align varies on whether the stack is
- // negative and if the chart is inverted or not.
- // First test the user supplied value, then use the dynamic.
- this.alignOptions = {
- align: options.align ||
- (inverted ? (isNegative ? 'left' : 'right') : 'center'),
- verticalAlign: options.verticalAlign ||
- (inverted ? 'middle' : (isNegative ? 'bottom' : 'top')),
- y: options.y,
- x: options.x
- };
- this.textAlign = options.textAlign ||
- (inverted ? (isNegative ? 'right' : 'left') : 'center');
- }
- /**
- * @private
- * @function Highcharts.StackItem#destroy
- */
- StackItem.prototype.destroy = function () {
- destroyObjectProperties(this, this.axis);
- };
- /**
- * Renders the stack total label and adds it to the stack label group.
- *
- * @private
- * @function Highcharts.StackItem#render
- * @param {Highcharts.SVGElement} group
- */
- StackItem.prototype.render = function (group) {
- var chart = this.axis.chart,
- options = this.options,
- formatOption = options.format,
- attr = {},
- str = formatOption ? // format the text in the label
- format(formatOption,
- this,
- chart) :
- options.formatter.call(this);
- // Change the text to reflect the new total and set visibility to hidden
- // in case the serie is hidden
- if (this.label) {
- this.label.attr({ text: str, visibility: 'hidden' });
- }
- else {
- // Create new label
- this.label = chart.renderer
- .label(str, null, null, options.shape, null, null, options.useHTML, false, 'stack-labels');
- attr = {
- r: options.borderRadius || 0,
- text: str,
- rotation: options.rotation,
- padding: pick(options.padding, 5),
- visibility: 'hidden' // hidden until setOffset is called
- };
- if (!chart.styledMode) {
- attr.fill = options.backgroundColor;
- attr.stroke = options.borderColor;
- attr['stroke-width'] = options.borderWidth;
- this.label.css(options.style);
- }
- this.label.attr(attr);
- if (!this.label.added) {
- this.label.add(group); // add to the labels-group
- }
- }
- // Rank it higher than data labels (#8742)
- this.label.labelrank = chart.plotHeight;
- };
- /**
- * Sets the offset that the stack has from the x value and repositions the
- * label.
- *
- * @private
- * @function Highcarts.StackItem#setOffset
- * @param {number} xOffset
- * @param {number} xWidth
- * @param {number} [boxBottom]
- * @param {number} [boxTop]
- * @param {number} [defaultX]
- */
- StackItem.prototype.setOffset = function (xOffset, xWidth, boxBottom, boxTop, defaultX) {
- var stackItem = this,
- axis = stackItem.axis,
- chart = axis.chart,
- // stack value translated mapped to chart coordinates
- y = axis.translate(axis.stacking.usePercentage ?
- 100 :
- (boxTop ?
- boxTop :
- stackItem.total), 0, 0, 0, 1),
- yZero = axis.translate(boxBottom ? boxBottom : 0), // stack origin
- // stack height:
- h = defined(y) && Math.abs(y - yZero),
- // x position:
- x = pick(defaultX,
- chart.xAxis[0].translate(stackItem.x)) +
- xOffset,
- stackBox = defined(y) && stackItem.getStackBox(chart,
- stackItem,
- x,
- y,
- xWidth,
- h,
- axis),
- label = stackItem.label,
- isNegative = stackItem.isNegative,
- isJustify = pick(stackItem.options.overflow, 'justify') === 'justify',
- textAlign = stackItem.textAlign,
- visible;
- if (label && stackBox) {
- var bBox = label.getBBox(),
- padding = label.padding,
- boxOffsetX,
- boxOffsetY;
- if (textAlign === 'left') {
- boxOffsetX = chart.inverted ? -padding : padding;
- }
- else if (textAlign === 'right') {
- boxOffsetX = bBox.width;
- }
- else {
- if (chart.inverted && textAlign === 'center') {
- boxOffsetX = bBox.width / 2;
- }
- else {
- boxOffsetX = chart.inverted ?
- (isNegative ? bBox.width + padding : -padding) : bBox.width / 2;
- }
- }
- boxOffsetY = chart.inverted ?
- bBox.height / 2 : (isNegative ? -padding : bBox.height);
- // Reset alignOptions property after justify #12337
- stackItem.alignOptions.x = pick(stackItem.options.x, 0);
- stackItem.alignOptions.y = pick(stackItem.options.y, 0);
- // Set the stackBox position
- stackBox.x -= boxOffsetX;
- stackBox.y -= boxOffsetY;
- // Align the label to the box
- label.align(stackItem.alignOptions, null, stackBox);
- // Check if label is inside the plotArea #12294
- if (chart.isInsidePlot(label.alignAttr.x + boxOffsetX - stackItem.alignOptions.x, label.alignAttr.y + boxOffsetY - stackItem.alignOptions.y)) {
- label.show();
- }
- else {
- // Move label away to avoid the overlapping issues
- label.alignAttr.y = -9999;
- isJustify = false;
- }
- if (isJustify) {
- // Justify stackLabel into the stackBox
- Series.prototype.justifyDataLabel.call(this.axis, label, stackItem.alignOptions, label.alignAttr, bBox, stackBox);
- }
- label.attr({
- x: label.alignAttr.x,
- y: label.alignAttr.y
- });
- if (pick(!isJustify && stackItem.options.crop, true)) {
- visible =
- isNumber(label.x) &&
- isNumber(label.y) &&
- chart.isInsidePlot(label.x - padding + label.width, label.y) &&
- chart.isInsidePlot(label.x + padding, label.y);
- if (!visible) {
- label.hide();
- }
- }
- }
- };
- /**
- * @private
- * @function Highcharts.StackItem#getStackBox
- *
- * @param {Highcharts.Chart} chart
- *
- * @param {Highcharts.StackItem} stackItem
- *
- * @param {number} x
- *
- * @param {number} y
- *
- * @param {number} xWidth
- *
- * @param {number} h
- *
- * @param {Highcharts.Axis} axis
- *
- * @return {Highcharts.BBoxObject}
- */
- StackItem.prototype.getStackBox = function (chart, stackItem, x, y, xWidth, h, axis) {
- var reversed = stackItem.axis.reversed,
- inverted = chart.inverted,
- axisPos = axis.height + axis.pos -
- (inverted ? chart.plotLeft : chart.plotTop),
- neg = (stackItem.isNegative && !reversed) ||
- (!stackItem.isNegative && reversed); // #4056
- return {
- x: inverted ? (neg ? y - axis.right : y - h + axis.pos - chart.plotLeft) :
- x + chart.xAxis[0].transB - chart.plotLeft,
- y: inverted ?
- axis.height - x - xWidth :
- (neg ?
- (axisPos - y - h) :
- axisPos - y),
- width: inverted ? h : xWidth,
- height: inverted ? xWidth : h
- };
- };
- return StackItem;
- }());
- /**
- * Generate stacks for each series and calculate stacks total values
- *
- * @private
- * @function Highcharts.Chart#getStacks
- */
- Chart.prototype.getStacks = function () {
- var chart = this,
- inverted = chart.inverted;
- // reset stacks for each yAxis
- chart.yAxis.forEach(function (axis) {
- if (axis.stacking && axis.stacking.stacks && axis.hasVisibleSeries) {
- axis.stacking.oldStacks = axis.stacking.stacks;
- }
- });
- chart.series.forEach(function (series) {
- var xAxisOptions = series.xAxis && series.xAxis.options || {};
- if (series.options.stacking &&
- (series.visible === true ||
- chart.options.chart.ignoreHiddenSeries === false)) {
- series.stackKey = [
- series.type,
- pick(series.options.stack, ''),
- inverted ? xAxisOptions.top : xAxisOptions.left,
- inverted ? xAxisOptions.height : xAxisOptions.width
- ].join(',');
- }
- });
- };
- // Stacking methods defined on the Axis prototype
- StackingAxis.compose(Axis);
- // Stacking methods defined for Series prototype
- /**
- * Set grouped points in a stack-like object. When `centerInCategory` is true,
- * and `stacking` is not enabled, we need a pseudo (horizontal) stack in order
- * to handle grouping of points within the same category.
- *
- * @private
- * @function Highcharts.Series#setStackedPoints
- * @return {void}
- */
- Series.prototype.setGroupedPoints = function () {
- if (this.options.centerInCategory &&
- (this.is('column') || this.is('columnrange')) &&
- // With stacking enabled, we already have stacks that we can compute
- // from
- !this.options.stacking &&
- // With only one series, we don't need to consider centerInCategory
- this.chart.series.length > 1) {
- Series.prototype.setStackedPoints.call(this, 'group');
- }
- };
- /**
- * Adds series' points value to corresponding stack
- *
- * @private
- * @function Highcharts.Series#setStackedPoints
- */
- Series.prototype.setStackedPoints = function (stackingParam) {
- var stacking = stackingParam || this.options.stacking;
- if (!stacking ||
- (this.visible !== true &&
- this.chart.options.chart.ignoreHiddenSeries !== false)) {
- return;
- }
- var series = this, xData = series.processedXData, yData = series.processedYData, stackedYData = [], yDataLength = yData.length, seriesOptions = series.options, threshold = seriesOptions.threshold, stackThreshold = pick(seriesOptions.startFromThreshold && threshold, 0), stackOption = seriesOptions.stack, stackKey = stackingParam ? series.type + "," + stacking : series.stackKey, negKey = '-' + stackKey, negStacks = series.negStacks, yAxis = series.yAxis, stacks = yAxis.stacking.stacks, oldStacks = yAxis.stacking.oldStacks, stackIndicator, isNegative, stack, other, key, pointKey, i, x, y;
- yAxis.stacking.stacksTouched += 1;
- // loop over the non-null y values and read them into a local array
- for (i = 0; i < yDataLength; i++) {
- x = xData[i];
- y = yData[i];
- stackIndicator = series.getStackIndicator(stackIndicator, x, series.index);
- pointKey = stackIndicator.key;
- // Read stacked values into a stack based on the x value,
- // the sign of y and the stack key. Stacking is also handled for null
- // values (#739)
- isNegative = negStacks && y < (stackThreshold ? 0 : threshold);
- key = isNegative ? negKey : stackKey;
- // Create empty object for this stack if it doesn't exist yet
- if (!stacks[key]) {
- stacks[key] =
- {};
- }
- // Initialize StackItem for this x
- if (!stacks[key][x]) {
- if (oldStacks[key] &&
- oldStacks[key][x]) {
- stacks[key][x] = oldStacks[key][x];
- stacks[key][x].total = null;
- }
- else {
- stacks[key][x] = new StackItem(yAxis, yAxis.options.stackLabels, isNegative, x, stackOption);
- }
- }
- // If the StackItem doesn't exist, create it first
- stack = stacks[key][x];
- if (y !== null) {
- stack.points[pointKey] = stack.points[series.index] =
- [pick(stack.cumulative, stackThreshold)];
- // Record the base of the stack
- if (!defined(stack.cumulative)) {
- stack.base = pointKey;
- }
- stack.touched = yAxis.stacking.stacksTouched;
- // In area charts, if there are multiple points on the same X value,
- // let the area fill the full span of those points
- if (stackIndicator.index > 0 && series.singleStacks === false) {
- stack.points[pointKey][0] =
- stack.points[series.index + ',' + x + ',0'][0];
- }
- // When updating to null, reset the point stack (#7493)
- }
- else {
- stack.points[pointKey] = stack.points[series.index] =
- null;
- }
- // Add value to the stack total
- if (stacking === 'percent') {
- // Percent stacked column, totals are the same for the positive and
- // negative stacks
- other = isNegative ? stackKey : negKey;
- if (negStacks && stacks[other] && stacks[other][x]) {
- other = stacks[other][x];
- stack.total = other.total =
- Math.max(other.total, stack.total) +
- Math.abs(y) ||
- 0;
- // Percent stacked areas
- }
- else {
- stack.total =
- correctFloat(stack.total + (Math.abs(y) || 0));
- }
- }
- else if (stacking === 'group') {
- // In this stack, the total is the number of valid points
- if (y !== null) {
- stack.total = (stack.total || 0) + 1;
- }
- }
- else {
- stack.total = correctFloat(stack.total + (y || 0));
- }
- if (stacking === 'group') {
- // This point's index within the stack, pushed to stack.points[1]
- stack.cumulative = (stack.total || 1) - 1;
- }
- else {
- stack.cumulative =
- pick(stack.cumulative, stackThreshold) + (y || 0);
- }
- if (y !== null) {
- stack.points[pointKey].push(stack.cumulative);
- stackedYData[i] = stack.cumulative;
- stack.hasValidPoints = true;
- }
- }
- if (stacking === 'percent') {
- yAxis.stacking.usePercentage = true;
- }
- if (stacking !== 'group') {
- this.stackedYData = stackedYData; // To be used in getExtremes
- }
- // Reset old stacks
- yAxis.stacking.oldStacks = {};
- };
- /**
- * Iterate over all stacks and compute the absolute values to percent
- *
- * @private
- * @function Highcharts.Series#modifyStacks
- */
- Series.prototype.modifyStacks = function () {
- var series = this,
- yAxis = series.yAxis,
- stackKey = series.stackKey,
- stacks = yAxis.stacking.stacks,
- processedXData = series.processedXData,
- stackIndicator,
- stacking = series.options.stacking;
- if (series[stacking + 'Stacker']) { // Modifier function exists
- [stackKey, '-' + stackKey].forEach(function (key) {
- var i = processedXData.length,
- x,
- stack,
- pointExtremes;
- while (i--) {
- x = processedXData[i];
- stackIndicator = series.getStackIndicator(stackIndicator, x, series.index, key);
- stack = stacks[key] && stacks[key][x];
- pointExtremes =
- stack && stack.points[stackIndicator.key];
- if (pointExtremes) {
- series[stacking + 'Stacker'](pointExtremes, stack, i);
- }
- }
- });
- }
- };
- /**
- * Modifier function for percent stacks. Blows up the stack to 100%.
- *
- * @private
- * @function Highcharts.Series#percentStacker
- * @param {Array<number>} pointExtremes
- * @param {Highcharts.StackItem} stack
- * @param {number} i
- */
- Series.prototype.percentStacker = function (pointExtremes, stack, i) {
- var totalFactor = stack.total ? 100 / stack.total : 0;
- // Y bottom value
- pointExtremes[0] = correctFloat(pointExtremes[0] * totalFactor);
- // Y value
- pointExtremes[1] = correctFloat(pointExtremes[1] * totalFactor);
- this.stackedYData[i] = pointExtremes[1];
- };
- /**
- * Get stack indicator, according to it's x-value, to determine points with the
- * same x-value
- *
- * @private
- * @function Highcharts.Series#getStackIndicator
- * @param {Highcharts.StackItemIndicatorObject|undefined} stackIndicator
- * @param {number} x
- * @param {number} index
- * @param {string} [key]
- * @return {Highcharts.StackItemIndicatorObject}
- */
- Series.prototype.getStackIndicator = function (stackIndicator, x, index, key) {
- // Update stack indicator, when:
- // first point in a stack || x changed || stack type (negative vs positive)
- // changed:
- if (!defined(stackIndicator) ||
- stackIndicator.x !== x ||
- (key && stackIndicator.key !== key)) {
- stackIndicator = {
- x: x,
- index: 0,
- key: key
- };
- }
- else {
- (stackIndicator).index++;
- }
- stackIndicator.key =
- [index, x, stackIndicator.index].join(',');
- return stackIndicator;
- };
- H.StackItem = StackItem;
- return H.StackItem;
- });
- _registerModule(_modules, 'Core/Dynamics.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Time.js'], _modules['Core/Utilities.js']], function (Axis, Chart, H, O, Point, Time, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var time = O.time;
- var addEvent = U.addEvent,
- animate = U.animate,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- erase = U.erase,
- error = U.error,
- extend = U.extend,
- fireEvent = U.fireEvent,
- isArray = U.isArray,
- isNumber = U.isNumber,
- isObject = U.isObject,
- isString = U.isString,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- relativeLength = U.relativeLength,
- setAnimation = U.setAnimation,
- splat = U.splat;
- var Series = H.Series,
- seriesTypes = H.seriesTypes;
- /* eslint-disable valid-jsdoc */
- /**
- * Remove settings that have not changed, to avoid unnecessary rendering or
- * computing (#9197).
- * @private
- */
- H.cleanRecursively = function (newer, older) {
- var result = {};
- objectEach(newer, function (val, key) {
- var ob;
- // Dive into objects (except DOM nodes)
- if (isObject(newer[key], true) &&
- !newer.nodeType && // #10044
- older[key]) {
- ob = H.cleanRecursively(newer[key], older[key]);
- if (Object.keys(ob).length) {
- result[key] = ob;
- }
- // Arrays, primitives and DOM nodes are copied directly
- }
- else if (isObject(newer[key]) ||
- newer[key] !== older[key]) {
- result[key] = newer[key];
- }
- });
- return result;
- };
- // Extend the Chart prototype for dynamic methods
- extend(Chart.prototype, /** @lends Highcharts.Chart.prototype */ {
- /**
- * Add a series to the chart after render time. Note that this method should
- * never be used when adding data synchronously at chart render time, as it
- * adds expense to the calculations and rendering. When adding data at the
- * same time as the chart is initialized, add the series as a configuration
- * option instead. With multiple axes, the `offset` is dynamically adjusted.
- *
- * @sample highcharts/members/chart-addseries/
- * Add a series from a button
- * @sample stock/members/chart-addseries/
- * Add a series in Highstock
- *
- * @function Highcharts.Chart#addSeries
- *
- * @param {Highcharts.SeriesOptionsType} options
- * The config options for the series.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after adding.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @return {Highcharts.Series}
- * The newly created series object.
- *
- * @fires Highcharts.Chart#event:addSeries
- * @fires Highcharts.Chart#event:afterAddSeries
- */
- addSeries: function (options, redraw, animation) {
- var series,
- chart = this;
- if (options) { // <- not necessary
- redraw = pick(redraw, true); // defaults to true
- fireEvent(chart, 'addSeries', { options: options }, function () {
- series = chart.initSeries(options);
- chart.isDirtyLegend = true;
- chart.linkSeries();
- if (series.enabledDataSorting) {
- // We need to call `setData` after `linkSeries`
- series.setData(options.data, false);
- }
- fireEvent(chart, 'afterAddSeries', { series: series });
- if (redraw) {
- chart.redraw(animation);
- }
- });
- }
- return series;
- },
- /**
- * Add an axis to the chart after render time. Note that this method should
- * never be used when adding data synchronously at chart render time, as it
- * adds expense to the calculations and rendering. When adding data at the
- * same time as the chart is initialized, add the axis as a configuration
- * option instead.
- *
- * @sample highcharts/members/chart-addaxis/
- * Add and remove axes
- *
- * @function Highcharts.Chart#addAxis
- *
- * @param {Highcharts.AxisOptions} options
- * The axis options.
- *
- * @param {boolean} [isX=false]
- * Whether it is an X axis or a value axis.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after adding.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Whether and how to apply animation in the redraw.
- *
- * @return {Highcharts.Axis}
- * The newly generated Axis object.
- */
- addAxis: function (options, isX, redraw, animation) {
- return this.createAxis(isX ? 'xAxis' : 'yAxis', { axis: options, redraw: redraw, animation: animation });
- },
- /**
- * Add a color axis to the chart after render time. Note that this method
- * should never be used when adding data synchronously at chart render time,
- * as it adds expense to the calculations and rendering. When adding data at
- * the same time as the chart is initialized, add the axis as a
- * configuration option instead.
- *
- * @sample highcharts/members/chart-addaxis/
- * Add and remove axes
- *
- * @function Highcharts.Chart#addColorAxis
- *
- * @param {Highcharts.ColorAxisOptions} options
- * The axis options.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after adding.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Whether and how to apply animation in the redraw.
- *
- * @return {Highcharts.ColorAxis}
- * The newly generated Axis object.
- */
- addColorAxis: function (options, redraw, animation) {
- return this.createAxis('colorAxis', { axis: options, redraw: redraw, animation: animation });
- },
- /**
- * Factory for creating different axis types.
- *
- * @private
- * @function Highcharts.Chart#createAxis
- *
- * @param {string} type
- * An axis type.
- *
- * @param {...Array<*>} arguments
- * All arguments for the constructor.
- *
- * @return {Highcharts.Axis | Highcharts.ColorAxis}
- * The newly generated Axis object.
- */
- createAxis: function (type, options) {
- var chartOptions = this.options,
- isColorAxis = type === 'colorAxis',
- axisOptions = options.axis,
- redraw = options.redraw,
- animation = options.animation,
- userOptions = merge(axisOptions, {
- index: this[type].length,
- isX: type === 'xAxis'
- }),
- axis;
- if (isColorAxis) {
- axis = new H.ColorAxis(this, userOptions);
- }
- else {
- axis = new Axis(this, userOptions);
- }
- // Push the new axis options to the chart options
- chartOptions[type] = splat(chartOptions[type] || {});
- chartOptions[type].push(userOptions);
- if (isColorAxis) {
- this.isDirtyLegend = true;
- // Clear before 'bindAxes' (#11924)
- this.axes.forEach(function (axis) {
- axis.series = [];
- });
- this.series.forEach(function (series) {
- series.bindAxes();
- series.isDirtyData = true;
- });
- }
- if (pick(redraw, true)) {
- this.redraw(animation);
- }
- return axis;
- },
- /**
- * Dim the chart and show a loading text or symbol. Options for the loading
- * screen are defined in {@link
- * https://api.highcharts.com/highcharts/loading|the loading options}.
- *
- * @sample highcharts/members/chart-hideloading/
- * Show and hide loading from a button
- * @sample highcharts/members/chart-showloading/
- * Apply different text labels
- * @sample stock/members/chart-show-hide-loading/
- * Toggle loading in Highstock
- *
- * @function Highcharts.Chart#showLoading
- *
- * @param {string} [str]
- * An optional text to show in the loading label instead of the
- * default one. The default text is set in
- * [lang.loading](https://api.highcharts.com/highcharts/lang.loading).
- */
- showLoading: function (str) {
- var chart = this,
- options = chart.options,
- loadingDiv = chart.loadingDiv,
- loadingOptions = options.loading,
- setLoadingSize = function () {
- if (loadingDiv) {
- css(loadingDiv, {
- left: chart.plotLeft + 'px',
- top: chart.plotTop + 'px',
- width: chart.plotWidth + 'px',
- height: chart.plotHeight + 'px'
- });
- }
- };
- // create the layer at the first call
- if (!loadingDiv) {
- chart.loadingDiv = loadingDiv = createElement('div', {
- className: 'highcharts-loading highcharts-loading-hidden'
- }, null, chart.container);
- chart.loadingSpan = createElement('span', { className: 'highcharts-loading-inner' }, null, loadingDiv);
- addEvent(chart, 'redraw', setLoadingSize); // #1080
- }
- loadingDiv.className = 'highcharts-loading';
- // Update text
- chart.loadingSpan.innerHTML =
- pick(str, options.lang.loading, '');
- if (!chart.styledMode) {
- // Update visuals
- css(loadingDiv, extend(loadingOptions.style, {
- zIndex: 10
- }));
- css(chart.loadingSpan, loadingOptions.labelStyle);
- // Show it
- if (!chart.loadingShown) {
- css(loadingDiv, {
- opacity: 0,
- display: ''
- });
- animate(loadingDiv, {
- opacity: loadingOptions.style.opacity || 0.5
- }, {
- duration: loadingOptions.showDuration || 0
- });
- }
- }
- chart.loadingShown = true;
- setLoadingSize();
- },
- /**
- * Hide the loading layer.
- *
- * @see Highcharts.Chart#showLoading
- *
- * @sample highcharts/members/chart-hideloading/
- * Show and hide loading from a button
- * @sample stock/members/chart-show-hide-loading/
- * Toggle loading in Highstock
- *
- * @function Highcharts.Chart#hideLoading
- */
- hideLoading: function () {
- var options = this.options,
- loadingDiv = this.loadingDiv;
- if (loadingDiv) {
- loadingDiv.className =
- 'highcharts-loading highcharts-loading-hidden';
- if (!this.styledMode) {
- animate(loadingDiv, {
- opacity: 0
- }, {
- duration: options.loading.hideDuration || 100,
- complete: function () {
- css(loadingDiv, { display: 'none' });
- }
- });
- }
- }
- this.loadingShown = false;
- },
- /**
- * These properties cause isDirtyBox to be set to true when updating. Can be
- * extended from plugins.
- */
- propsRequireDirtyBox: [
- 'backgroundColor',
- 'borderColor',
- 'borderWidth',
- 'borderRadius',
- 'plotBackgroundColor',
- 'plotBackgroundImage',
- 'plotBorderColor',
- 'plotBorderWidth',
- 'plotShadow',
- 'shadow'
- ],
- /**
- * These properties require a full reflow of chart elements, best
- * implemented through running `Chart.setSize` internally (#8190).
- * @type {Array}
- */
- propsRequireReflow: [
- 'margin',
- 'marginTop',
- 'marginRight',
- 'marginBottom',
- 'marginLeft',
- 'spacing',
- 'spacingTop',
- 'spacingRight',
- 'spacingBottom',
- 'spacingLeft'
- ],
- /**
- * These properties cause all series to be updated when updating. Can be
- * extended from plugins.
- */
- propsRequireUpdateSeries: [
- 'chart.inverted',
- 'chart.polar',
- 'chart.ignoreHiddenSeries',
- 'chart.type',
- 'colors',
- 'plotOptions',
- 'time',
- 'tooltip'
- ],
- /**
- * These collections (arrays) implement update() methods with support for
- * one-to-one option.
- */
- collectionsWithUpdate: [
- 'xAxis',
- 'yAxis',
- 'zAxis',
- 'series'
- ],
- /**
- * A generic function to update any element of the chart. Elements can be
- * enabled and disabled, moved, re-styled, re-formatted etc.
- *
- * A special case is configuration objects that take arrays, for example
- * [xAxis](https://api.highcharts.com/highcharts/xAxis),
- * [yAxis](https://api.highcharts.com/highcharts/yAxis) or
- * [series](https://api.highcharts.com/highcharts/series). For these
- * collections, an `id` option is used to map the new option set to an
- * existing object. If an existing object of the same id is not found, the
- * corresponding item is updated. So for example, running `chart.update`
- * with a series item without an id, will cause the existing chart's series
- * with the same index in the series array to be updated. When the
- * `oneToOne` parameter is true, `chart.update` will also take care of
- * adding and removing items from the collection. Read more under the
- * parameter description below.
- *
- * Note that when changing series data, `chart.update` may mutate the passed
- * data options.
- *
- * See also the
- * [responsive option set](https://api.highcharts.com/highcharts/responsive).
- * Switching between `responsive.rules` basically runs `chart.update` under
- * the hood.
- *
- * @sample highcharts/members/chart-update/
- * Update chart geometry
- *
- * @function Highcharts.Chart#update
- *
- * @param {Highcharts.Options} options
- * A configuration object for the new chart options.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart.
- *
- * @param {boolean} [oneToOne=false]
- * When `true`, the `series`, `xAxis`, `yAxis` and `annotations`
- * collections will be updated one to one, and items will be either
- * added or removed to match the new updated options. For example,
- * if the chart has two series and we call `chart.update` with a
- * configuration containing three series, one will be added. If we
- * call `chart.update` with one series, one will be removed. Setting
- * an empty `series` array will remove all series, but leaving out
- * the`series` property will leave all series untouched. If the
- * series have id's, the new series options will be matched by id,
- * and the remaining ones removed.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @fires Highcharts.Chart#event:update
- * @fires Highcharts.Chart#event:afterUpdate
- */
- update: function (options, redraw, oneToOne, animation) {
- var chart = this,
- adders = {
- credits: 'addCredits',
- title: 'setTitle',
- subtitle: 'setSubtitle',
- caption: 'setCaption'
- },
- optionsChart,
- updateAllAxes,
- updateAllSeries,
- newWidth,
- newHeight,
- runSetSize,
- isResponsiveOptions = options.isResponsiveOptions,
- itemsForRemoval = [];
- fireEvent(chart, 'update', { options: options });
- // If there are responsive rules in action, undo the responsive rules
- // before we apply the updated options and replay the responsive rules
- // on top from the chart.redraw function (#9617).
- if (!isResponsiveOptions) {
- chart.setResponsive(false, true);
- }
- options = H.cleanRecursively(options, chart.options);
- merge(true, chart.userOptions, options);
- // If the top-level chart option is present, some special updates are
- // required
- optionsChart = options.chart;
- if (optionsChart) {
- merge(true, chart.options.chart, optionsChart);
- // Setter function
- if ('className' in optionsChart) {
- chart.setClassName(optionsChart.className);
- }
- if ('reflow' in optionsChart) {
- chart.setReflow(optionsChart.reflow);
- }
- if ('inverted' in optionsChart ||
- 'polar' in optionsChart ||
- 'type' in optionsChart) {
- // Parse options.chart.inverted and options.chart.polar together
- // with the available series.
- chart.propFromSeries();
- updateAllAxes = true;
- }
- if ('alignTicks' in optionsChart) { // #6452
- updateAllAxes = true;
- }
- objectEach(optionsChart, function (val, key) {
- if (chart.propsRequireUpdateSeries.indexOf('chart.' + key) !==
- -1) {
- updateAllSeries = true;
- }
- // Only dirty box
- if (chart.propsRequireDirtyBox.indexOf(key) !== -1) {
- chart.isDirtyBox = true;
- }
- // Chart setSize
- if (chart.propsRequireReflow.indexOf(key) !== -1) {
- if (isResponsiveOptions) {
- chart.isDirtyBox = true;
- }
- else {
- runSetSize = true;
- }
- }
- });
- if (!chart.styledMode && 'style' in optionsChart) {
- chart.renderer.setStyle(optionsChart.style);
- }
- }
- // Moved up, because tooltip needs updated plotOptions (#6218)
- if (!chart.styledMode && options.colors) {
- this.options.colors = options.colors;
- }
- if (options.plotOptions) {
- merge(true, this.options.plotOptions, options.plotOptions);
- }
- // Maintaining legacy global time. If the chart is instanciated first
- // with global time, then updated with time options, we need to create a
- // new Time instance to avoid mutating the global time (#10536).
- if (options.time && this.time === time) {
- this.time = new Time(options.time);
- }
- // Some option stuctures correspond one-to-one to chart objects that
- // have update methods, for example
- // options.credits => chart.credits
- // options.legend => chart.legend
- // options.title => chart.title
- // options.tooltip => chart.tooltip
- // options.subtitle => chart.subtitle
- // options.mapNavigation => chart.mapNavigation
- // options.navigator => chart.navigator
- // options.scrollbar => chart.scrollbar
- objectEach(options, function (val, key) {
- if (chart[key] &&
- typeof chart[key].update === 'function') {
- chart[key].update(val, false);
- // If a one-to-one object does not exist, look for an adder function
- }
- else if (typeof chart[adders[key]] === 'function') {
- chart[adders[key]](val);
- }
- if (key !== 'chart' &&
- chart.propsRequireUpdateSeries.indexOf(key) !== -1) {
- updateAllSeries = true;
- }
- });
- // Setters for collections. For axes and series, each item is referred
- // by an id. If the id is not found, it defaults to the corresponding
- // item in the collection, so setting one series without an id, will
- // update the first series in the chart. Setting two series without
- // an id will update the first and the second respectively (#6019)
- // chart.update and responsive.
- this.collectionsWithUpdate.forEach(function (coll) {
- var indexMap;
- if (options[coll]) {
- // In stock charts, the navigator series are also part of the
- // chart.series array, but those series should not be handled
- // here (#8196).
- if (coll === 'series') {
- indexMap = [];
- chart[coll].forEach(function (s, i) {
- if (!s.options.isInternal) {
- indexMap.push(pick(s.options.index, i));
- }
- });
- }
- splat(options[coll]).forEach(function (newOptions, i) {
- var hasId = defined(newOptions.id);
- var item;
- // Match by id
- if (hasId) {
- item = chart.get(newOptions.id);
- }
- // No match by id found, match by index instead
- if (!item) {
- item = chart[coll][indexMap ? indexMap[i] : i];
- // Check if we grabbed an item with an exising but
- // different id (#13541)
- if (item && hasId && defined(item.options.id)) {
- item = void 0;
- }
- }
- if (item && item.coll === coll) {
- item.update(newOptions, false);
- if (oneToOne) {
- item.touched = true;
- }
- }
- // If oneToOne and no matching item is found, add one
- if (!item && oneToOne && chart.collectionsWithInit[coll]) {
- chart.collectionsWithInit[coll][0].apply(chart,
- // [newOptions, ...extraArguments, redraw=false]
- [
- newOptions
- ].concat(
- // Not all initializers require extra args
- chart.collectionsWithInit[coll][1] || []).concat([
- false
- ])).touched = true;
- }
- });
- // Add items for removal
- if (oneToOne) {
- chart[coll].forEach(function (item) {
- if (!item.touched && !item.options.isInternal) {
- itemsForRemoval.push(item);
- }
- else {
- delete item.touched;
- }
- });
- }
- }
- });
- itemsForRemoval.forEach(function (item) {
- if (item.remove) {
- item.remove(false);
- }
- });
- if (updateAllAxes) {
- chart.axes.forEach(function (axis) {
- axis.update({}, false);
- });
- }
- // Certain options require the whole series structure to be thrown away
- // and rebuilt
- if (updateAllSeries) {
- chart.getSeriesOrderByLinks().forEach(function (series) {
- // Avoid removed navigator series
- if (series.chart) {
- series.update({}, false);
- }
- }, this);
- }
- // For loading, just update the options, do not redraw
- if (options.loading) {
- merge(true, chart.options.loading, options.loading);
- }
- // Update size. Redraw is forced.
- newWidth = optionsChart && optionsChart.width;
- newHeight = optionsChart && optionsChart.height;
- if (isString(newHeight)) {
- newHeight = relativeLength(newHeight, newWidth || chart.chartWidth);
- }
- if (
- // In this case, run chart.setSize with newWidth and newHeight which
- // are undefined, only for reflowing chart elements because margin
- // or spacing has been set (#8190)
- runSetSize ||
- // In this case, the size is actually set
- (isNumber(newWidth) && newWidth !== chart.chartWidth) ||
- (isNumber(newHeight) && newHeight !== chart.chartHeight)) {
- chart.setSize(newWidth, newHeight, animation);
- }
- else if (pick(redraw, true)) {
- chart.redraw(animation);
- }
- fireEvent(chart, 'afterUpdate', {
- options: options,
- redraw: redraw,
- animation: animation
- });
- },
- /**
- * Shortcut to set the subtitle options. This can also be done from {@link
- * Chart#update} or {@link Chart#setTitle}.
- *
- * @function Highcharts.Chart#setSubtitle
- *
- * @param {Highcharts.SubtitleOptions} options
- * New subtitle options. The subtitle text itself is set by the
- * `options.text` property.
- */
- setSubtitle: function (options, redraw) {
- this.applyDescription('subtitle', options);
- this.layOutTitles(redraw);
- },
- /**
- * Set the caption options. This can also be done from {@link
- * Chart#update}.
- *
- * @function Highcharts.Chart#setCaption
- *
- * @param {Highcharts.CaptionOptions} options
- * New caption options. The caption text itself is set by the
- * `options.text` property.
- */
- setCaption: function (options, redraw) {
- this.applyDescription('caption', options);
- this.layOutTitles(redraw);
- }
- });
- /**
- * These collections (arrays) implement `Chart.addSomethig` method used in
- * chart.update() to create new object in the collection. Equivalent for
- * deleting is resolved by simple `Somethig.remove()`.
- *
- * Note: We need to define these references after initializers are bound to
- * chart's prototype.
- */
- Chart.prototype.collectionsWithInit = {
- // collectionName: [ initializingMethod, [extraArguments] ]
- xAxis: [Chart.prototype.addAxis, [true]],
- yAxis: [Chart.prototype.addAxis, [false]],
- series: [Chart.prototype.addSeries]
- };
- // extend the Point prototype for dynamic methods
- extend(Point.prototype, /** @lends Highcharts.Point.prototype */ {
- /**
- * Update point with new options (typically x/y data) and optionally redraw
- * the series.
- *
- * @sample highcharts/members/point-update-column/
- * Update column value
- * @sample highcharts/members/point-update-pie/
- * Update pie slice
- * @sample maps/members/point-update/
- * Update map area value in Highmaps
- *
- * @function Highcharts.Point#update
- *
- * @param {Highcharts.PointOptionsType} options
- * The point options. Point options are handled as described under
- * the `series.type.data` item for each series type. For example
- * for a line series, if options is a single number, the point will
- * be given that number as the marin y value. If it is an array, it
- * will be interpreted as x and y values respectively. If it is an
- * object, advanced options are applied.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the point is updated. If doing
- * more operations on the chart, it is best practice to set
- * `redraw` to false and call `chart.redraw()` after.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=true]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @return {void}
- *
- * @fires Highcharts.Point#event:update
- */
- update: function (options, redraw, animation, runEvent) {
- var point = this,
- series = point.series,
- graphic = point.graphic,
- i,
- chart = series.chart,
- seriesOptions = series.options;
- redraw = pick(redraw, true);
- /**
- * @private
- */
- function update() {
- point.applyOptions(options);
- // Update visuals, #4146
- // Handle dummy graphic elements for a11y, #12718
- var hasDummyGraphic = graphic && point.hasDummyGraphic;
- var shouldDestroyGraphic = point.y === null ? !hasDummyGraphic : hasDummyGraphic;
- if (graphic && shouldDestroyGraphic) {
- point.graphic = graphic.destroy();
- delete point.hasDummyGraphic;
- }
- if (isObject(options, true)) {
- // Destroy so we can get new elements
- if (graphic && graphic.element) {
- // "null" is also a valid symbol
- if (options &&
- options.marker &&
- typeof options.marker.symbol !== 'undefined') {
- point.graphic = graphic.destroy();
- }
- }
- if (options && options.dataLabels && point.dataLabel) {
- point.dataLabel = point.dataLabel.destroy(); // #2468
- }
- if (point.connector) {
- point.connector = point.connector.destroy(); // #7243
- }
- }
- // record changes in the parallel arrays
- i = point.index;
- series.updateParallelArrays(point, i);
- // Record the options to options.data. If the old or the new config
- // is an object, use point options, otherwise use raw options
- // (#4701, #4916).
- seriesOptions.data[i] = (isObject(seriesOptions.data[i], true) ||
- isObject(options, true)) ?
- point.options :
- pick(options, seriesOptions.data[i]);
- // redraw
- series.isDirty = series.isDirtyData = true;
- if (!series.fixedBox && series.hasCartesianSeries) { // #1906, #2320
- chart.isDirtyBox = true;
- }
- if (seriesOptions.legendType === 'point') { // #1831, #1885
- chart.isDirtyLegend = true;
- }
- if (redraw) {
- chart.redraw(animation);
- }
- }
- // Fire the event with a default handler of doing the update
- if (runEvent === false) { // When called from setData
- update();
- }
- else {
- point.firePointEvent('update', { options: options }, update);
- }
- },
- /**
- * Remove a point and optionally redraw the series and if necessary the axes
- *
- * @sample highcharts/plotoptions/series-point-events-remove/
- * Remove point and confirm
- * @sample highcharts/members/point-remove/
- * Remove pie slice
- * @sample maps/members/point-remove/
- * Remove selected points in Highmaps
- *
- * @function Highcharts.Point#remove
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call. When
- * doing more operations on the chart, for example running
- * `point.remove()` in a loop, it is best practice to set `redraw`
- * to false and call `chart.redraw()` after.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation=false]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @return {void}
- */
- remove: function (redraw, animation) {
- this.series.removePoint(this.series.data.indexOf(this), redraw, animation);
- }
- });
- // Extend the series prototype for dynamic methods
- extend(Series.prototype, /** @lends Series.prototype */ {
- /**
- * Add a point to the series after render time. The point can be added at
- * the end, or by giving it an X value, to the start or in the middle of the
- * series.
- *
- * @sample highcharts/members/series-addpoint-append/
- * Append point
- * @sample highcharts/members/series-addpoint-append-and-shift/
- * Append and shift
- * @sample highcharts/members/series-addpoint-x-and-y/
- * Both X and Y values given
- * @sample highcharts/members/series-addpoint-pie/
- * Append pie slice
- * @sample stock/members/series-addpoint/
- * Append 100 points in Highstock
- * @sample stock/members/series-addpoint-shift/
- * Append and shift in Highstock
- * @sample maps/members/series-addpoint/
- * Add a point in Highmaps
- *
- * @function Highcharts.Series#addPoint
- *
- * @param {Highcharts.PointOptionsType} options
- * The point options. If options is a single number, a point with
- * that y value is appended to the series. If it is an array, it will
- * be interpreted as x and y values respectively. If it is an
- * object, advanced options as outlined under `series.data` are
- * applied.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the point is added. When adding
- * more than one point, it is highly recommended that the redraw
- * option be set to false, and instead {@link Chart#redraw} is
- * explicitly called after the adding of points is finished.
- * Otherwise, the chart will redraw after adding each point.
- *
- * @param {boolean} [shift=false]
- * If true, a point is shifted off the start of the series as one is
- * appended to the end.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @param {boolean} [withEvent=true]
- * Used internally, whether to fire the series `addPoint` event.
- *
- * @return {void}
- *
- * @fires Highcharts.Series#event:addPoint
- */
- addPoint: function (options, redraw, shift, animation, withEvent) {
- var series = this,
- seriesOptions = series.options,
- data = series.data,
- chart = series.chart,
- xAxis = series.xAxis,
- names = xAxis && xAxis.hasNames && xAxis.names,
- dataOptions = seriesOptions.data,
- point,
- xData = series.xData,
- isInTheMiddle,
- i,
- x;
- // Optional redraw, defaults to true
- redraw = pick(redraw, true);
- // Get options and push the point to xData, yData and series.options. In
- // series.generatePoints the Point instance will be created on demand
- // and pushed to the series.data array.
- point = { series: series };
- series.pointClass.prototype.applyOptions.apply(point, [options]);
- x = point.x;
- // Get the insertion point
- i = xData.length;
- if (series.requireSorting && x < xData[i - 1]) {
- isInTheMiddle = true;
- while (i && xData[i - 1] > x) {
- i--;
- }
- }
- // Insert undefined item
- series.updateParallelArrays(point, 'splice', i, 0, 0);
- // Update it
- series.updateParallelArrays(point, i);
- if (names && point.name) {
- names[x] = point.name;
- }
- dataOptions.splice(i, 0, options);
- if (isInTheMiddle) {
- series.data.splice(i, 0, null);
- series.processData();
- }
- // Generate points to be added to the legend (#1329)
- if (seriesOptions.legendType === 'point') {
- series.generatePoints();
- }
- // Shift the first point off the parallel arrays
- if (shift) {
- if (data[0] && data[0].remove) {
- data[0].remove(false);
- }
- else {
- data.shift();
- series.updateParallelArrays(point, 'shift');
- dataOptions.shift();
- }
- }
- // Fire event
- if (withEvent !== false) {
- fireEvent(series, 'addPoint', { point: point });
- }
- // redraw
- series.isDirty = true;
- series.isDirtyData = true;
- if (redraw) {
- chart.redraw(animation); // Animation is set anyway on redraw, #5665
- }
- },
- /**
- * Remove a point from the series. Unlike the
- * {@link Highcharts.Point#remove} method, this can also be done on a point
- * that is not instanciated because it is outside the view or subject to
- * Highstock data grouping.
- *
- * @sample highcharts/members/series-removepoint/
- * Remove cropped point
- *
- * @function Highcharts.Series#removePoint
- *
- * @param {number} i
- * The index of the point in the {@link Highcharts.Series.data|data}
- * array.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the point is added. When
- * removing more than one point, it is highly recommended that the
- * `redraw` option be set to `false`, and instead {@link
- * Highcharts.Chart#redraw} is explicitly called after the adding of
- * points is finished.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Whether and optionally how the series should be animated.
- *
- * @return {void}
- *
- * @fires Highcharts.Point#event:remove
- */
- removePoint: function (i, redraw, animation) {
- var series = this,
- data = series.data,
- point = data[i],
- points = series.points,
- chart = series.chart,
- remove = function () {
- if (points && points.length === data.length) { // #4935
- points.splice(i, 1);
- }
- data.splice(i, 1);
- series.options.data.splice(i, 1);
- series.updateParallelArrays(point || { series: series }, 'splice', i, 1);
- if (point) {
- point.destroy();
- }
- // redraw
- series.isDirty = true;
- series.isDirtyData = true;
- if (redraw) {
- chart.redraw();
- }
- };
- setAnimation(animation, chart);
- redraw = pick(redraw, true);
- // Fire the event with a default handler of removing the point
- if (point) {
- point.firePointEvent('remove', null, remove);
- }
- else {
- remove();
- }
- },
- /**
- * Remove a series and optionally redraw the chart.
- *
- * @sample highcharts/members/series-remove/
- * Remove first series from a button
- *
- * @function Highcharts.Series#remove
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call to
- * {@link Highcharts.Chart#redraw}.
- *
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Whether to apply animation, and optionally animation
- * configuration.
- *
- * @param {boolean} [withEvent=true]
- * Used internally, whether to fire the series `remove` event.
- *
- * @return {void}
- *
- * @fires Highcharts.Series#event:remove
- */
- remove: function (redraw, animation, withEvent, keepEvents) {
- var series = this,
- chart = series.chart;
- /**
- * @private
- */
- function remove() {
- // Destroy elements
- series.destroy(keepEvents);
- series.remove = null; // Prevent from doing again (#9097)
- // Redraw
- chart.isDirtyLegend = chart.isDirtyBox = true;
- chart.linkSeries();
- if (pick(redraw, true)) {
- chart.redraw(animation);
- }
- }
- // Fire the event with a default handler of removing the point
- if (withEvent !== false) {
- fireEvent(series, 'remove', null, remove);
- }
- else {
- remove();
- }
- },
- /**
- * Update the series with a new set of options. For a clean and precise
- * handling of new options, all methods and elements from the series are
- * removed, and it is initialized from scratch. Therefore, this method is
- * more performance expensive than some other utility methods like {@link
- * Series#setData} or {@link Series#setVisible}.
- *
- * Note that `Series.update` may mutate the passed `data` options.
- *
- * @sample highcharts/members/series-update/
- * Updating series options
- * @sample maps/members/series-update/
- * Update series options in Highmaps
- *
- * @function Highcharts.Series#update
- *
- * @param {Highcharts.SeriesOptionsType} options
- * New options that will be merged with the series' existing options.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the series is altered. If doing
- * more operations on the chart, it is a good idea to set redraw to
- * false and call {@link Chart#redraw} after.
- *
- * @return {void}
- *
- * @fires Highcharts.Series#event:update
- * @fires Highcharts.Series#event:afterUpdate
- */
- update: function (options, redraw) {
- options = H.cleanRecursively(options, this.userOptions);
- fireEvent(this, 'update', { options: options });
- var series = this,
- chart = series.chart,
- // must use user options when changing type because series.options
- // is merged in with type specific plotOptions
- oldOptions = series.userOptions,
- seriesOptions,
- initialType = series.initialType || series.type,
- newType = (options.type ||
- oldOptions.type ||
- chart.options.chart.type),
- keepPoints = !(
- // Indicators, histograms etc recalculate the data. It should be
- // possible to omit this.
- this.hasDerivedData ||
- // Changes to data grouping requires new points in new groups
- options.dataGrouping ||
- // New type requires new point classes
- (newType && newType !== this.type) ||
- // New options affecting how the data points are built
- typeof options.pointStart !== 'undefined' ||
- options.pointInterval ||
- options.pointIntervalUnit ||
- options.keys),
- initialSeriesProto = seriesTypes[initialType].prototype,
- n,
- groups = [
- 'group',
- 'markerGroup',
- 'dataLabelsGroup',
- 'transformGroup'
- ],
- preserve = [
- 'eventOptions',
- 'navigatorSeries',
- 'baseSeries'
- ],
- // Animation must be enabled when calling update before the initial
- // animation has first run. This happens when calling update
- // directly after chart initialization, or when applying responsive
- // rules (#6912).
- animation = series.finishedAnimating && { animation: false },
- kinds = {};
- if (keepPoints) {
- preserve.push('data', 'isDirtyData', 'points', 'processedXData', 'processedYData', 'xIncrement', 'cropped', '_hasPointMarkers', '_hasPointLabels',
- // Map specific, consider moving it to series-specific preserve-
- // properties (#10617)
- 'mapMap', 'mapData', 'minY', 'maxY', 'minX', 'maxX');
- if (options.visible !== false) {
- preserve.push('area', 'graph');
- }
- series.parallelArrays.forEach(function (key) {
- preserve.push(key + 'Data');
- });
- if (options.data) {
- // setData uses dataSorting options so we need to update them
- // earlier
- if (options.dataSorting) {
- extend(series.options.dataSorting, options.dataSorting);
- }
- this.setData(options.data, false);
- }
- }
- // Do the merge, with some forced options
- options = merge(oldOptions, animation, {
- // When oldOptions.index is null it should't be cleared.
- // Otherwise navigator series will have wrong indexes (#10193).
- index: typeof oldOptions.index === 'undefined' ?
- series.index : oldOptions.index,
- pointStart: pick(
- // when updating from blank (#7933)
- oldOptions.pointStart,
- // when updating after addPoint
- series.xData[0])
- }, (!keepPoints && { data: series.options.data }), options);
- // Merge does not merge arrays, but replaces them. Since points were
- // updated, `series.options.data` has correct merged options, use it:
- if (keepPoints && options.data) {
- options.data = series.options.data;
- }
- // Make sure preserved properties are not destroyed (#3094)
- preserve = groups.concat(preserve);
- preserve.forEach(function (prop) {
- preserve[prop] = series[prop];
- delete series[prop];
- });
- // Destroy the series and delete all properties. Reinsert all
- // methods and properties from the new type prototype (#2270,
- // #3719).
- series.remove(false, null, false, true);
- for (n in initialSeriesProto) { // eslint-disable-line guard-for-in
- series[n] = void 0;
- }
- if (seriesTypes[newType || initialType]) {
- extend(series, seriesTypes[newType || initialType].prototype);
- }
- else {
- error(17, true, chart, { missingModuleFor: (newType || initialType) });
- }
- // Re-register groups (#3094) and other preserved properties
- preserve.forEach(function (prop) {
- series[prop] = preserve[prop];
- });
- series.init(chart, options);
- // Remove particular elements of the points. Check `series.options`
- // because we need to consider the options being set on plotOptions as
- // well.
- if (keepPoints && this.points) {
- seriesOptions = series.options;
- // What kind of elements to destroy
- if (seriesOptions.visible === false) {
- kinds.graphic = 1;
- kinds.dataLabel = 1;
- }
- else if (!series._hasPointLabels) {
- var marker = seriesOptions.marker,
- dataLabels = seriesOptions.dataLabels;
- if (marker && (marker.enabled === false ||
- 'symbol' in marker // #10870
- )) {
- kinds.graphic = 1;
- }
- if (dataLabels &&
- dataLabels.enabled === false) {
- kinds.dataLabel = 1;
- }
- }
- this.points.forEach(function (point) {
- if (point && point.series) {
- point.resolveColor();
- // Destroy elements in order to recreate based on updated
- // series options.
- if (Object.keys(kinds).length) {
- point.destroyElements(kinds);
- }
- if (seriesOptions.showInLegend === false &&
- point.legendItem) {
- chart.legend.destroyItem(point);
- }
- }
- }, this);
- }
- series.initialType = initialType;
- chart.linkSeries(); // Links are lost in series.remove (#3028)
- fireEvent(this, 'afterUpdate');
- if (pick(redraw, true)) {
- chart.redraw(keepPoints ? void 0 : false);
- }
- },
- /**
- * Used from within series.update
- *
- * @private
- * @function Highcharts.Series#setName
- *
- * @param {string} name
- *
- * @return {void}
- */
- setName: function (name) {
- this.name = this.options.name = this.userOptions.name = name;
- this.chart.isDirtyLegend = true;
- }
- });
- // Extend the Axis.prototype for dynamic methods
- extend(Axis.prototype, /** @lends Highcharts.Axis.prototype */ {
- /**
- * Update an axis object with a new set of options. The options are merged
- * with the existing options, so only new or altered options need to be
- * specified.
- *
- * @sample highcharts/members/axis-update/
- * Axis update demo
- *
- * @function Highcharts.Axis#update
- *
- * @param {Highcharts.AxisOptions} options
- * The new options that will be merged in with existing options on
- * the axis.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the axis is altered. If doing
- * more operations on the chart, it is a good idea to set redraw to
- * false and call {@link Chart#redraw} after.
- *
- * @return {void}
- */
- update: function (options, redraw) {
- var chart = this.chart,
- newEvents = ((options && options.events) || {});
- options = merge(this.userOptions, options);
- // Color Axis is not an array,
- // This change is applied in the ColorAxis wrapper
- if (chart.options[this.coll].indexOf) {
- // Don't use this.options.index,
- // StockChart has Axes in navigator too
- chart.options[this.coll][chart.options[this.coll].indexOf(this.userOptions)] = options;
- }
- // Remove old events, if no new exist (#8161)
- objectEach(chart.options[this.coll].events, function (fn, ev) {
- if (typeof newEvents[ev] === 'undefined') {
- newEvents[ev] = void 0;
- }
- });
- this.destroy(true);
- this.init(chart, extend(options, { events: newEvents }));
- chart.isDirtyBox = true;
- if (pick(redraw, true)) {
- chart.redraw();
- }
- },
- /**
- * Remove the axis from the chart.
- *
- * @sample highcharts/members/chart-addaxis/
- * Add and remove axes
- *
- * @function Highcharts.Axis#remove
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart following the remove.
- *
- * @return {void}
- */
- remove: function (redraw) {
- var chart = this.chart,
- key = this.coll, // xAxis or yAxis
- axisSeries = this.series,
- i = axisSeries.length;
- // Remove associated series (#2687)
- while (i--) {
- if (axisSeries[i]) {
- axisSeries[i].remove(false);
- }
- }
- // Remove the axis
- erase(chart.axes, this);
- erase(chart[key], this);
- if (isArray(chart.options[key])) {
- chart.options[key].splice(this.options.index, 1);
- }
- else { // color axis, #6488
- delete chart.options[key];
- }
- chart[key].forEach(function (axis, i) {
- // Re-index, #1706, #8075
- axis.options.index = axis.userOptions.index = i;
- });
- this.destroy();
- chart.isDirtyBox = true;
- if (pick(redraw, true)) {
- chart.redraw();
- }
- },
- /**
- * Update the axis title by options after render time.
- *
- * @sample highcharts/members/axis-settitle/
- * Set a new Y axis title
- *
- * @function Highcharts.Axis#setTitle
- *
- * @param {Highcharts.AxisTitleOptions} titleOptions
- * The additional title options.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after setting the title.
- *
- * @return {void}
- */
- setTitle: function (titleOptions, redraw) {
- this.update({ title: titleOptions }, redraw);
- },
- /**
- * Set new axis categories and optionally redraw.
- *
- * @sample highcharts/members/axis-setcategories/
- * Set categories by click on a button
- *
- * @function Highcharts.Axis#setCategories
- *
- * @param {Array<string>} categories
- * The new categories.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart.
- *
- * @return {void}
- */
- setCategories: function (categories, redraw) {
- this.update({ categories: categories }, redraw);
- }
- });
- });
- _registerModule(_modules, 'Series/AreaSeries.js', [_modules['Core/Globals.js'], _modules['Core/Color.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Utilities.js']], function (H, Color, LegendSymbolMixin, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var color = Color.parse;
- var objectEach = U.objectEach,
- pick = U.pick,
- seriesType = U.seriesType;
- var Series = H.Series;
- /**
- * Area series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.area
- *
- * @augments Highcharts.Series
- */
- seriesType('area', 'line',
- /**
- * The area series type.
- *
- * @sample {highcharts} highcharts/demo/area-basic/
- * Area chart
- * @sample {highstock} stock/demo/area/
- * Area chart
- *
- * @extends plotOptions.line
- * @excluding useOhlcData
- * @product highcharts highstock
- * @optionparent plotOptions.area
- */
- {
- /**
- * Fill color or gradient for the area. When `null`, the series' `color`
- * is used with the series' `fillOpacity`.
- *
- * In styled mode, the fill color can be set with the `.highcharts-area`
- * class name.
- *
- * @sample {highcharts} highcharts/plotoptions/area-fillcolor-default/
- * Null by default
- * @sample {highcharts} highcharts/plotoptions/area-fillcolor-gradient/
- * Gradient
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock
- * @apioption plotOptions.area.fillColor
- */
- /**
- * Fill opacity for the area. When you set an explicit `fillColor`,
- * the `fillOpacity` is not applied. Instead, you should define the
- * opacity in the `fillColor` with an rgba color definition. The
- * `fillOpacity` setting, also the default setting, overrides the alpha
- * component of the `color` setting.
- *
- * In styled mode, the fill opacity can be set with the
- * `.highcharts-area` class name.
- *
- * @sample {highcharts} highcharts/plotoptions/area-fillopacity/
- * Automatic fill color and fill opacity of 0.1
- *
- * @type {number}
- * @default {highcharts} 0.75
- * @default {highstock} 0.75
- * @product highcharts highstock
- * @apioption plotOptions.area.fillOpacity
- */
- /**
- * A separate color for the graph line. By default the line takes the
- * `color` of the series, but the lineColor setting allows setting a
- * separate color for the line without altering the `fillColor`.
- *
- * In styled mode, the line stroke can be set with the
- * `.highcharts-graph` class name.
- *
- * @sample {highcharts} highcharts/plotoptions/area-linecolor/
- * Dark gray line
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock
- * @apioption plotOptions.area.lineColor
- */
- /**
- * A separate color for the negative part of the area.
- *
- * In styled mode, a negative color is set with the
- * `.highcharts-negative` class name.
- *
- * @see [negativeColor](#plotOptions.area.negativeColor)
- *
- * @sample {highcharts} highcharts/css/series-negative-color/
- * Negative color in styled mode
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 3.0
- * @product highcharts
- * @apioption plotOptions.area.negativeFillColor
- */
- /**
- * Whether the whole area or just the line should respond to mouseover
- * tooltips and other mouse or touch events.
- *
- * @sample {highcharts|highstock} highcharts/plotoptions/area-trackbyarea/
- * Display the tooltip when the area is hovered
- *
- * @type {boolean}
- * @default false
- * @since 1.1.6
- * @product highcharts highstock
- * @apioption plotOptions.area.trackByArea
- */
- /**
- * The Y axis value to serve as the base for the area, for
- * distinguishing between values above and below a threshold. The area
- * between the graph and the threshold is filled.
- *
- * * If a number is given, the Y axis will scale to the threshold.
- * * If `null`, the scaling behaves like a line series with fill between
- * the graph and the Y axis minimum.
- * * If `Infinity` or `-Infinity`, the area between the graph and the
- * corresponding Y axis extreme is filled (since v6.1.0).
- *
- * @sample {highcharts} highcharts/plotoptions/area-threshold/
- * A threshold of 100
- * @sample {highcharts} highcharts/plotoptions/area-threshold-infinity/
- * A threshold of Infinity
- *
- * @type {number|null}
- * @since 2.0
- * @product highcharts highstock
- */
- threshold: 0
- },
- /* eslint-disable valid-jsdoc */
- /**
- * @lends seriesTypes.area.prototype
- */
- {
- singleStacks: false,
- /**
- * Return an array of stacked points, where null and missing points are
- * replaced by dummy points in order for gaps to be drawn correctly in
- * stacks.
- * @private
- */
- getStackPoints: function (points) {
- var series = this,
- segment = [],
- keys = [],
- xAxis = this.xAxis,
- yAxis = this.yAxis,
- stack = yAxis.stacking.stacks[this.stackKey],
- pointMap = {},
- seriesIndex = series.index,
- yAxisSeries = yAxis.series,
- seriesLength = yAxisSeries.length,
- visibleSeries,
- upOrDown = pick(yAxis.options.reversedStacks,
- true) ? 1 : -1,
- i;
- points = points || this.points;
- if (this.options.stacking) {
- for (i = 0; i < points.length; i++) {
- // Reset after point update (#7326)
- points[i].leftNull = points[i].rightNull = void 0;
- // Create a map where we can quickly look up the points by
- // their X values.
- pointMap[points[i].x] = points[i];
- }
- // Sort the keys (#1651)
- objectEach(stack, function (stackX, x) {
- // nulled after switching between
- // grouping and not (#1651, #2336)
- if (stackX.total !== null) {
- keys.push(x);
- }
- });
- keys.sort(function (a, b) {
- return a - b;
- });
- visibleSeries = yAxisSeries.map(function (s) {
- return s.visible;
- });
- keys.forEach(function (x, idx) {
- var y = 0,
- stackPoint,
- stackedValues;
- if (pointMap[x] && !pointMap[x].isNull) {
- segment.push(pointMap[x]);
- // Find left and right cliff. -1 goes left, 1 goes
- // right.
- [-1, 1].forEach(function (direction) {
- var nullName = direction === 1 ?
- 'rightNull' :
- 'leftNull',
- cliffName = direction === 1 ?
- 'rightCliff' :
- 'leftCliff',
- cliff = 0,
- otherStack = stack[keys[idx + direction]];
- // If there is a stack next to this one,
- // to the left or to the right...
- if (otherStack) {
- i = seriesIndex;
- // Can go either up or down,
- // depending on reversedStacks
- while (i >= 0 && i < seriesLength) {
- stackPoint = otherStack.points[i];
- if (!stackPoint) {
- // If the next point in this series
- // is missing, mark the point
- // with point.leftNull or
- // point.rightNull = true.
- if (i === seriesIndex) {
- pointMap[x][nullName] =
- true;
- // If there are missing points in
- // the next stack in any of the
- // series below this one, we need
- // to substract the missing values
- // and add a hiatus to the left or
- // right.
- }
- else if (visibleSeries[i]) {
- stackedValues =
- stack[x].points[i];
- if (stackedValues) {
- cliff -=
- stackedValues[1] -
- stackedValues[0];
- }
- }
- }
- // When reversedStacks is true, loop up,
- // else loop down
- i += upOrDown;
- }
- }
- pointMap[x][cliffName] = cliff;
- });
- // There is no point for this X value in this series, so we
- // insert a dummy point in order for the areas to be drawn
- // correctly.
- }
- else {
- // Loop down the stack to find the series below this
- // one that has a value (#1991)
- i = seriesIndex;
- while (i >= 0 && i < seriesLength) {
- stackPoint = stack[x].points[i];
- if (stackPoint) {
- y = stackPoint[1];
- break;
- }
- // When reversedStacks is true, loop up, else loop
- // down
- i += upOrDown;
- }
- y = yAxis.translate(// #6272
- y, 0, 1, 0, 1);
- segment.push({
- isNull: true,
- plotX: xAxis.translate(// #6272
- x, 0, 0, 0, 1),
- x: x,
- plotY: y,
- yBottom: y
- });
- }
- });
- }
- return segment;
- },
- /**
- * @private
- */
- getGraphPath: function (points) {
- var getGraphPath = Series.prototype.getGraphPath, graphPath, options = this.options, stacking = options.stacking, yAxis = this.yAxis, topPath, bottomPath, bottomPoints = [], graphPoints = [], seriesIndex = this.index, i, areaPath, plotX, stacks = yAxis.stacking.stacks[this.stackKey], threshold = options.threshold, translatedThreshold = Math.round(// #10909
- yAxis.getThreshold(options.threshold)), isNull, yBottom, connectNulls = pick(// #10574
- options.connectNulls, stacking === 'percent'),
- // To display null points in underlying stacked series, this
- // series graph must be broken, and the area also fall down to
- // fill the gap left by the null point. #2069
- addDummyPoints = function (i, otherI, side) {
- var point = points[i], stackedValues = stacking &&
- stacks[point.x].points[seriesIndex], nullVal = point[side + 'Null'] || 0, cliffVal = point[side + 'Cliff'] || 0, top, bottom, isNull = true;
- if (cliffVal || nullVal) {
- top = (nullVal ?
- stackedValues[0] :
- stackedValues[1]) + cliffVal;
- bottom = stackedValues[0] + cliffVal;
- isNull = !!nullVal;
- }
- else if (!stacking &&
- points[otherI] &&
- points[otherI].isNull) {
- top = bottom = threshold;
- }
- // Add to the top and bottom line of the area
- if (typeof top !== 'undefined') {
- graphPoints.push({
- plotX: plotX,
- plotY: top === null ?
- translatedThreshold :
- yAxis.getThreshold(top),
- isNull: isNull,
- isCliff: true
- });
- bottomPoints.push({
- plotX: plotX,
- plotY: bottom === null ?
- translatedThreshold :
- yAxis.getThreshold(bottom),
- doCurve: false // #1041, gaps in areaspline areas
- });
- }
- };
- // Find what points to use
- points = points || this.points;
- // Fill in missing points
- if (stacking) {
- points = this.getStackPoints(points);
- }
- for (i = 0; i < points.length; i++) {
- // Reset after series.update of stacking property (#12033)
- if (!stacking) {
- points[i].leftCliff = points[i].rightCliff =
- points[i].leftNull = points[i].rightNull = void 0;
- }
- isNull = points[i].isNull;
- plotX = pick(points[i].rectPlotX, points[i].plotX);
- yBottom = stacking ? points[i].yBottom : translatedThreshold;
- if (!isNull || connectNulls) {
- if (!connectNulls) {
- addDummyPoints(i, i - 1, 'left');
- }
- // Skip null point when stacking is false and connectNulls
- // true
- if (!(isNull && !stacking && connectNulls)) {
- graphPoints.push(points[i]);
- bottomPoints.push({
- x: i,
- plotX: plotX,
- plotY: yBottom
- });
- }
- if (!connectNulls) {
- addDummyPoints(i, i + 1, 'right');
- }
- }
- }
- topPath = getGraphPath.call(this, graphPoints, true, true);
- bottomPoints.reversed = true;
- bottomPath = getGraphPath.call(this, bottomPoints, true, true);
- var firstBottomPoint = bottomPath[0];
- if (firstBottomPoint && firstBottomPoint[0] === 'M') {
- bottomPath[0] = ['L', firstBottomPoint[1], firstBottomPoint[2]];
- }
- areaPath = topPath.concat(bottomPath);
- // TODO: don't set leftCliff and rightCliff when connectNulls?
- graphPath = getGraphPath
- .call(this, graphPoints, false, connectNulls);
- areaPath.xMap = topPath.xMap;
- this.areaPath = areaPath;
- return graphPath;
- },
- /**
- * Draw the graph and the underlying area. This method calls the Series
- * base function and adds the area. The areaPath is calculated in the
- * getSegmentPath method called from Series.prototype.drawGraph.
- * @private
- */
- drawGraph: function () {
- // Define or reset areaPath
- this.areaPath = [];
- // Call the base method
- Series.prototype.drawGraph.apply(this);
- // Define local variables
- var series = this,
- areaPath = this.areaPath,
- options = this.options,
- zones = this.zones,
- props = [[
- 'area',
- 'highcharts-area',
- this.color,
- options.fillColor
- ]]; // area name, main color, fill color
- zones.forEach(function (zone,
- i) {
- props.push([
- 'zone-area-' + i,
- 'highcharts-area highcharts-zone-area-' + i + ' ' +
- zone.className,
- zone.color || series.color,
- zone.fillColor || options.fillColor
- ]);
- });
- props.forEach(function (prop) {
- var areaKey = prop[0],
- area = series[areaKey],
- verb = area ? 'animate' : 'attr',
- attribs = {};
- // Create or update the area
- if (area) { // update
- area.endX = series.preventGraphAnimation ?
- null :
- areaPath.xMap;
- area.animate({ d: areaPath });
- }
- else { // create
- attribs.zIndex = 0; // #1069
- area = series[areaKey] = series.chart.renderer
- .path(areaPath)
- .addClass(prop[1])
- .add(series.group);
- area.isArea = true;
- }
- if (!series.chart.styledMode) {
- attribs.fill = pick(prop[3], color(prop[2])
- .setOpacity(pick(options.fillOpacity, 0.75))
- .get());
- }
- area[verb](attribs);
- area.startX = areaPath.xMap;
- area.shiftUnit = options.step ? 2 : 1;
- });
- },
- drawLegendSymbol: LegendSymbolMixin.drawRectangle
- });
- /* eslint-enable valid-jsdoc */
- /**
- * A `area` series. If the [type](#series.area.type) option is not
- * specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.area
- * @excluding dataParser, dataURL, useOhlcData
- * @product highcharts highstock
- * @apioption series.area
- */
- /**
- * An array of data points for the series. For the `area` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` * and `pointInterval` given in the series options. If the
- * axis has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 9],
- * [1, 7],
- * [2, 6]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.area.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * x: 1,
- * y: 9,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 6,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.line.data
- * @product highcharts highstock
- * @apioption series.area.data
- */
- ''; // adds doclets above to transpilat
- });
- _registerModule(_modules, 'Series/SplineSeries.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var pick = U.pick,
- seriesType = U.seriesType;
- /**
- * Spline series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.spline
- *
- * @augments Highcarts.Series
- */
- seriesType('spline', 'line',
- /**
- * A spline series is a special type of line series, where the segments
- * between the data points are smoothed.
- *
- * @sample {highcharts} highcharts/demo/spline-irregular-time/
- * Spline chart
- * @sample {highstock} stock/demo/spline/
- * Spline chart
- *
- * @extends plotOptions.series
- * @excluding step, boostThreshold, boostBlending
- * @product highcharts highstock
- * @optionparent plotOptions.spline
- */
- {},
- /**
- * @lends seriesTypes.spline.prototype
- */
- {
- /* eslint-disable valid-jsdoc */
- /**
- * Get the spline segment from a given point's previous neighbour to the
- * given point.
- *
- * @private
- * @function Highcharts.seriesTypes.spline#getPointSpline
- *
- * @param {Array<Highcharts.Point>}
- *
- * @param {Highcharts.Point} point
- *
- * @param {number} i
- *
- * @return {Highcharts.SVGPathArray}
- */
- getPointSpline: function (points, point, i) {
- var
- // 1 means control points midway between points, 2 means 1/3
- // from the point, 3 is 1/4 etc
- smoothing = 1.5,
- denom = smoothing + 1,
- plotX = point.plotX || 0,
- plotY = point.plotY || 0,
- lastPoint = points[i - 1],
- nextPoint = points[i + 1],
- leftContX,
- leftContY,
- rightContX,
- rightContY,
- ret;
- /**
- * @private
- */
- function doCurve(otherPoint) {
- return otherPoint &&
- !otherPoint.isNull &&
- otherPoint.doCurve !== false &&
- // #6387, area splines next to null:
- !point.isCliff;
- }
- // Find control points
- if (doCurve(lastPoint) && doCurve(nextPoint)) {
- var lastX = lastPoint.plotX || 0,
- lastY = lastPoint.plotY || 0,
- nextX = nextPoint.plotX || 0,
- nextY = nextPoint.plotY || 0,
- correction = 0;
- leftContX = (smoothing * plotX + lastX) / denom;
- leftContY = (smoothing * plotY + lastY) / denom;
- rightContX = (smoothing * plotX + nextX) / denom;
- rightContY = (smoothing * plotY + nextY) / denom;
- // Have the two control points make a straight line through main
- // point
- if (rightContX !== leftContX) { // #5016, division by zero
- correction = (((rightContY - leftContY) *
- (rightContX - plotX)) /
- (rightContX - leftContX) + plotY - rightContY);
- }
- leftContY += correction;
- rightContY += correction;
- // to prevent false extremes, check that control points are
- // between neighbouring points' y values
- if (leftContY > lastY && leftContY > plotY) {
- leftContY = Math.max(lastY, plotY);
- // mirror of left control point
- rightContY = 2 * plotY - leftContY;
- }
- else if (leftContY < lastY && leftContY < plotY) {
- leftContY = Math.min(lastY, plotY);
- rightContY = 2 * plotY - leftContY;
- }
- if (rightContY > nextY && rightContY > plotY) {
- rightContY = Math.max(nextY, plotY);
- leftContY = 2 * plotY - rightContY;
- }
- else if (rightContY < nextY && rightContY < plotY) {
- rightContY = Math.min(nextY, plotY);
- leftContY = 2 * plotY - rightContY;
- }
- // record for drawing in next point
- point.rightContX = rightContX;
- point.rightContY = rightContY;
- }
- // Visualize control points for debugging
- /*
- if (leftContX) {
- this.chart.renderer.circle(
- leftContX + this.chart.plotLeft,
- leftContY + this.chart.plotTop,
- 2
- )
- .attr({
- stroke: 'red',
- 'stroke-width': 2,
- fill: 'none',
- zIndex: 9
- })
- .add();
- this.chart.renderer.path(['M', leftContX + this.chart.plotLeft,
- leftContY + this.chart.plotTop,
- 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop])
- .attr({
- stroke: 'red',
- 'stroke-width': 2,
- zIndex: 9
- })
- .add();
- }
- if (rightContX) {
- this.chart.renderer.circle(
- rightContX + this.chart.plotLeft,
- rightContY + this.chart.plotTop,
- 2
- )
- .attr({
- stroke: 'green',
- 'stroke-width': 2,
- fill: 'none',
- zIndex: 9
- })
- .add();
- this.chart.renderer.path(['M', rightContX + this.chart.plotLeft,
- rightContY + this.chart.plotTop,
- 'L', plotX + this.chart.plotLeft, plotY + this.chart.plotTop])
- .attr({
- stroke: 'green',
- 'stroke-width': 2,
- zIndex: 9
- })
- .add();
- }
- // */
- ret = [
- 'C',
- pick(lastPoint.rightContX, lastPoint.plotX, 0),
- pick(lastPoint.rightContY, lastPoint.plotY, 0),
- pick(leftContX, plotX, 0),
- pick(leftContY, plotY, 0),
- plotX,
- plotY
- ];
- // reset for updating series later
- lastPoint.rightContX = lastPoint.rightContY = void 0;
- return ret;
- }
- /* eslint-enable valid-jsdoc */
- });
- /**
- * A `spline` series. If the [type](#series.spline.type) option is
- * not specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.spline
- * @excluding dataParser, dataURL, step, boostThreshold, boostBlending
- * @product highcharts highstock
- * @apioption series.spline
- */
- /**
- * An array of data points for the series. For the `spline` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 9],
- * [1, 2],
- * [2, 8]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.spline.turboThreshold),
- * this option is not available.
- * ```js
- * data: [{
- * x: 1,
- * y: 9,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 0,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.line.data
- * @product highcharts highstock
- * @apioption series.spline.data
- */
- ''; // adds doclets above intro transpilat
- });
- _registerModule(_modules, 'Series/AreaSplineSeries.js', [_modules['Core/Globals.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Options.js'], _modules['Core/Utilities.js']], function (H, LegendSymbolMixin, O, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var defaultOptions = O.defaultOptions;
- var seriesType = U.seriesType;
- var areaProto = H.seriesTypes.area.prototype;
- /**
- * AreaSpline series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.areaspline
- *
- * @augments Highcharts.Series
- */
- seriesType('areaspline', 'spline',
- /**
- * The area spline series is an area series where the graph between the
- * points is smoothed into a spline.
- *
- * @sample {highcharts} highcharts/demo/areaspline/
- * Area spline chart
- * @sample {highstock} stock/demo/areaspline/
- * Area spline chart
- *
- * @extends plotOptions.area
- * @excluding step, boostThreshold, boostBlending
- * @product highcharts highstock
- * @apioption plotOptions.areaspline
- */
- defaultOptions.plotOptions.area, {
- getStackPoints: areaProto.getStackPoints,
- getGraphPath: areaProto.getGraphPath,
- drawGraph: areaProto.drawGraph,
- drawLegendSymbol: LegendSymbolMixin.drawRectangle
- });
- /**
- * A `areaspline` series. If the [type](#series.areaspline.type) option
- * is not specified, it is inherited from [chart.type](#chart.type).
- *
- *
- * @extends series,plotOptions.areaspline
- * @excluding dataParser, dataURL, step, boostThreshold, boostBlending
- * @product highcharts highstock
- * @apioption series.areaspline
- */
- /**
- * An array of data points for the series. For the `areaspline` series
- * type, points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 10],
- * [1, 9],
- * [2, 3]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.areaspline.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * x: 1,
- * y: 4,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 4,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.line.data
- * @product highcharts highstock
- * @apioption series.areaspline.data
- */
- ''; // adds doclets above into transpilat
- });
- _registerModule(_modules, 'Series/ColumnSeries.js', [_modules['Core/Globals.js'], _modules['Core/Color.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Utilities.js']], function (H, Color, LegendSymbolMixin, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Adjusted width and x offset of the columns for grouping.
- *
- * @private
- * @interface Highcharts.ColumnMetricsObject
- */ /**
- * Width of the columns.
- * @name Highcharts.ColumnMetricsObject#width
- * @type {number}
- */ /**
- * Offset of the columns.
- * @name Highcharts.ColumnMetricsObject#offset
- * @type {number}
- */
- ''; // detach doclets above
- var color = Color.parse;
- var animObject = U.animObject,
- clamp = U.clamp,
- defined = U.defined,
- extend = U.extend,
- isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- seriesType = U.seriesType,
- objectEach = U.objectEach;
- var noop = H.noop,
- Series = H.Series,
- svg = H.svg;
- /**
- * The column series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.column
- *
- * @augments Highcharts.Series
- */
- seriesType('column', 'line',
- /**
- * Column series display one column per value along an X axis.
- *
- * @sample {highcharts} highcharts/demo/column-basic/
- * Column chart
- * @sample {highstock} stock/demo/column/
- * Column chart
- *
- * @extends plotOptions.line
- * @excluding connectEnds, connectNulls, gapSize, gapUnit, linecap,
- * lineWidth, marker, step, useOhlcData
- * @product highcharts highstock
- * @optionparent plotOptions.column
- */
- {
- /**
- * The corner radius of the border surrounding each column or bar.
- *
- * @sample {highcharts} highcharts/plotoptions/column-borderradius/
- * Rounded columns
- *
- * @product highcharts highstock gantt
- *
- * @private
- */
- borderRadius: 0,
- /**
- * When using automatic point colors pulled from the global
- * [colors](colors) or series-specific
- * [plotOptions.column.colors](series.colors) collections, this option
- * determines whether the chart should receive one color per series or
- * one color per point.
- *
- * In styled mode, the `colors` or `series.colors` arrays are not
- * supported, and instead this option gives the points individual color
- * class names on the form `highcharts-color-{n}`.
- *
- * @see [series colors](#plotOptions.column.colors)
- *
- * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-false/
- * False by default
- * @sample {highcharts} highcharts/plotoptions/column-colorbypoint-true/
- * True
- *
- * @type {boolean}
- * @default false
- * @since 2.0
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.colorByPoint
- */
- /**
- * A series specific or series type specific color set to apply instead
- * of the global [colors](#colors) when [colorByPoint](
- * #plotOptions.column.colorByPoint) is true.
- *
- * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>}
- * @since 3.0
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.colors
- */
- /**
- * When `true`, the columns will center in the category, ignoring null
- * or missing points. When `false`, space will be reserved for null or
- * missing points.
- *
- * @sample {highcharts} highcharts/series-column/centerincategory/
- * Center in category
- *
- * @since 8.0.1
- * @product highcharts highstock gantt
- */
- centerInCategory: false,
- /**
- * Padding between each value groups, in x axis units.
- *
- * @sample {highcharts} highcharts/plotoptions/column-grouppadding-default/
- * 0.2 by default
- * @sample {highcharts} highcharts/plotoptions/column-grouppadding-none/
- * No group padding - all columns are evenly spaced
- *
- * @product highcharts highstock gantt
- *
- * @private
- */
- groupPadding: 0.2,
- /**
- * Whether to group non-stacked columns or to let them render
- * independent of each other. Non-grouped columns will be laid out
- * individually and overlap each other.
- *
- * @sample {highcharts} highcharts/plotoptions/column-grouping-false/
- * Grouping disabled
- * @sample {highstock} highcharts/plotoptions/column-grouping-false/
- * Grouping disabled
- *
- * @type {boolean}
- * @default true
- * @since 2.3.0
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.grouping
- */
- /**
- * @ignore-option
- * @private
- */
- marker: null,
- /**
- * The maximum allowed pixel width for a column, translated to the
- * height of a bar in a bar chart. This prevents the columns from
- * becoming too wide when there is a small number of points in the
- * chart.
- *
- * @see [pointWidth](#plotOptions.column.pointWidth)
- *
- * @sample {highcharts} highcharts/plotoptions/column-maxpointwidth-20/
- * Limited to 50
- * @sample {highstock} highcharts/plotoptions/column-maxpointwidth-20/
- * Limited to 50
- *
- * @type {number}
- * @since 4.1.8
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.maxPointWidth
- */
- /**
- * Padding between each column or bar, in x axis units.
- *
- * @sample {highcharts} highcharts/plotoptions/column-pointpadding-default/
- * 0.1 by default
- * @sample {highcharts} highcharts/plotoptions/column-pointpadding-025/
- * 0.25
- * @sample {highcharts} highcharts/plotoptions/column-pointpadding-none/
- * 0 for tightly packed columns
- *
- * @product highcharts highstock gantt
- *
- * @private
- */
- pointPadding: 0.1,
- /**
- * A pixel value specifying a fixed width for each column or bar point.
- * When `null`, the width is calculated from the `pointPadding` and
- * `groupPadding`. The width effects the dimension that is not based on
- * the point value. For column series it is the hoizontal length and for
- * bar series it is the vertical length.
- *
- * @see [maxPointWidth](#plotOptions.column.maxPointWidth)
- *
- * @sample {highcharts} highcharts/plotoptions/column-pointwidth-20/
- * 20px wide columns regardless of chart width or the amount of
- * data points
- *
- * @type {number}
- * @since 1.2.5
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.pointWidth
- */
- /**
- * A pixel value specifying a fixed width for the column or bar.
- * Overrides pointWidth on the series.
- *
- * @see [series.pointWidth](#plotOptions.column.pointWidth)
- *
- * @type {number}
- * @default undefined
- * @since 7.0.0
- * @product highcharts highstock gantt
- * @apioption series.column.data.pointWidth
- */
- /**
- * The minimal height for a column or width for a bar. By default,
- * 0 values are not shown. To visualize a 0 (or close to zero) point,
- * set the minimal point length to a pixel value like 3\. In stacked
- * column charts, minPointLength might not be respected for tightly
- * packed values.
- *
- * @sample {highcharts} highcharts/plotoptions/column-minpointlength/
- * Zero base value
- * @sample {highcharts} highcharts/plotoptions/column-minpointlength-pos-and-neg/
- * Positive and negative close to zero values
- *
- * @product highcharts highstock gantt
- *
- * @private
- */
- minPointLength: 0,
- /**
- * When the series contains less points than the crop threshold, all
- * points are drawn, event if the points fall outside the visible plot
- * area at the current zoom. The advantage of drawing all points
- * (including markers and columns), is that animation is performed on
- * updates. On the other hand, when the series contains more points than
- * the crop threshold, the series data is cropped to only contain points
- * that fall within the plot area. The advantage of cropping away
- * invisible points is to increase performance on large series.
- *
- * @product highcharts highstock gantt
- *
- * @private
- */
- cropThreshold: 50,
- /**
- * The X axis range that each point is valid for. This determines the
- * width of the column. On a categorized axis, the range will be 1
- * by default (one category unit). On linear and datetime axes, the
- * range will be computed as the distance between the two closest data
- * points.
- *
- * The default `null` means it is computed automatically, but this
- * option can be used to override the automatic value.
- *
- * This option is set by default to 1 if data sorting is enabled.
- *
- * @sample {highcharts} highcharts/plotoptions/column-pointrange/
- * Set the point range to one day on a data set with one week
- * between the points
- *
- * @type {number|null}
- * @since 2.3
- * @product highcharts highstock gantt
- *
- * @private
- */
- pointRange: null,
- states: {
- /**
- * Options for the hovered point. These settings override the normal
- * state options when a point is moused over or touched.
- *
- * @extends plotOptions.series.states.hover
- * @excluding halo, lineWidth, lineWidthPlus, marker
- * @product highcharts highstock gantt
- */
- hover: {
- /** @ignore-option */
- halo: false,
- /**
- * A specific border color for the hovered point. Defaults to
- * inherit the normal state border color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts gantt
- * @apioption plotOptions.column.states.hover.borderColor
- */
- /**
- * A specific color for the hovered point.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts gantt
- * @apioption plotOptions.column.states.hover.color
- */
- /**
- * How much to brighten the point on interaction. Requires the
- * main color to be defined in hex or rgb(a) format.
- *
- * In styled mode, the hover brightening is by default replaced
- * with a fill-opacity set in the `.highcharts-point:hover`
- * rule.
- *
- * @sample {highcharts} highcharts/plotoptions/column-states-hover-brightness/
- * Brighten by 0.5
- *
- * @product highcharts highstock gantt
- */
- brightness: 0.1
- },
- /**
- * Options for the selected point. These settings override the
- * normal state options when a point is selected.
- *
- * @extends plotOptions.series.states.select
- * @excluding halo, lineWidth, lineWidthPlus, marker
- * @product highcharts highstock gantt
- */
- select: {
- /**
- * A specific color for the selected point.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #cccccc
- * @product highcharts highstock gantt
- */
- color: '#cccccc',
- /**
- * A specific border color for the selected point.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #000000
- * @product highcharts highstock gantt
- */
- borderColor: '#000000'
- }
- },
- dataLabels: {
- align: void 0,
- verticalAlign: void 0,
- /**
- * The y position offset of the label relative to the point in
- * pixels.
- *
- * @type {number}
- */
- y: void 0
- },
- // false doesn't work well: https://jsfiddle.net/highcharts/hz8fopan/14/
- /**
- * @ignore-option
- * @private
- */
- startFromThreshold: true,
- stickyTracking: false,
- tooltip: {
- distance: 6
- },
- /**
- * The Y axis value to serve as the base for the columns, for
- * distinguishing between values above and below a threshold. If `null`,
- * the columns extend from the padding Y axis minimum.
- *
- * @type {number|null}
- * @since 2.0
- * @product highcharts
- *
- * @private
- */
- threshold: 0,
- /**
- * The width of the border surrounding each column or bar. Defaults to
- * `1` when there is room for a border, but to `0` when the columns are
- * so dense that a border would cover the next column.
- *
- * In styled mode, the stroke width can be set with the
- * `.highcharts-point` rule.
- *
- * @sample {highcharts} highcharts/plotoptions/column-borderwidth/
- * 2px black border
- *
- * @type {number}
- * @default undefined
- * @product highcharts highstock gantt
- * @apioption plotOptions.column.borderWidth
- */
- /**
- * The color of the border surrounding each column or bar.
- *
- * In styled mode, the border stroke can be set with the
- * `.highcharts-point` rule.
- *
- * @sample {highcharts} highcharts/plotoptions/column-bordercolor/
- * Dark gray border
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #ffffff
- * @product highcharts highstock gantt
- *
- * @private
- */
- borderColor: '#ffffff'
- },
- /**
- * @lends seriesTypes.column.prototype
- */
- {
- cropShoulder: 0,
- // When tooltip is not shared, this series (and derivatives) requires
- // direct touch/hover. KD-tree does not apply.
- directTouch: true,
- trackerGroups: ['group', 'dataLabelsGroup'],
- // use separate negative stacks, unlike area stacks where a negative
- // point is substracted from previous (#1910)
- negStacks: true,
- /* eslint-disable valid-jsdoc */
- /**
- * Initialize the series. Extends the basic Series.init method by
- * marking other series of the same type as dirty.
- *
- * @private
- * @function Highcharts.seriesTypes.column#init
- * @return {void}
- */
- init: function () {
- Series.prototype.init.apply(this, arguments);
- var series = this,
- chart = series.chart;
- // if the series is added dynamically, force redraw of other
- // series affected by a new column
- if (chart.hasRendered) {
- chart.series.forEach(function (otherSeries) {
- if (otherSeries.type === series.type) {
- otherSeries.isDirty = true;
- }
- });
- }
- },
- /**
- * Return the width and x offset of the columns adjusted for grouping,
- * groupPadding, pointPadding, pointWidth etc.
- *
- * @private
- * @function Highcharts.seriesTypes.column#getColumnMetrics
- * @return {Highcharts.ColumnMetricsObject}
- */
- getColumnMetrics: function () {
- var series = this,
- options = series.options,
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- reversedStacks = xAxis.options.reversedStacks,
- // Keep backward compatibility: reversed xAxis had reversed
- // stacks
- reverseStacks = (xAxis.reversed && !reversedStacks) ||
- (!xAxis.reversed && reversedStacks),
- stackKey,
- stackGroups = {},
- columnCount = 0;
- // Get the total number of column type series. This is called on
- // every series. Consider moving this logic to a chart.orderStacks()
- // function and call it on init, addSeries and removeSeries
- if (options.grouping === false) {
- columnCount = 1;
- }
- else {
- series.chart.series.forEach(function (otherSeries) {
- var otherYAxis = otherSeries.yAxis,
- otherOptions = otherSeries.options,
- columnIndex;
- if (otherSeries.type === series.type &&
- (otherSeries.visible ||
- !series.chart.options.chart
- .ignoreHiddenSeries) &&
- yAxis.len === otherYAxis.len &&
- yAxis.pos === otherYAxis.pos) { // #642, #2086
- if (otherOptions.stacking && otherOptions.stacking !== 'group') {
- stackKey = otherSeries.stackKey;
- if (typeof stackGroups[stackKey] ===
- 'undefined') {
- stackGroups[stackKey] = columnCount++;
- }
- columnIndex = stackGroups[stackKey];
- }
- else if (otherOptions.grouping !== false) { // #1162
- columnIndex = columnCount++;
- }
- otherSeries.columnIndex = columnIndex;
- }
- });
- }
- var categoryWidth = Math.min(Math.abs(xAxis.transA) * ((xAxis.ordinal && xAxis.ordinal.slope) ||
- options.pointRange ||
- xAxis.closestPointRange ||
- xAxis.tickInterval ||
- 1), // #2610
- xAxis.len // #1535
- ),
- groupPadding = categoryWidth * options.groupPadding,
- groupWidth = categoryWidth - 2 * groupPadding,
- pointOffsetWidth = groupWidth / (columnCount || 1),
- pointWidth = Math.min(options.maxPointWidth || xAxis.len,
- pick(options.pointWidth,
- pointOffsetWidth * (1 - 2 * options.pointPadding))),
- pointPadding = (pointOffsetWidth - pointWidth) / 2,
- // #1251, #3737
- colIndex = (series.columnIndex || 0) + (reverseStacks ? 1 : 0),
- pointXOffset = pointPadding +
- (groupPadding +
- colIndex * pointOffsetWidth -
- (categoryWidth / 2)) * (reverseStacks ? -1 : 1);
- // Save it for reading in linked series (Error bars particularly)
- series.columnMetrics = {
- width: pointWidth,
- offset: pointXOffset,
- paddedWidth: pointOffsetWidth,
- columnCount: columnCount
- };
- return series.columnMetrics;
- },
- /**
- * Make the columns crisp. The edges are rounded to the nearest full
- * pixel.
- *
- * @private
- * @function Highcharts.seriesTypes.column#crispCol
- * @param {number} x
- * @param {number} y
- * @param {number} w
- * @param {number} h
- * @return {Highcharts.BBoxObject}
- */
- crispCol: function (x, y, w, h) {
- var chart = this.chart,
- borderWidth = this.borderWidth,
- xCrisp = -(borderWidth % 2 ? 0.5 : 0),
- yCrisp = borderWidth % 2 ? 0.5 : 1,
- right,
- bottom,
- fromTop;
- if (chart.inverted && chart.renderer.isVML) {
- yCrisp += 1;
- }
- // Horizontal. We need to first compute the exact right edge, then
- // round it and compute the width from there.
- if (this.options.crisp) {
- right = Math.round(x + w) + xCrisp;
- x = Math.round(x) + xCrisp;
- w = right - x;
- }
- // Vertical
- bottom = Math.round(y + h) + yCrisp;
- fromTop = Math.abs(y) <= 0.5 && bottom > 0.5; // #4504, #4656
- y = Math.round(y) + yCrisp;
- h = bottom - y;
- // Top edges are exceptions
- if (fromTop && h) { // #5146
- y -= 1;
- h += 1;
- }
- return {
- x: x,
- y: y,
- width: w,
- height: h
- };
- },
- /**
- * Adjust for missing columns, according to the `centerInCategory`
- * option. Missing columns are either single points or stacks where the
- * point or points are either missing or null.
- *
- * @private
- * @function Highcharts.seriesTypes.column#adjustForMissingColumns
- * @param {number} x
- * The x coordinate of the column, left side
- * @param {number} pointWidth
- * The pointWidth, already computed upstream
- * @param {Highcharts.ColumnPoint} point
- * The point instance
- * @param {Highcharts.ColumnMetricsObject} metrics
- * The series-wide column metrics
- * @return {number}
- * The adjusted x position, or the original if not adjusted
- */
- adjustForMissingColumns: function (x, pointWidth, point, metrics) {
- var _this = this;
- var stacking = this.options.stacking;
- if (!point.isNull && metrics.columnCount > 1) {
- var indexInCategory_1 = 0;
- var totalInCategory_1 = 0;
- // Loop over all the stacks on the Y axis. When stacking is
- // enabled, these are real point stacks. When stacking is not
- // enabled, but `centerInCategory` is true, there is one stack
- // handling the grouping of points in each category. This is
- // done in the `setGroupedPoints` function.
- objectEach(this.yAxis.stacking && this.yAxis.stacking.stacks, function (stack) {
- if (typeof point.x === 'number') {
- var stackItem = stack[point.x.toString()];
- if (stackItem) {
- var pointValues = stackItem.points[_this.index],
- total = stackItem.total;
- // If true `stacking` is enabled, count the
- // total number of non-null stacks in the
- // category, and note which index this point is
- // within those stacks.
- if (stacking) {
- if (pointValues) {
- indexInCategory_1 = totalInCategory_1;
- }
- if (stackItem.hasValidPoints) {
- totalInCategory_1++;
- }
- // If `stacking` is not enabled, look for the
- // index and total of the `group` stack.
- }
- else if (H.isArray(pointValues)) {
- indexInCategory_1 = pointValues[1];
- totalInCategory_1 = total || 0;
- }
- }
- }
- });
- // Compute the adjusted x position
- var boxWidth = (totalInCategory_1 - 1) * metrics.paddedWidth +
- pointWidth;
- x = (point.plotX || 0) + boxWidth / 2 - pointWidth -
- indexInCategory_1 * metrics.paddedWidth;
- }
- return x;
- },
- /**
- * Translate each point to the plot area coordinate system and find
- * shape positions
- *
- * @private
- * @function Highcharts.seriesTypes.column#translate
- */
- translate: function () {
- var series = this,
- chart = series.chart,
- options = series.options,
- dense = series.dense =
- series.closestPointRange * series.xAxis.transA < 2,
- borderWidth = series.borderWidth = pick(options.borderWidth,
- dense ? 0 : 1 // #3635
- ),
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- threshold = options.threshold,
- translatedThreshold = series.translatedThreshold =
- yAxis.getThreshold(threshold),
- minPointLength = pick(options.minPointLength, 5),
- metrics = series.getColumnMetrics(),
- seriesPointWidth = metrics.width,
- // postprocessed for border width
- seriesBarW = series.barW =
- Math.max(seriesPointWidth, 1 + 2 * borderWidth),
- seriesXOffset = series.pointXOffset = metrics.offset,
- dataMin = series.dataMin,
- dataMax = series.dataMax;
- if (chart.inverted) {
- translatedThreshold -= 0.5; // #3355
- }
- // When the pointPadding is 0, we want the columns to be packed
- // tightly, so we allow individual columns to have individual sizes.
- // When pointPadding is greater, we strive for equal-width columns
- // (#2694).
- if (options.pointPadding) {
- seriesBarW = Math.ceil(seriesBarW);
- }
- Series.prototype.translate.apply(series);
- // Record the new values
- series.points.forEach(function (point) {
- var yBottom = pick(point.yBottom,
- translatedThreshold),
- safeDistance = 999 + Math.abs(yBottom),
- pointWidth = seriesPointWidth,
- plotX = point.plotX || 0,
- // Don't draw too far outside plot area (#1303, #2241,
- // #4264)
- plotY = clamp(point.plotY, -safeDistance,
- yAxis.len + safeDistance),
- barX = plotX + seriesXOffset,
- barW = seriesBarW,
- barY = Math.min(plotY,
- yBottom),
- up,
- barH = Math.max(plotY,
- yBottom) - barY;
- // Handle options.minPointLength
- if (minPointLength && Math.abs(barH) < minPointLength) {
- barH = minPointLength;
- up = (!yAxis.reversed && !point.negative) ||
- (yAxis.reversed && point.negative);
- // Reverse zeros if there's no positive value in the series
- // in visible range (#7046)
- if (isNumber(threshold) &&
- isNumber(dataMax) &&
- point.y === threshold &&
- dataMax <= threshold &&
- // and if there's room for it (#7311)
- (yAxis.min || 0) < threshold &&
- // if all points are the same value (i.e zero) not draw
- // as negative points (#10646)
- dataMin !== dataMax) {
- up = !up;
- }
- // If stacked...
- barY = (Math.abs(barY - translatedThreshold) > minPointLength ?
- // ...keep position
- yBottom - minPointLength :
- // #1485, #4051
- translatedThreshold -
- (up ? minPointLength : 0));
- }
- // Handle point.options.pointWidth
- // @todo Handle grouping/stacking too. Calculate offset properly
- if (defined(point.options.pointWidth)) {
- pointWidth = barW =
- Math.ceil(point.options.pointWidth);
- barX -= Math.round((pointWidth - seriesPointWidth) / 2);
- }
- // Adjust for null or missing points
- if (options.centerInCategory) {
- barX = series.adjustForMissingColumns(barX, pointWidth, point, metrics);
- }
- // Cache for access in polar
- point.barX = barX;
- point.pointWidth = pointWidth;
- // Fix the tooltip on center of grouped columns (#1216, #424,
- // #3648)
- point.tooltipPos = chart.inverted ?
- [
- yAxis.len + yAxis.pos - chart.plotLeft - plotY,
- xAxis.len + xAxis.pos - chart.plotTop - (plotX || 0) - seriesXOffset - barW / 2,
- barH
- ] :
- [barX + barW / 2, plotY + yAxis.pos -
- chart.plotTop, barH];
- // Register shape type and arguments to be used in drawPoints
- // Allow shapeType defined on pointClass level
- point.shapeType =
- series.pointClass.prototype.shapeType || 'rect';
- point.shapeArgs = series.crispCol.apply(series, point.isNull ?
- // #3169, drilldown from null must have a position to work
- // from #6585, dataLabel should be placed on xAxis, not
- // floating in the middle of the chart
- [barX, translatedThreshold, barW, 0] :
- [barX, barY, barW, barH]);
- });
- },
- getSymbol: noop,
- /**
- * Use a solid rectangle like the area series types
- *
- * @private
- * @function Highcharts.seriesTypes.column#drawLegendSymbol
- *
- * @param {Highcharts.Legend} legend
- * The legend object
- *
- * @param {Highcharts.Series|Highcharts.Point} item
- * The series (this) or point
- */
- drawLegendSymbol: LegendSymbolMixin.drawRectangle,
- /**
- * Columns have no graph
- *
- * @private
- * @function Highcharts.seriesTypes.column#drawGraph
- */
- drawGraph: function () {
- this.group[this.dense ? 'addClass' : 'removeClass']('highcharts-dense-data');
- },
- /**
- * Get presentational attributes
- *
- * @private
- * @function Highcharts.seriesTypes.column#pointAttribs
- *
- * @param {Highcharts.ColumnPoint} point
- *
- * @param {string} state
- *
- * @return {Highcharts.SVGAttributes}
- */
- pointAttribs: function (point, state) {
- var options = this.options, stateOptions, ret, p2o = this.pointAttrToOptions || {}, strokeOption = p2o.stroke || 'borderColor', strokeWidthOption = p2o['stroke-width'] || 'borderWidth', fill = (point && point.color) || this.color,
- // set to fill when borderColor null:
- stroke = ((point && point[strokeOption]) ||
- options[strokeOption] ||
- this.color ||
- fill), strokeWidth = (point && point[strokeWidthOption]) ||
- options[strokeWidthOption] ||
- this[strokeWidthOption] || 0, dashstyle = (point && point.options.dashStyle) || options.dashStyle, opacity = pick(point && point.opacity, options.opacity, 1), zone, brightness;
- // Handle zone colors
- if (point && this.zones.length) {
- zone = point.getZone();
- // When zones are present, don't use point.color (#4267).
- // Changed order (#6527), added support for colorAxis (#10670)
- fill = (point.options.color ||
- (zone && (zone.color || point.nonZonedColor)) ||
- this.color);
- if (zone) {
- stroke = zone.borderColor || stroke;
- dashstyle = zone.dashStyle || dashstyle;
- strokeWidth = zone.borderWidth || strokeWidth;
- }
- }
- // Select or hover states
- if (state && point) {
- stateOptions = merge(options.states[state],
- // #6401
- point.options.states &&
- point.options.states[state] ||
- {});
- brightness = stateOptions.brightness;
- fill =
- stateOptions.color || (typeof brightness !== 'undefined' &&
- color(fill)
- .brighten(stateOptions.brightness)
- .get()) || fill;
- stroke = stateOptions[strokeOption] || stroke;
- strokeWidth =
- stateOptions[strokeWidthOption] || strokeWidth;
- dashstyle = stateOptions.dashStyle || dashstyle;
- opacity = pick(stateOptions.opacity, opacity);
- }
- ret = {
- fill: fill,
- stroke: stroke,
- 'stroke-width': strokeWidth,
- opacity: opacity
- };
- if (dashstyle) {
- ret.dashstyle = dashstyle;
- }
- return ret;
- },
- /**
- * Draw the columns. For bars, the series.group is rotated, so the same
- * coordinates apply for columns and bars. This method is inherited by
- * scatter series.
- *
- * @private
- * @function Highcharts.seriesTypes.column#drawPoints
- */
- drawPoints: function () {
- var series = this,
- chart = this.chart,
- options = series.options,
- renderer = chart.renderer,
- animationLimit = options.animationLimit || 250,
- shapeArgs;
- // draw the columns
- series.points.forEach(function (point) {
- var plotY = point.plotY,
- graphic = point.graphic,
- hasGraphic = !!graphic,
- verb = graphic && chart.pointCount < animationLimit ?
- 'animate' : 'attr';
- if (isNumber(plotY) && point.y !== null) {
- shapeArgs = point.shapeArgs;
- // When updating a series between 2d and 3d or cartesian and
- // polar, the shape type changes.
- if (graphic && point.hasNewShapeType()) {
- graphic = graphic.destroy();
- }
- // Set starting position for point sliding animation.
- if (series.enabledDataSorting) {
- point.startXPos = series.xAxis.reversed ?
- -(shapeArgs ? shapeArgs.width : 0) :
- series.xAxis.width;
- }
- if (!graphic) {
- point.graphic = graphic =
- renderer[point.shapeType](shapeArgs)
- .add(point.group || series.group);
- if (graphic &&
- series.enabledDataSorting &&
- chart.hasRendered &&
- chart.pointCount < animationLimit) {
- graphic.attr({
- x: point.startXPos
- });
- hasGraphic = true;
- verb = 'animate';
- }
- }
- if (graphic && hasGraphic) { // update
- graphic[verb](merge(shapeArgs));
- }
- // Border radius is not stylable (#6900)
- if (options.borderRadius) {
- graphic[verb]({
- r: options.borderRadius
- });
- }
- // Presentational
- if (!chart.styledMode) {
- graphic[verb](series.pointAttribs(point, (point.selected && 'select')))
- .shadow(point.allowShadow !== false && options.shadow, null, options.stacking && !options.borderRadius);
- }
- graphic.addClass(point.getClassName(), true);
- }
- else if (graphic) {
- point.graphic = graphic.destroy(); // #1269
- }
- });
- },
- /**
- * Animate the column heights one by one from zero.
- *
- * @private
- * @function Highcharts.seriesTypes.column#animate
- *
- * @param {boolean} init
- * Whether to initialize the animation or run it
- */
- animate: function (init) {
- var series = this,
- yAxis = this.yAxis,
- options = series.options,
- inverted = this.chart.inverted,
- attr = {},
- translateProp = inverted ? 'translateX' : 'translateY',
- translateStart,
- translatedThreshold;
- if (init) {
- attr.scaleY = 0.001;
- translatedThreshold = clamp(yAxis.toPixels(options.threshold), yAxis.pos, yAxis.pos + yAxis.len);
- if (inverted) {
- attr.translateX = translatedThreshold - yAxis.len;
- }
- else {
- attr.translateY = translatedThreshold;
- }
- // apply finnal clipping (used in Highstock) (#7083)
- // animation is done by scaleY, so cliping is for panes
- if (series.clipBox) {
- series.setClip();
- }
- series.group.attr(attr);
- }
- else { // run the animation
- translateStart = series.group.attr(translateProp);
- series.group.animate({ scaleY: 1 }, extend(animObject(series.options.animation), {
- // Do the scale synchronously to ensure smooth
- // updating (#5030, #7228)
- step: function (val, fx) {
- if (series.group) {
- attr[translateProp] = translateStart +
- fx.pos * (yAxis.pos - translateStart);
- series.group.attr(attr);
- }
- }
- }));
- }
- },
- /**
- * Remove this series from the chart
- *
- * @private
- * @function Highcharts.seriesTypes.column#remove
- */
- remove: function () {
- var series = this,
- chart = series.chart;
- // column and bar series affects other series of the same type
- // as they are either stacked or grouped
- if (chart.hasRendered) {
- chart.series.forEach(function (otherSeries) {
- if (otherSeries.type === series.type) {
- otherSeries.isDirty = true;
- }
- });
- }
- Series.prototype.remove.apply(series, arguments);
- }
- });
- /* eslint-enable valid-jsdoc */
- /**
- * A `column` series. If the [type](#series.column.type) option is
- * not specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.column
- * @excluding connectNulls, dataParser, dataURL, gapSize, gapUnit, linecap,
- * lineWidth, marker, connectEnds, step
- * @product highcharts highstock
- * @apioption series.column
- */
- /**
- * An array of data points for the series. For the `column` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 6],
- * [1, 2],
- * [2, 6]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.column.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * x: 1,
- * y: 9,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 6,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.line.data
- * @excluding marker
- * @product highcharts highstock
- * @apioption series.column.data
- */
- /**
- * The color of the border surrounding the column or bar.
- *
- * In styled mode, the border stroke can be set with the `.highcharts-point`
- * rule.
- *
- * @sample {highcharts} highcharts/plotoptions/column-bordercolor/
- * Dark gray border
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock
- * @apioption series.column.data.borderColor
- */
- /**
- * The width of the border surrounding the column or bar.
- *
- * In styled mode, the stroke width can be set with the `.highcharts-point`
- * rule.
- *
- * @sample {highcharts} highcharts/plotoptions/column-borderwidth/
- * 2px black border
- *
- * @type {number}
- * @product highcharts highstock
- * @apioption series.column.data.borderWidth
- */
- /**
- * A name for the dash style to use for the column or bar. Overrides
- * dashStyle on the series.
- *
- * In styled mode, the stroke dash-array can be set with the same classes as
- * listed under [data.color](#series.column.data.color).
- *
- * @see [series.pointWidth](#plotOptions.column.dashStyle)
- *
- * @type {Highcharts.DashStyleValue}
- * @apioption series.column.data.dashStyle
- */
- /**
- * A pixel value specifying a fixed width for the column or bar. Overrides
- * pointWidth on the series. The width effects the dimension that is not based
- * on the point value.
- *
- * @see [series.pointWidth](#plotOptions.column.pointWidth)
- *
- * @type {number}
- * @apioption series.column.data.pointWidth
- */
- /**
- * @excluding halo, lineWidth, lineWidthPlus, marker
- * @product highcharts highstock
- * @apioption series.column.states.hover
- */
- /**
- * @excluding halo, lineWidth, lineWidthPlus, marker
- * @product highcharts highstock
- * @apioption series.column.states.select
- */
- ''; // includes above doclets in transpilat
- });
- _registerModule(_modules, 'Series/BarSeries.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var seriesType = U.seriesType;
- /**
- * Bar series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.bar
- *
- * @augments Highcharts.Series
- */
- seriesType('bar', 'column',
- /**
- * A bar series is a special type of column series where the columns are
- * horizontal.
- *
- * @sample highcharts/demo/bar-basic/
- * Bar chart
- *
- * @extends plotOptions.column
- * @product highcharts
- * @apioption plotOptions.bar
- */
- /**
- * @ignore
- */
- null, {
- inverted: true
- });
- /**
- * A `bar` series. If the [type](#series.bar.type) option is not specified,
- * it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.bar
- * @excluding connectNulls, dashStyle, dataParser, dataURL, gapSize, gapUnit,
- * linecap, lineWidth, marker, connectEnds, step
- * @product highcharts
- * @apioption series.bar
- */
- /**
- * An array of data points for the series. For the `bar` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 5],
- * [1, 10],
- * [2, 3]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.bar.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * x: 1,
- * y: 1,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 10,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.column.data
- * @product highcharts
- * @apioption series.bar.data
- */
- /**
- * @excluding halo,lineWidth,lineWidthPlus,marker
- * @product highcharts highstock
- * @apioption series.bar.states.hover
- */
- /**
- * @excluding halo,lineWidth,lineWidthPlus,marker
- * @product highcharts highstock
- * @apioption series.bar.states.select
- */
- ''; // gets doclets above into transpilat
- });
- _registerModule(_modules, 'Series/ScatterSeries.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- seriesType = U.seriesType;
- var Series = H.Series;
- /**
- * Scatter series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.scatter
- *
- * @augments Highcharts.Series
- */
- seriesType('scatter', 'line',
- /**
- * A scatter plot uses cartesian coordinates to display values for two
- * variables for a set of data.
- *
- * @sample {highcharts} highcharts/demo/scatter/
- * Scatter plot
- *
- * @extends plotOptions.line
- * @excluding cropThreshold, pointPlacement, shadow, useOhlcData
- * @product highcharts highstock
- * @optionparent plotOptions.scatter
- */
- {
- /**
- * The width of the line connecting the data points.
- *
- * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-none/
- * 0 by default
- * @sample {highcharts} highcharts/plotoptions/scatter-linewidth-1/
- * 1px
- *
- * @product highcharts highstock
- */
- lineWidth: 0,
- findNearestPointBy: 'xy',
- /**
- * Apply a jitter effect for the rendered markers. When plotting
- * discrete values, a little random noise may help telling the points
- * apart. The jitter setting applies a random displacement of up to `n`
- * axis units in either direction. So for example on a horizontal X
- * axis, setting the `jitter.x` to 0.24 will render the point in a
- * random position between 0.24 units to the left and 0.24 units to the
- * right of the true axis position. On a category axis, setting it to
- * 0.5 will fill up the bin and make the data appear continuous.
- *
- * When rendered on top of a box plot or a column series, a jitter value
- * of 0.24 will correspond to the underlying series' default
- * [groupPadding](
- * https://api.highcharts.com/highcharts/plotOptions.column.groupPadding)
- * and [pointPadding](
- * https://api.highcharts.com/highcharts/plotOptions.column.pointPadding)
- * settings.
- *
- * @sample {highcharts} highcharts/series-scatter/jitter
- * Jitter on a scatter plot
- *
- * @sample {highcharts} highcharts/series-scatter/jitter-boxplot
- * Jittered scatter plot on top of a box plot
- *
- * @product highcharts highstock
- * @since 7.0.2
- */
- jitter: {
- /**
- * The maximal X offset for the random jitter effect.
- */
- x: 0,
- /**
- * The maximal Y offset for the random jitter effect.
- */
- y: 0
- },
- marker: {
- enabled: true // Overrides auto-enabling in line series (#3647)
- },
- /**
- * Sticky tracking of mouse events. When true, the `mouseOut` event
- * on a series isn't triggered until the mouse moves over another
- * series, or out of the plot area. When false, the `mouseOut` event on
- * a series is triggered when the mouse leaves the area around the
- * series' graph or markers. This also implies the tooltip. When
- * `stickyTracking` is false and `tooltip.shared` is false, the tooltip
- * will be hidden when moving the mouse between series.
- *
- * @type {boolean}
- * @default false
- * @product highcharts highstock
- * @apioption plotOptions.scatter.stickyTracking
- */
- /**
- * A configuration object for the tooltip rendering of each single
- * series. Properties are inherited from [tooltip](#tooltip).
- * Overridable properties are `headerFormat`, `pointFormat`,
- * `yDecimals`, `xDateFormat`, `yPrefix` and `ySuffix`. Unlike other
- * series, in a scatter plot the series.name by default shows in the
- * headerFormat and point.x and point.y in the pointFormat.
- *
- * @product highcharts highstock
- */
- tooltip: {
- headerFormat: '<span style="color:{point.color}">\u25CF</span> ' +
- '<span style="font-size: 10px"> {series.name}</span><br/>',
- pointFormat: 'x: <b>{point.x}</b><br/>y: <b>{point.y}</b><br/>'
- }
- // Prototype members
- }, {
- sorted: false,
- requireSorting: false,
- noSharedTooltip: true,
- trackerGroups: ['group', 'markerGroup', 'dataLabelsGroup'],
- takeOrdinalPosition: false,
- /* eslint-disable valid-jsdoc */
- /**
- * @private
- * @function Highcharts.seriesTypes.scatter#drawGraph
- */
- drawGraph: function () {
- if (this.options.lineWidth) {
- Series.prototype.drawGraph.call(this);
- }
- },
- // Optionally add the jitter effect
- applyJitter: function () {
- var series = this,
- jitter = this.options.jitter,
- len = this.points.length;
- /**
- * Return a repeatable, pseudo-random number based on an integer
- * seed.
- * @private
- */
- function unrandom(seed) {
- var rand = Math.sin(seed) * 10000;
- return rand - Math.floor(rand);
- }
- if (jitter) {
- this.points.forEach(function (point, i) {
- ['x', 'y'].forEach(function (dim, j) {
- var axis,
- plotProp = 'plot' + dim.toUpperCase(),
- min,
- max,
- translatedJitter;
- if (jitter[dim] && !point.isNull) {
- axis = series[dim + 'Axis'];
- translatedJitter =
- jitter[dim] * axis.transA;
- if (axis && !axis.isLog) {
- // Identify the outer bounds of the jitter range
- min = Math.max(0, point[plotProp] - translatedJitter);
- max = Math.min(axis.len, point[plotProp] + translatedJitter);
- // Find a random position within this range
- point[plotProp] = min +
- (max - min) * unrandom(i + j * len);
- // Update clientX for the tooltip k-d-tree
- if (dim === 'x') {
- point.clientX = point.plotX;
- }
- }
- }
- });
- });
- }
- }
- /* eslint-enable valid-jsdoc */
- });
- /* eslint-disable no-invalid-this */
- addEvent(Series, 'afterTranslate', function () {
- if (this.applyJitter) {
- this.applyJitter();
- }
- });
- /* eslint-enable no-invalid-this */
- /**
- * A `scatter` series. If the [type](#series.scatter.type) option is
- * not specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.scatter
- * @excluding cropThreshold, dataParser, dataURL, useOhlcData
- * @product highcharts highstock
- * @apioption series.scatter
- */
- /**
- * An array of data points for the series. For the `scatter` series
- * type, points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. The `x` values will be automatically
- * calculated, either starting at 0 and incremented by 1, or from
- * `pointStart` and `pointInterval` given in the series options. If the axis
- * has categories, these will be used. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of arrays with 2 values. In this case, the values correspond to
- * `x,y`. If the first value is a string, it is applied as the name of the
- * point, and the `x` value is inferred.
- * ```js
- * data: [
- * [0, 0],
- * [1, 8],
- * [2, 9]
- * ]
- * ```
- *
- * 3. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.scatter.turboThreshold), this option is not
- * available.
- * ```js
- * data: [{
- * x: 1,
- * y: 2,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * x: 1,
- * y: 4,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<(number|string),(number|null)>|null|*>}
- * @extends series.line.data
- * @product highcharts highstock
- * @apioption series.scatter.data
- */
- ''; // adds doclets above to transpilat
- });
- _registerModule(_modules, 'Mixins/CenteredSeries.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * @private
- * @interface Highcharts.RadianAngles
- */ /**
- * @name Highcharts.RadianAngles#end
- * @type {number}
- */ /**
- * @name Highcharts.RadianAngles#start
- * @type {number}
- */
- var isNumber = U.isNumber,
- pick = U.pick,
- relativeLength = U.relativeLength;
- var deg2rad = H.deg2rad;
- /* eslint-disable valid-jsdoc */
- /**
- * @private
- * @mixin Highcharts.CenteredSeriesMixin
- */
- var centeredSeriesMixin = H.CenteredSeriesMixin = {
- /**
- * Get the center of the pie based on the size and center options relative
- * to the plot area. Borrowed by the polar and gauge series types.
- *
- * @private
- * @function Highcharts.CenteredSeriesMixin.getCenter
- *
- * @return {Array<number>}
- */
- getCenter: function () {
- var options = this.options,
- chart = this.chart,
- slicingRoom = 2 * (options.slicedOffset || 0),
- handleSlicingRoom,
- plotWidth = chart.plotWidth - 2 * slicingRoom,
- plotHeight = chart.plotHeight - 2 * slicingRoom,
- centerOption = options.center,
- smallestSize = Math.min(plotWidth,
- plotHeight),
- size = options.size,
- innerSize = options.innerSize || 0,
- positions,
- i,
- value;
- if (typeof size === 'string') {
- size = parseFloat(size);
- }
- if (typeof innerSize === 'string') {
- innerSize = parseFloat(innerSize);
- }
- positions = [
- pick(centerOption[0], '50%'),
- pick(centerOption[1], '50%'),
- // Prevent from negative values
- pick(size && size < 0 ? void 0 : options.size, '100%'),
- pick(innerSize && innerSize < 0 ? void 0 : options.innerSize || 0, '0%')
- ];
- // No need for inner size in angular (gauges) series but still required
- // for pie series
- if (chart.angular && !(this instanceof H.Series)) {
- positions[3] = 0;
- }
- for (i = 0; i < 4; ++i) {
- value = positions[i];
- handleSlicingRoom = i < 2 || (i === 2 && /%$/.test(value));
- // i == 0: centerX, relative to width
- // i == 1: centerY, relative to height
- // i == 2: size, relative to smallestSize
- // i == 3: innerSize, relative to size
- positions[i] = relativeLength(value, [plotWidth, plotHeight, smallestSize, positions[2]][i]) + (handleSlicingRoom ? slicingRoom : 0);
- }
- // innerSize cannot be larger than size (#3632)
- if (positions[3] > positions[2]) {
- positions[3] = positions[2];
- }
- return positions;
- },
- /**
- * getStartAndEndRadians - Calculates start and end angles in radians.
- * Used in series types such as pie and sunburst.
- *
- * @private
- * @function Highcharts.CenteredSeriesMixin.getStartAndEndRadians
- *
- * @param {number} [start]
- * Start angle in degrees.
- *
- * @param {number} [end]
- * Start angle in degrees.
- *
- * @return {Highcharts.RadianAngles}
- * Returns an object containing start and end angles as radians.
- */
- getStartAndEndRadians: function (start, end) {
- var startAngle = isNumber(start) ? start : 0, // must be a number
- endAngle = ((isNumber(end) && // must be a number
- end > startAngle && // must be larger than the start angle
- // difference must be less than 360 degrees
- (end - startAngle) < 360) ?
- end :
- startAngle + 360),
- correction = -90;
- return {
- start: deg2rad * (startAngle + correction),
- end: deg2rad * (endAngle + correction)
- };
- }
- };
- return centeredSeriesMixin;
- });
- _registerModule(_modules, 'Series/PieSeries.js', [_modules['Core/Globals.js'], _modules['Core/Renderer/SVG/SVGRenderer.js'], _modules['Mixins/LegendSymbol.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js'], _modules['Mixins/CenteredSeries.js']], function (H, SVGRenderer, LegendSymbolMixin, Point, U, centeredSeriesMixin) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- clamp = U.clamp,
- defined = U.defined,
- fireEvent = U.fireEvent,
- isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- relativeLength = U.relativeLength,
- seriesType = U.seriesType,
- setAnimation = U.setAnimation;
- var getStartAndEndRadians = centeredSeriesMixin.getStartAndEndRadians,
- noop = H.noop,
- Series = H.Series,
- seriesTypes = H.seriesTypes;
- /**
- * Pie series type.
- *
- * @private
- * @class
- * @name Highcharts.seriesTypes.pie
- *
- * @augments Highcharts.Series
- */
- seriesType('pie', 'line',
- /**
- * A pie chart is a circular graphic which is divided into slices to
- * illustrate numerical proportion.
- *
- * @sample highcharts/demo/pie-basic/
- * Pie chart
- *
- * @extends plotOptions.line
- * @excluding animationLimit, boostThreshold, connectEnds, connectNulls,
- * cropThreshold, dashStyle, dataSorting, dragDrop,
- * findNearestPointBy, getExtremesFromAll, label, lineWidth,
- * marker, negativeColor, pointInterval, pointIntervalUnit,
- * pointPlacement, pointStart, softThreshold, stacking, step,
- * threshold, turboThreshold, zoneAxis, zones, dataSorting,
- * boostBlending
- * @product highcharts
- * @optionparent plotOptions.pie
- */
- {
- /**
- * @excluding legendItemClick
- * @apioption plotOptions.pie.events
- */
- /**
- * Fires when the checkbox next to the point name in the legend is
- * clicked. One parameter, event, is passed to the function. The state
- * of the checkbox is found by event.checked. The checked item is found
- * by event.item. Return false to prevent the default action which is to
- * toggle the select state of the series.
- *
- * @sample {highcharts} highcharts/plotoptions/series-events-checkboxclick/
- * Alert checkbox status
- *
- * @type {Function}
- * @since 1.2.0
- * @product highcharts
- * @context Highcharts.Point
- * @apioption plotOptions.pie.events.checkboxClick
- */
- /**
- * Fires when the legend item belonging to the pie point (slice) is
- * clicked. The `this` keyword refers to the point itself. One
- * parameter, `event`, is passed to the function, containing common
- * event information. The default action is to toggle the visibility of
- * the point. This can be prevented by calling `event.preventDefault()`.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-point-events-legenditemclick/
- * Confirm toggle visibility
- *
- * @type {Highcharts.PointLegendItemClickCallbackFunction}
- * @since 1.2.0
- * @product highcharts
- * @apioption plotOptions.pie.point.events.legendItemClick
- */
- /**
- * The center of the pie chart relative to the plot area. Can be
- * percentages or pixel values. The default behaviour (as of 3.0) is to
- * center the pie so that all slices and data labels are within the plot
- * area. As a consequence, the pie may actually jump around in a chart
- * with dynamic values, as the data labels move. In that case, the
- * center should be explicitly set, for example to `["50%", "50%"]`.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-center/
- * Centered at 100, 100
- *
- * @type {Array<(number|string|null),(number|string|null)>}
- * @default [null, null]
- * @product highcharts
- *
- * @private
- */
- center: [null, null],
- /**
- * The color of the pie series. A pie series is represented as an empty
- * circle if the total sum of its values is 0. Use this property to
- * define the color of its border.
- *
- * In styled mode, the color can be defined by the
- * [colorIndex](#plotOptions.series.colorIndex) option. Also, the series
- * color can be set with the `.highcharts-series`,
- * `.highcharts-color-{n}`, `.highcharts-{type}-series` or
- * `.highcharts-series-{n}` class, or individual classes given by the
- * `className` option.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/
- * Empty pie series
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #cccccc
- * @apioption plotOptions.pie.color
- */
- /**
- * @product highcharts
- *
- * @private
- */
- clip: false,
- /**
- * @ignore-option
- *
- * @private
- */
- colorByPoint: true,
- /**
- * A series specific or series type specific color set to use instead
- * of the global [colors](#colors).
- *
- * @sample {highcharts} highcharts/demo/pie-monochrome/
- * Set default colors for all pies
- *
- * @type {Array<Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject>}
- * @since 3.0
- * @product highcharts
- * @apioption plotOptions.pie.colors
- */
- /**
- * @declare Highcharts.SeriesPieDataLabelsOptionsObject
- * @extends plotOptions.series.dataLabels
- * @excluding align, allowOverlap, inside, staggerLines, step
- * @private
- */
- dataLabels: {
- /**
- * Alignment method for data labels. Possible values are:
- *
- * - `toPlotEdges`: Each label touches the nearest vertical edge of
- * the plot area.
- *
- * - `connectors`: Connectors have the same x position and the
- * widest label of each half (left & right) touches the nearest
- * vertical edge of the plot area.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-connectors/
- * alignTo: connectors
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-alignto-plotedges/
- * alignTo: plotEdges
- *
- * @type {string}
- * @since 7.0.0
- * @product highcharts
- * @apioption plotOptions.pie.dataLabels.alignTo
- */
- allowOverlap: true,
- /**
- * The color of the line connecting the data label to the pie slice.
- * The default color is the same as the point's color.
- *
- * In styled mode, the connector stroke is given in the
- * `.highcharts-data-label-connector` class.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorcolor/
- * Blue connectors
- * @sample {highcharts} highcharts/css/pie-point/
- * Styled connectors
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 2.1
- * @product highcharts
- * @apioption plotOptions.pie.dataLabels.connectorColor
- */
- /**
- * The distance from the data label to the connector. Note that
- * data labels also have a default `padding`, so in order for the
- * connector to touch the text, the `padding` must also be 0.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorpadding/
- * No padding
- *
- * @since 2.1
- * @product highcharts
- */
- connectorPadding: 5,
- /**
- * Specifies the method that is used to generate the connector path.
- * Highcharts provides 3 built-in connector shapes: `'fixedOffset'`
- * (default), `'straight'` and `'crookedLine'`. Using
- * `'crookedLine'` has the most sense (in most of the cases) when
- * `'alignTo'` is set.
- *
- * Users can provide their own method by passing a function instead
- * of a String. 3 arguments are passed to the callback:
- *
- * - Object that holds the information about the coordinates of the
- * label (`x` & `y` properties) and how the label is located in
- * relation to the pie (`alignment` property). `alignment` can by
- * one of the following:
- * `'left'` (pie on the left side of the data label),
- * `'right'` (pie on the right side of the data label) or
- * `'center'` (data label overlaps the pie).
- *
- * - Object that holds the information about the position of the
- * connector. Its `touchingSliceAt` porperty tells the position
- * of the place where the connector touches the slice.
- *
- * - Data label options
- *
- * The function has to return an SVG path definition in array form
- * (see the example).
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-string/
- * connectorShape is a String
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorshape-function/
- * connectorShape is a function
- *
- * @type {string|Function}
- * @since 7.0.0
- * @product highcharts
- */
- connectorShape: 'fixedOffset',
- /**
- * The width of the line connecting the data label to the pie slice.
- *
- * In styled mode, the connector stroke width is given in the
- * `.highcharts-data-label-connector` class.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-connectorwidth-disabled/
- * Disable the connector
- * @sample {highcharts} highcharts/css/pie-point/
- * Styled connectors
- *
- * @type {number}
- * @default 1
- * @since 2.1
- * @product highcharts
- * @apioption plotOptions.pie.dataLabels.connectorWidth
- */
- /**
- * Works only if `connectorShape` is `'crookedLine'`. It defines how
- * far from the vertical plot edge the coonnector path should be
- * crooked.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-crookdistance/
- * crookDistance set to 90%
- *
- * @since 7.0.0
- * @product highcharts
- */
- crookDistance: '70%',
- /**
- * The distance of the data label from the pie's edge. Negative
- * numbers put the data label on top of the pie slices. Can also be
- * defined as a percentage of pie's radius. Connectors are only
- * shown for data labels outside the pie.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-distance/
- * Data labels on top of the pie
- *
- * @type {number|string}
- * @since 2.1
- * @product highcharts
- */
- distance: 30,
- enabled: true,
- formatter: function () {
- return this.point.isNull ? void 0 : this.point.name;
- },
- /**
- * Whether to render the connector as a soft arc or a line with
- * sharp break. Works only if `connectorShape` equals to
- * `fixedOffset`.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-true/
- * Soft
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-softconnector-false/
- * Non soft
- *
- * @since 2.1.7
- * @product highcharts
- */
- softConnector: true,
- /**
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow
- * Long labels truncated with an ellipsis
- * @sample {highcharts} highcharts/plotoptions/pie-datalabels-overflow-wrap
- * Long labels are wrapped
- *
- * @type {Highcharts.CSSObject}
- * @apioption plotOptions.pie.dataLabels.style
- */
- x: 0
- },
- /**
- * If the total sum of the pie's values is 0, the series is represented
- * as an empty circle . The `fillColor` option defines the color of that
- * circle. Use [pie.borderWidth](#plotOptions.pie.borderWidth) to set
- * the border thickness.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-emptyseries/
- * Empty pie series
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @private
- */
- fillColor: void 0,
- /**
- * The end angle of the pie in degrees where 0 is top and 90 is right.
- * Defaults to `startAngle` plus 360.
- *
- * @sample {highcharts} highcharts/demo/pie-semi-circle/
- * Semi-circle donut
- *
- * @type {number}
- * @since 1.3.6
- * @product highcharts
- * @apioption plotOptions.pie.endAngle
- */
- /**
- * Equivalent to [chart.ignoreHiddenSeries](#chart.ignoreHiddenSeries),
- * this option tells whether the series shall be redrawn as if the
- * hidden point were `null`.
- *
- * The default value changed from `false` to `true` with Highcharts
- * 3.0.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-ignorehiddenpoint/
- * True, the hiddden point is ignored
- *
- * @since 2.3.0
- * @product highcharts
- *
- * @private
- */
- ignoreHiddenPoint: true,
- /**
- * @ignore-option
- *
- * @private
- */
- inactiveOtherPoints: true,
- /**
- * The size of the inner diameter for the pie. A size greater than 0
- * renders a donut chart. Can be a percentage or pixel value.
- * Percentages are relative to the pie size. Pixel values are given as
- * integers.
- *
- *
- * Note: in Highcharts < 4.1.2, the percentage was relative to the plot
- * area, not the pie size.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-innersize-80px/
- * 80px inner size
- * @sample {highcharts} highcharts/plotoptions/pie-innersize-50percent/
- * 50% of the plot area
- * @sample {highcharts} highcharts/demo/3d-pie-donut/
- * 3D donut
- *
- * @type {number|string}
- * @default 0
- * @since 2.0
- * @product highcharts
- * @apioption plotOptions.pie.innerSize
- */
- /**
- * @ignore-option
- *
- * @private
- */
- legendType: 'point',
- /**
- * @ignore-option
- *
- * @private
- */
- marker: null,
- /**
- * The minimum size for a pie in response to auto margins. The pie will
- * try to shrink to make room for data labels in side the plot area,
- * but only to this size.
- *
- * @type {number|string}
- * @default 80
- * @since 3.0
- * @product highcharts
- * @apioption plotOptions.pie.minSize
- */
- /**
- * The diameter of the pie relative to the plot area. Can be a
- * percentage or pixel value. Pixel values are given as integers. The
- * default behaviour (as of 3.0) is to scale to the plot area and give
- * room for data labels within the plot area.
- * [slicedOffset](#plotOptions.pie.slicedOffset) is also included in the
- * default size calculation. As a consequence, the size of the pie may
- * vary when points are updated and data labels more around. In that
- * case it is best to set a fixed value, for example `"75%"`.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-size/
- * Smaller pie
- *
- * @type {number|string|null}
- * @product highcharts
- *
- * @private
- */
- size: null,
- /**
- * Whether to display this particular series or series type in the
- * legend. Since 2.1, pies are not shown in the legend by default.
- *
- * @sample {highcharts} highcharts/plotoptions/series-showinlegend/
- * One series in the legend, one hidden
- *
- * @product highcharts
- *
- * @private
- */
- showInLegend: false,
- /**
- * If a point is sliced, moved out from the center, how many pixels
- * should it be moved?.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-slicedoffset-20/
- * 20px offset
- *
- * @product highcharts
- *
- * @private
- */
- slicedOffset: 10,
- /**
- * The start angle of the pie slices in degrees where 0 is top and 90
- * right.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-startangle-90/
- * Start from right
- *
- * @type {number}
- * @default 0
- * @since 2.3.4
- * @product highcharts
- * @apioption plotOptions.pie.startAngle
- */
- /**
- * Sticky tracking of mouse events. When true, the `mouseOut` event
- * on a series isn't triggered until the mouse moves over another
- * series, or out of the plot area. When false, the `mouseOut` event on
- * a series is triggered when the mouse leaves the area around the
- * series' graph or markers. This also implies the tooltip. When
- * `stickyTracking` is false and `tooltip.shared` is false, the tooltip
- * will be hidden when moving the mouse between series.
- *
- * @product highcharts
- *
- * @private
- */
- stickyTracking: false,
- tooltip: {
- followPointer: true
- },
- /**
- * The color of the border surrounding each slice. When `null`, the
- * border takes the same color as the slice fill. This can be used
- * together with a `borderWidth` to fill drawing gaps created by
- * antialiazing artefacts in borderless pies.
- *
- * In styled mode, the border stroke is given in the `.highcharts-point`
- * class.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-bordercolor-black/
- * Black border
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #ffffff
- * @product highcharts
- *
- * @private
- */
- borderColor: '#ffffff',
- /**
- * The width of the border surrounding each slice.
- *
- * When setting the border width to 0, there may be small gaps between
- * the slices due to SVG antialiasing artefacts. To work around this,
- * keep the border width at 0.5 or 1, but set the `borderColor` to
- * `null` instead.
- *
- * In styled mode, the border stroke width is given in the
- * `.highcharts-point` class.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-borderwidth/
- * 3px border
- *
- * @product highcharts
- *
- * @private
- */
- borderWidth: 1,
- /**
- * @ignore-options
- * @private
- */
- lineWidth: void 0,
- states: {
- /**
- * @extends plotOptions.series.states.hover
- * @excluding marker, lineWidth, lineWidthPlus
- * @product highcharts
- */
- hover: {
- /**
- * How much to brighten the point on interaction. Requires the
- * main color to be defined in hex or rgb(a) format.
- *
- * In styled mode, the hover brightness is by default replaced
- * by a fill-opacity given in the `.highcharts-point-hover`
- * class.
- *
- * @sample {highcharts} highcharts/plotoptions/pie-states-hover-brightness/
- * Brightened by 0.5
- *
- * @product highcharts
- */
- brightness: 0.1
- }
- }
- },
- /* eslint-disable valid-jsdoc */
- /**
- * @lends seriesTypes.pie.prototype
- */
- {
- isCartesian: false,
- requireSorting: false,
- directTouch: true,
- noSharedTooltip: true,
- trackerGroups: ['group', 'dataLabelsGroup'],
- axisTypes: [],
- pointAttribs: seriesTypes.column.prototype.pointAttribs,
- /**
- * Animate the pies in
- *
- * @private
- * @function Highcharts.seriesTypes.pie#animate
- *
- * @param {boolean} [init=false]
- */
- animate: function (init) {
- var series = this,
- points = series.points,
- startAngleRad = series.startAngleRad;
- if (!init) {
- points.forEach(function (point) {
- var graphic = point.graphic,
- args = point.shapeArgs;
- if (graphic && args) {
- // start values
- graphic.attr({
- // animate from inner radius (#779)
- r: pick(point.startR, (series.center && series.center[3] / 2)),
- start: startAngleRad,
- end: startAngleRad
- });
- // animate
- graphic.animate({
- r: args.r,
- start: args.start,
- end: args.end
- }, series.options.animation);
- }
- });
- }
- },
- // Define hasData function for non-cartesian series.
- // Returns true if the series has points at all.
- hasData: function () {
- return !!this.processedXData.length; // != 0
- },
- /**
- * Recompute total chart sum and update percentages of points.
- *
- * @private
- * @function Highcharts.seriesTypes.pie#updateTotals
- * @return {void}
- */
- updateTotals: function () {
- var i,
- total = 0,
- points = this.points,
- len = points.length,
- point,
- ignoreHiddenPoint = this.options.ignoreHiddenPoint;
- // Get the total sum
- for (i = 0; i < len; i++) {
- point = points[i];
- total += (ignoreHiddenPoint && !point.visible) ?
- 0 :
- point.isNull ?
- 0 :
- point.y;
- }
- this.total = total;
- // Set each point's properties
- for (i = 0; i < len; i++) {
- point = points[i];
- point.percentage =
- (total > 0 && (point.visible || !ignoreHiddenPoint)) ?
- point.y / total * 100 :
- 0;
- point.total = total;
- }
- },
- /**
- * Extend the generatePoints method by adding total and percentage
- * properties to each point
- *
- * @private
- * @function Highcharts.seriesTypes.pie#generatePoints
- * @return {void}
- */
- generatePoints: function () {
- Series.prototype.generatePoints.call(this);
- this.updateTotals();
- },
- /**
- * Utility for getting the x value from a given y, used for
- * anticollision logic in data labels. Added point for using specific
- * points' label distance.
- * @private
- */
- getX: function (y, left, point) {
- var center = this.center,
- // Variable pie has individual radius
- radius = this.radii ?
- this.radii[point.index] :
- center[2] / 2,
- angle,
- x;
- angle = Math.asin(clamp((y - center[1]) / (radius + point.labelDistance), -1, 1));
- x = center[0] +
- (left ? -1 : 1) *
- (Math.cos(angle) * (radius + point.labelDistance)) +
- (point.labelDistance > 0 ?
- (left ? -1 : 1) * this.options.dataLabels.padding :
- 0);
- return x;
- },
- /**
- * Do translation for pie slices
- *
- * @private
- * @function Highcharts.seriesTypes.pie#translate
- * @param {Array<number>} [positions]
- * @return {void}
- */
- translate: function (positions) {
- this.generatePoints();
- var series = this,
- cumulative = 0,
- precision = 1000, // issue #172
- options = series.options,
- slicedOffset = options.slicedOffset,
- connectorOffset = slicedOffset + (options.borderWidth || 0),
- finalConnectorOffset,
- start,
- end,
- angle,
- radians = getStartAndEndRadians(options.startAngle,
- options.endAngle),
- startAngleRad = series.startAngleRad = radians.start,
- endAngleRad = series.endAngleRad = radians.end,
- circ = endAngleRad - startAngleRad, // 2 * Math.PI,
- points = series.points,
- // the x component of the radius vector for a given point
- radiusX,
- radiusY,
- labelDistance = options.dataLabels.distance,
- ignoreHiddenPoint = options.ignoreHiddenPoint,
- i,
- len = points.length,
- point;
- // Get positions - either an integer or a percentage string must be
- // given. If positions are passed as a parameter, we're in a
- // recursive loop for adjusting space for data labels.
- if (!positions) {
- series.center = positions = series.getCenter();
- }
- // Calculate the geometry for each point
- for (i = 0; i < len; i++) {
- point = points[i];
- // set start and end angle
- start = startAngleRad + (cumulative * circ);
- if (!ignoreHiddenPoint || point.visible) {
- cumulative += point.percentage / 100;
- }
- end = startAngleRad + (cumulative * circ);
- // set the shape
- point.shapeType = 'arc';
- point.shapeArgs = {
- x: positions[0],
- y: positions[1],
- r: positions[2] / 2,
- innerR: positions[3] / 2,
- start: Math.round(start * precision) / precision,
- end: Math.round(end * precision) / precision
- };
- // Used for distance calculation for specific point.
- point.labelDistance = pick((point.options.dataLabels &&
- point.options.dataLabels.distance), labelDistance);
- // Compute point.labelDistance if it's defined as percentage
- // of slice radius (#8854)
- point.labelDistance = relativeLength(point.labelDistance, point.shapeArgs.r);
- // Saved for later dataLabels distance calculation.
- series.maxLabelDistance = Math.max(series.maxLabelDistance || 0, point.labelDistance);
- // The angle must stay within -90 and 270 (#2645)
- angle = (end + start) / 2;
- if (angle > 1.5 * Math.PI) {
- angle -= 2 * Math.PI;
- }
- else if (angle < -Math.PI / 2) {
- angle += 2 * Math.PI;
- }
- // Center for the sliced out slice
- point.slicedTranslation = {
- translateX: Math.round(Math.cos(angle) * slicedOffset),
- translateY: Math.round(Math.sin(angle) * slicedOffset)
- };
- // set the anchor point for tooltips
- radiusX = Math.cos(angle) * positions[2] / 2;
- radiusY = Math.sin(angle) * positions[2] / 2;
- point.tooltipPos = [
- positions[0] + radiusX * 0.7,
- positions[1] + radiusY * 0.7
- ];
- point.half = angle < -Math.PI / 2 || angle > Math.PI / 2 ?
- 1 :
- 0;
- point.angle = angle;
- // Set the anchor point for data labels. Use point.labelDistance
- // instead of labelDistance // #1174
- // finalConnectorOffset - not override connectorOffset value.
- finalConnectorOffset = Math.min(connectorOffset, point.labelDistance / 5); // #1678
- point.labelPosition = {
- natural: {
- // initial position of the data label - it's utilized for
- // finding the final position for the label
- x: positions[0] + radiusX + Math.cos(angle) *
- point.labelDistance,
- y: positions[1] + radiusY + Math.sin(angle) *
- point.labelDistance
- },
- 'final': {
- // used for generating connector path -
- // initialized later in drawDataLabels function
- // x: undefined,
- // y: undefined
- },
- // left - pie on the left side of the data label
- // right - pie on the right side of the data label
- // center - data label overlaps the pie
- alignment: point.labelDistance < 0 ?
- 'center' : point.half ? 'right' : 'left',
- connectorPosition: {
- breakAt: {
- x: positions[0] + radiusX + Math.cos(angle) *
- finalConnectorOffset,
- y: positions[1] + radiusY + Math.sin(angle) *
- finalConnectorOffset
- },
- touchingSliceAt: {
- x: positions[0] + radiusX,
- y: positions[1] + radiusY
- }
- }
- };
- }
- fireEvent(series, 'afterTranslate');
- },
- /**
- * Called internally to draw auxiliary graph in pie-like series in
- * situtation when the default graph is not sufficient enough to present
- * the data well. Auxiliary graph is saved in the same object as
- * regular graph.
- *
- * @private
- * @function Highcharts.seriesTypes.pie#drawEmpty
- */
- drawEmpty: function () {
- var centerX,
- centerY,
- start = this.startAngleRad,
- end = this.endAngleRad,
- options = this.options;
- // Draw auxiliary graph if there're no visible points.
- if (this.total === 0 && this.center) {
- centerX = this.center[0];
- centerY = this.center[1];
- if (!this.graph) {
- this.graph = this.chart.renderer
- .arc(centerX, centerY, this.center[1] / 2, 0, start, end)
- .addClass('highcharts-empty-series')
- .add(this.group);
- }
- this.graph.attr({
- d: SVGRenderer.prototype.symbols.arc(centerX, centerY, this.center[2] / 2, 0, {
- start: start,
- end: end,
- innerR: this.center[3] / 2
- })
- });
- if (!this.chart.styledMode) {
- this.graph.attr({
- 'stroke-width': options.borderWidth,
- fill: options.fillColor || 'none',
- stroke: options.color ||
- '#cccccc'
- });
- }
- }
- else if (this.graph) { // Destroy the graph object.
- this.graph = this.graph.destroy();
- }
- },
- /**
- * Draw the data points
- *
- * @private
- * @function Highcharts.seriesTypes.pie#drawPoints
- * @return {void}
- */
- redrawPoints: function () {
- var series = this,
- chart = series.chart,
- renderer = chart.renderer,
- groupTranslation,
- graphic,
- pointAttr,
- shapeArgs,
- shadow = series.options.shadow;
- this.drawEmpty();
- if (shadow && !series.shadowGroup && !chart.styledMode) {
- series.shadowGroup = renderer.g('shadow')
- .attr({ zIndex: -1 })
- .add(series.group);
- }
- // draw the slices
- series.points.forEach(function (point) {
- var animateTo = {};
- graphic = point.graphic;
- if (!point.isNull && graphic) {
- shapeArgs = point.shapeArgs;
- // If the point is sliced, use special translation, else use
- // plot area translation
- groupTranslation = point.getTranslate();
- if (!chart.styledMode) {
- // Put the shadow behind all points
- var shadowGroup = point.shadowGroup;
- if (shadow && !shadowGroup) {
- shadowGroup = point.shadowGroup = renderer
- .g('shadow')
- .add(series.shadowGroup);
- }
- if (shadowGroup) {
- shadowGroup.attr(groupTranslation);
- }
- pointAttr = series.pointAttribs(point, (point.selected && 'select'));
- }
- // Draw the slice
- if (!point.delayedRendering) {
- graphic
- .setRadialReference(series.center);
- if (!chart.styledMode) {
- merge(true, animateTo, pointAttr);
- }
- merge(true, animateTo, shapeArgs, groupTranslation);
- graphic.animate(animateTo);
- }
- else {
- graphic
- .setRadialReference(series.center)
- .attr(shapeArgs)
- .attr(groupTranslation);
- if (!chart.styledMode) {
- graphic
- .attr(pointAttr)
- .attr({ 'stroke-linejoin': 'round' })
- .shadow(shadow, shadowGroup);
- }
- point.delayedRendering = false;
- }
- graphic.attr({
- visibility: point.visible ? 'inherit' : 'hidden'
- });
- graphic.addClass(point.getClassName());
- }
- else if (graphic) {
- point.graphic = graphic.destroy();
- }
- });
- },
- /**
- * Slices in pie chart are initialized in DOM, but it's shapes and
- * animations are normally run in `drawPoints()`.
- * @private
- */
- drawPoints: function () {
- var renderer = this.chart.renderer;
- this.points.forEach(function (point) {
- // When updating a series between 2d and 3d or cartesian and
- // polar, the shape type changes.
- if (point.graphic && point.hasNewShapeType()) {
- point.graphic = point.graphic.destroy();
- }
- if (!point.graphic) {
- point.graphic = renderer[point.shapeType](point.shapeArgs)
- .add(point.series.group);
- point.delayedRendering = true;
- }
- });
- },
- /**
- * @private
- * @deprecated
- * @function Highcharts.seriesTypes.pie#searchPoint
- */
- searchPoint: noop,
- /**
- * Utility for sorting data labels
- *
- * @private
- * @function Highcharts.seriesTypes.pie#sortByAngle
- * @param {Array<Highcharts.Point>} points
- * @param {number} sign
- * @return {void}
- */
- sortByAngle: function (points, sign) {
- points.sort(function (a, b) {
- return ((typeof a.angle !== 'undefined') &&
- (b.angle - a.angle) * sign);
- });
- },
- /**
- * Use a simple symbol from LegendSymbolMixin.
- *
- * @private
- * @borrows Highcharts.LegendSymbolMixin.drawRectangle as Highcharts.seriesTypes.pie#drawLegendSymbol
- */
- drawLegendSymbol: LegendSymbolMixin.drawRectangle,
- /**
- * Use the getCenter method from drawLegendSymbol.
- *
- * @private
- * @borrows Highcharts.CenteredSeriesMixin.getCenter as Highcharts.seriesTypes.pie#getCenter
- */
- getCenter: centeredSeriesMixin.getCenter,
- /**
- * Pies don't have point marker symbols.
- *
- * @deprecated
- * @private
- * @function Highcharts.seriesTypes.pie#getSymbol
- */
- getSymbol: noop,
- /**
- * @private
- * @type {null}
- */
- drawGraph: null
- },
- /**
- * @lends seriesTypes.pie.prototype.pointClass.prototype
- */
- {
- /**
- * Initialize the pie slice
- *
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#init
- * @return {Highcharts.Point}
- */
- init: function () {
- Point.prototype.init.apply(this, arguments);
- var point = this,
- toggleSlice;
- point.name = pick(point.name, 'Slice');
- // add event listener for select
- toggleSlice = function (e) {
- point.slice(e.type === 'select');
- };
- addEvent(point, 'select', toggleSlice);
- addEvent(point, 'unselect', toggleSlice);
- return point;
- },
- /**
- * Negative points are not valid (#1530, #3623, #5322)
- *
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#isValid
- * @return {boolean}
- */
- isValid: function () {
- return isNumber(this.y) && this.y >= 0;
- },
- /**
- * Toggle the visibility of the pie slice
- *
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#setVisible
- * @param {boolean} vis
- * Whether to show the slice or not. If undefined, the visibility
- * is toggled.
- * @param {boolean} [redraw=false]
- * @return {void}
- */
- setVisible: function (vis, redraw) {
- var point = this,
- series = point.series,
- chart = series.chart,
- ignoreHiddenPoint = series.options.ignoreHiddenPoint;
- redraw = pick(redraw, ignoreHiddenPoint);
- if (vis !== point.visible) {
- // If called without an argument, toggle visibility
- point.visible = point.options.visible = vis =
- typeof vis === 'undefined' ? !point.visible : vis;
- // update userOptions.data
- series.options.data[series.data.indexOf(point)] =
- point.options;
- // Show and hide associated elements. This is performed
- // regardless of redraw or not, because chart.redraw only
- // handles full series.
- ['graphic', 'dataLabel', 'connector', 'shadowGroup'].forEach(function (key) {
- if (point[key]) {
- point[key][vis ? 'show' : 'hide'](true);
- }
- });
- if (point.legendItem) {
- chart.legend.colorizeItem(point, vis);
- }
- // #4170, hide halo after hiding point
- if (!vis && point.state === 'hover') {
- point.setState('');
- }
- // Handle ignore hidden slices
- if (ignoreHiddenPoint) {
- series.isDirty = true;
- }
- if (redraw) {
- chart.redraw();
- }
- }
- },
- /**
- * Set or toggle whether the slice is cut out from the pie
- *
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#slice
- * @param {boolean} sliced
- * When undefined, the slice state is toggled.
- * @param {boolean} redraw
- * Whether to redraw the chart. True by default.
- * @param {boolean|Partial<Highcharts.AnimationOptionsObject>}
- * Animation options.
- * @return {void}
- */
- slice: function (sliced, redraw, animation) {
- var point = this,
- series = point.series,
- chart = series.chart;
- setAnimation(animation, chart);
- // redraw is true by default
- redraw = pick(redraw, true);
- /**
- * Pie series only. Whether to display a slice offset from the
- * center.
- * @name Highcharts.Point#sliced
- * @type {boolean|undefined}
- */
- // if called without an argument, toggle
- point.sliced = point.options.sliced = sliced =
- defined(sliced) ? sliced : !point.sliced;
- // update userOptions.data
- series.options.data[series.data.indexOf(point)] =
- point.options;
- if (point.graphic) {
- point.graphic.animate(this.getTranslate());
- }
- if (point.shadowGroup) {
- point.shadowGroup.animate(this.getTranslate());
- }
- },
- /**
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#getTranslate
- * @return {Highcharts.TranslationAttributes}
- */
- getTranslate: function () {
- return this.sliced ? this.slicedTranslation : {
- translateX: 0,
- translateY: 0
- };
- },
- /**
- * @private
- * @function Highcharts.seriesTypes.pie#pointClass#haloPath
- * @param {number} size
- * @return {Highcharts.SVGPathArray}
- */
- haloPath: function (size) {
- var shapeArgs = this.shapeArgs;
- return this.sliced || !this.visible ?
- [] :
- this.series.chart.renderer.symbols.arc(shapeArgs.x, shapeArgs.y, shapeArgs.r + size, shapeArgs.r + size, {
- // Substract 1px to ensure the background is not bleeding
- // through between the halo and the slice (#7495).
- innerR: shapeArgs.r - 1,
- start: shapeArgs.start,
- end: shapeArgs.end
- });
- },
- connectorShapes: {
- // only one available before v7.0.0
- fixedOffset: function (labelPosition, connectorPosition, options) {
- var breakAt = connectorPosition.breakAt,
- touchingSliceAt = connectorPosition.touchingSliceAt,
- lineSegment = options.softConnector ? [
- 'C',
- // 1st control point (of the curve)
- labelPosition.x +
- // 5 gives the connector a little horizontal bend
- (labelPosition.alignment === 'left' ? -5 : 5),
- labelPosition.y,
- 2 * breakAt.x - touchingSliceAt.x,
- 2 * breakAt.y - touchingSliceAt.y,
- breakAt.x,
- breakAt.y //
- ] : [
- 'L',
- breakAt.x,
- breakAt.y
- ];
- // assemble the path
- return ([
- ['M', labelPosition.x, labelPosition.y],
- lineSegment,
- ['L', touchingSliceAt.x, touchingSliceAt.y]
- ]);
- },
- straight: function (labelPosition, connectorPosition) {
- var touchingSliceAt = connectorPosition.touchingSliceAt;
- // direct line to the slice
- return [
- ['M', labelPosition.x, labelPosition.y],
- ['L', touchingSliceAt.x, touchingSliceAt.y]
- ];
- },
- crookedLine: function (labelPosition, connectorPosition, options) {
- var touchingSliceAt = connectorPosition.touchingSliceAt,
- series = this.series,
- pieCenterX = series.center[0],
- plotWidth = series.chart.plotWidth,
- plotLeft = series.chart.plotLeft,
- alignment = labelPosition.alignment,
- radius = this.shapeArgs.r,
- crookDistance = relativeLength(// % to fraction
- options.crookDistance, 1),
- crookX = alignment === 'left' ?
- pieCenterX + radius + (plotWidth + plotLeft -
- pieCenterX - radius) * (1 - crookDistance) :
- plotLeft + (pieCenterX - radius) * crookDistance,
- segmentWithCrook = [
- 'L',
- crookX,
- labelPosition.y
- ],
- useCrook = true;
- // crookedLine formula doesn't make sense if the path overlaps
- // the label - use straight line instead in that case
- if (alignment === 'left' ?
- (crookX > labelPosition.x || crookX < touchingSliceAt.x) :
- (crookX < labelPosition.x || crookX > touchingSliceAt.x)) {
- useCrook = false;
- }
- // assemble the path
- var path = [
- ['M',
- labelPosition.x,
- labelPosition.y]
- ];
- if (useCrook) {
- path.push(segmentWithCrook);
- }
- path.push(['L', touchingSliceAt.x, touchingSliceAt.y]);
- return path;
- }
- },
- /**
- * Extendable method for getting the path of the connector between the
- * data label and the pie slice.
- */
- getConnectorPath: function () {
- var labelPosition = this.labelPosition,
- options = this.series.options.dataLabels,
- connectorShape = options.connectorShape,
- predefinedShapes = this.connectorShapes;
- // find out whether to use the predefined shape
- if (predefinedShapes[connectorShape]) {
- connectorShape = predefinedShapes[connectorShape];
- }
- return connectorShape.call(this, {
- // pass simplified label position object for user's convenience
- x: labelPosition.final.x,
- y: labelPosition.final.y,
- alignment: labelPosition.alignment
- }, labelPosition.connectorPosition, options);
- }
- }
- /* eslint-enable valid-jsdoc */
- );
- /**
- * A `pie` series. If the [type](#series.pie.type) option is not specified,
- * it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.pie
- * @excluding cropThreshold, dataParser, dataURL, stack, xAxis, yAxis,
- * dataSorting, step, boostThreshold, boostBlending
- * @product highcharts
- * @apioption series.pie
- */
- /**
- * An array of data points for the series. For the `pie` series type,
- * points can be given in the following ways:
- *
- * 1. An array of numerical values. In this case, the numerical values will be
- * interpreted as `y` options. Example:
- * ```js
- * data: [0, 5, 3, 5]
- * ```
- *
- * 2. An array of objects with named values. The following snippet shows only a
- * few settings, see the complete options set below. If the total number of
- * data points exceeds the series'
- * [turboThreshold](#series.pie.turboThreshold),
- * this option is not available.
- * ```js
- * data: [{
- * y: 1,
- * name: "Point2",
- * color: "#00FF00"
- * }, {
- * y: 7,
- * name: "Point1",
- * color: "#FF00FF"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/chart/reflow-true/
- * Numerical values
- * @sample {highcharts} highcharts/series/data-array-of-arrays/
- * Arrays of numeric x and y
- * @sample {highcharts} highcharts/series/data-array-of-arrays-datetime/
- * Arrays of datetime x and y
- * @sample {highcharts} highcharts/series/data-array-of-name-value/
- * Arrays of point.name and y
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @type {Array<number|Array<string,(number|null)>|null|*>}
- * @extends series.line.data
- * @excluding marker, x
- * @product highcharts
- * @apioption series.pie.data
- */
- /**
- * @type {Highcharts.SeriesPieDataLabelsOptionsObject}
- * @product highcharts
- * @apioption series.pie.data.dataLabels
- */
- /**
- * The sequential index of the data point in the legend.
- *
- * @type {number}
- * @product highcharts
- * @apioption series.pie.data.legendIndex
- */
- /**
- * Whether to display a slice offset from the center.
- *
- * @sample {highcharts} highcharts/point/sliced/
- * One sliced point
- *
- * @type {boolean}
- * @product highcharts
- * @apioption series.pie.data.sliced
- */
- /**
- * @excluding legendItemClick
- * @product highcharts
- * @apioption series.pie.events
- */
- ''; // placeholder for transpiled doclets above
- });
- _registerModule(_modules, 'Core/Series/DataLabels.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var noop = H.noop,
- seriesTypes = H.seriesTypes;
- var arrayMax = U.arrayMax,
- clamp = U.clamp,
- defined = U.defined,
- extend = U.extend,
- fireEvent = U.fireEvent,
- format = U.format,
- getDeferredAnimation = U.getDeferredAnimation,
- isArray = U.isArray,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- relativeLength = U.relativeLength,
- splat = U.splat,
- stableSort = U.stableSort;
- /**
- * Callback JavaScript function to format the data label as a string. Note that
- * if a `format` is defined, the format takes precedence and the formatter is
- * ignored.
- *
- * @callback Highcharts.DataLabelsFormatterCallbackFunction
- *
- * @param {Highcharts.PointLabelObject} this
- * Data label context to format
- *
- * @param {Highcharts.DataLabelsOptions} options
- * [API options](/highcharts/plotOptions.series.dataLabels) of the data label
- *
- * @return {number|string|null|undefined}
- * Formatted data label text
- */
- /**
- * Values for handling data labels that flow outside the plot area.
- *
- * @typedef {"allow"|"justify"} Highcharts.DataLabelsOverflowValue
- */
- var Series = H.Series;
- /* eslint-disable valid-jsdoc */
- /**
- * General distribution algorithm for distributing labels of differing size
- * along a confined length in two dimensions. The algorithm takes an array of
- * objects containing a size, a target and a rank. It will place the labels as
- * close as possible to their targets, skipping the lowest ranked labels if
- * necessary.
- *
- * @private
- * @function Highcharts.distribute
- * @param {Highcharts.DataLabelsBoxArray} boxes
- * @param {number} len
- * @param {number} [maxDistance]
- * @return {void}
- */
- H.distribute = function (boxes, len, maxDistance) {
- var i,
- overlapping = true,
- origBoxes = boxes, // Original array will be altered with added .pos
- restBoxes = [], // The outranked overshoot
- box,
- target,
- total = 0,
- reducedLen = origBoxes.reducedLen || len;
- /**
- * @private
- */
- function sortByTarget(a, b) {
- return a.target - b.target;
- }
- // If the total size exceeds the len, remove those boxes with the lowest
- // rank
- i = boxes.length;
- while (i--) {
- total += boxes[i].size;
- }
- // Sort by rank, then slice away overshoot
- if (total > reducedLen) {
- stableSort(boxes, function (a, b) {
- return (b.rank || 0) - (a.rank || 0);
- });
- i = 0;
- total = 0;
- while (total <= reducedLen) {
- total += boxes[i].size;
- i++;
- }
- restBoxes = boxes.splice(i - 1, boxes.length);
- }
- // Order by target
- stableSort(boxes, sortByTarget);
- // So far we have been mutating the original array. Now
- // create a copy with target arrays
- boxes = boxes.map(function (box) {
- return {
- size: box.size,
- targets: [box.target],
- align: pick(box.align, 0.5)
- };
- });
- while (overlapping) {
- // Initial positions: target centered in box
- i = boxes.length;
- while (i--) {
- box = boxes[i];
- // Composite box, average of targets
- target = (Math.min.apply(0, box.targets) +
- Math.max.apply(0, box.targets)) / 2;
- box.pos = clamp(target - box.size * box.align, 0, len - box.size);
- }
- // Detect overlap and join boxes
- i = boxes.length;
- overlapping = false;
- while (i--) {
- // Overlap
- if (i > 0 &&
- boxes[i - 1].pos + boxes[i - 1].size >
- boxes[i].pos) {
- // Add this size to the previous box
- boxes[i - 1].size += boxes[i].size;
- boxes[i - 1].targets = boxes[i - 1]
- .targets
- .concat(boxes[i].targets);
- boxes[i - 1].align = 0.5;
- // Overlapping right, push left
- if (boxes[i - 1].pos + boxes[i - 1].size > len) {
- boxes[i - 1].pos = len - boxes[i - 1].size;
- }
- boxes.splice(i, 1); // Remove this item
- overlapping = true;
- }
- }
- }
- // Add the rest (hidden boxes)
- origBoxes.push.apply(origBoxes, restBoxes);
- // Now the composite boxes are placed, we need to put the original boxes
- // within them
- i = 0;
- boxes.some(function (box) {
- var posInCompositeBox = 0;
- if (box.targets.some(function () {
- origBoxes[i].pos = box.pos + posInCompositeBox;
- // If the distance between the position and the target exceeds
- // maxDistance, abort the loop and decrease the length in increments
- // of 10% to recursively reduce the number of visible boxes by
- // rank. Once all boxes are within the maxDistance, we're good.
- if (typeof maxDistance !== 'undefined' &&
- Math.abs(origBoxes[i].pos - origBoxes[i].target) > maxDistance) {
- // Reset the positions that are already set
- origBoxes.slice(0, i + 1).forEach(function (box) {
- delete box.pos;
- });
- // Try with a smaller length
- origBoxes.reducedLen =
- (origBoxes.reducedLen || len) - (len * 0.1);
- // Recurse
- if (origBoxes.reducedLen > len * 0.1) {
- H.distribute(origBoxes, len, maxDistance);
- }
- // Exceeded maxDistance => abort
- return true;
- }
- posInCompositeBox += origBoxes[i].size;
- i++;
- })) {
- // Exceeded maxDistance => abort
- return true;
- }
- });
- // Add the rest (hidden) boxes and sort by target
- stableSort(origBoxes, sortByTarget);
- };
- /**
- * Draw the data labels
- *
- * @private
- * @function Highcharts.Series#drawDataLabels
- * @return {void}
- * @fires Highcharts.Series#event:afterDrawDataLabels
- */
- Series.prototype.drawDataLabels = function () {
- var series = this,
- chart = series.chart,
- seriesOptions = series.options,
- seriesDlOptions = seriesOptions.dataLabels,
- points = series.points,
- pointOptions,
- hasRendered = series.hasRendered || 0,
- dataLabelsGroup,
- dataLabelAnim = seriesDlOptions.animation,
- animationConfig = seriesDlOptions.defer ?
- getDeferredAnimation(chart,
- dataLabelAnim,
- series) :
- { defer: 0,
- duration: 0 },
- renderer = chart.renderer;
- /**
- * Handle the dataLabels.filter option.
- * @private
- */
- function applyFilter(point, options) {
- var filter = options.filter,
- op,
- prop,
- val;
- if (filter) {
- op = filter.operator;
- prop = point[filter.property];
- val = filter.value;
- if ((op === '>' && prop > val) ||
- (op === '<' && prop < val) ||
- (op === '>=' && prop >= val) ||
- (op === '<=' && prop <= val) ||
- (op === '==' && prop == val) || // eslint-disable-line eqeqeq
- (op === '===' && prop === val)) {
- return true;
- }
- return false;
- }
- return true;
- }
- /**
- * Merge two objects that can be arrays. If one of them is an array, the
- * other is merged into each element. If both are arrays, each element is
- * merged by index. If neither are arrays, we use normal merge.
- * @private
- */
- function mergeArrays(one, two) {
- var res = [],
- i;
- if (isArray(one) && !isArray(two)) {
- res = one.map(function (el) {
- return merge(el, two);
- });
- }
- else if (isArray(two) && !isArray(one)) {
- res = two.map(function (el) {
- return merge(one, el);
- });
- }
- else if (!isArray(one) && !isArray(two)) {
- res = merge(one, two);
- }
- else {
- i = Math.max(one.length, two.length);
- while (i--) {
- res[i] = merge(one[i], two[i]);
- }
- }
- return res;
- }
- // Merge in plotOptions.dataLabels for series
- seriesDlOptions = mergeArrays(mergeArrays(chart.options.plotOptions &&
- chart.options.plotOptions.series &&
- chart.options.plotOptions.series.dataLabels, chart.options.plotOptions &&
- chart.options.plotOptions[series.type] &&
- chart.options.plotOptions[series.type].dataLabels), seriesDlOptions);
- fireEvent(this, 'drawDataLabels');
- if (isArray(seriesDlOptions) ||
- seriesDlOptions.enabled ||
- series._hasPointLabels) {
- // Create a separate group for the data labels to avoid rotation
- dataLabelsGroup = series.plotGroup('dataLabelsGroup', 'data-labels', !hasRendered ? 'hidden' : 'inherit', // #5133, #10220
- seriesDlOptions.zIndex || 6);
- dataLabelsGroup.attr({ opacity: +hasRendered }); // #3300
- if (!hasRendered) {
- var group = series.dataLabelsGroup;
- if (group) {
- if (series.visible) { // #2597, #3023, #3024
- dataLabelsGroup.show(true);
- }
- group[seriesOptions.animation ? 'animate' : 'attr']({ opacity: 1 }, animationConfig);
- }
- }
- // Make the labels for each point
- points.forEach(function (point) {
- // Merge in series options for the point.
- // @note dataLabelAttribs (like pointAttribs) would eradicate
- // the need for dlOptions, and simplify the section below.
- pointOptions = splat(mergeArrays(seriesDlOptions, point.dlOptions || // dlOptions is used in treemaps
- (point.options && point.options.dataLabels)));
- // Handle each individual data label for this point
- pointOptions.forEach(function (labelOptions, i) {
- // Options for one datalabel
- var labelEnabled = (labelOptions.enabled &&
- // #2282, #4641, #7112, #10049
- (!point.isNull || point.dataLabelOnNull) &&
- applyFilter(point,
- labelOptions)),
- labelConfig,
- formatString,
- labelText,
- style,
- rotation,
- attr,
- dataLabel = point.dataLabels ? point.dataLabels[i] :
- point.dataLabel,
- connector = point.connectors ? point.connectors[i] :
- point.connector,
- labelDistance = pick(labelOptions.distance,
- point.labelDistance),
- isNew = !dataLabel;
- if (labelEnabled) {
- // Create individual options structure that can be extended
- // without affecting others
- labelConfig = point.getLabelConfig();
- formatString = pick(labelOptions[point.formatPrefix + 'Format'], labelOptions.format);
- labelText = defined(formatString) ?
- format(formatString, labelConfig, chart) :
- (labelOptions[point.formatPrefix + 'Formatter'] ||
- labelOptions.formatter).call(labelConfig, labelOptions);
- style = labelOptions.style;
- rotation = labelOptions.rotation;
- if (!chart.styledMode) {
- // Determine the color
- style.color = pick(labelOptions.color, style.color, series.color, '#000000');
- // Get automated contrast color
- if (style.color === 'contrast') {
- point.contrastColor = renderer.getContrast((point.color || series.color));
- style.color = (!defined(labelDistance) &&
- labelOptions.inside) ||
- labelDistance < 0 ||
- !!seriesOptions.stacking ?
- point.contrastColor :
- '#000000';
- }
- else {
- delete point.contrastColor;
- }
- if (seriesOptions.cursor) {
- style.cursor = seriesOptions.cursor;
- }
- }
- attr = {
- r: labelOptions.borderRadius || 0,
- rotation: rotation,
- padding: labelOptions.padding,
- zIndex: 1
- };
- if (!chart.styledMode) {
- attr.fill = labelOptions.backgroundColor;
- attr.stroke = labelOptions.borderColor;
- attr['stroke-width'] = labelOptions.borderWidth;
- }
- // Remove unused attributes (#947)
- objectEach(attr, function (val, name) {
- if (typeof val === 'undefined') {
- delete attr[name];
- }
- });
- }
- // If the point is outside the plot area, destroy it. #678, #820
- if (dataLabel && (!labelEnabled || !defined(labelText))) {
- point.dataLabel =
- point.dataLabel && point.dataLabel.destroy();
- if (point.dataLabels) {
- // Remove point.dataLabels if this was the last one
- if (point.dataLabels.length === 1) {
- delete point.dataLabels;
- }
- else {
- delete point.dataLabels[i];
- }
- }
- if (!i) {
- delete point.dataLabel;
- }
- if (connector) {
- point.connector = point.connector.destroy();
- if (point.connectors) {
- // Remove point.connectors if this was the last one
- if (point.connectors.length === 1) {
- delete point.connectors;
- }
- else {
- delete point.connectors[i];
- }
- }
- }
- // Individual labels are disabled if the are explicitly disabled
- // in the point options, or if they fall outside the plot area.
- }
- else if (labelEnabled && defined(labelText)) {
- if (!dataLabel) {
- // Create new label element
- point.dataLabels = point.dataLabels || [];
- dataLabel = point.dataLabels[i] = rotation ?
- // Labels don't rotate, use text element
- renderer.text(labelText, 0, -9999, labelOptions.useHTML)
- .addClass('highcharts-data-label') :
- // We can use label
- renderer.label(labelText, 0, -9999, labelOptions.shape, null, null, labelOptions.useHTML, null, 'data-label');
- // Store for backwards compatibility
- if (!i) {
- point.dataLabel = dataLabel;
- }
- dataLabel.addClass(' highcharts-data-label-color-' + point.colorIndex +
- ' ' + (labelOptions.className || '') +
- ( // #3398
- labelOptions.useHTML ?
- ' highcharts-tracker' :
- ''));
- }
- else {
- // Use old element and just update text
- attr.text = labelText;
- }
- // Store data label options for later access
- dataLabel.options = labelOptions;
- dataLabel.attr(attr);
- if (!chart.styledMode) {
- // Styles must be applied before add in order to read
- // text bounding box
- dataLabel.css(style).shadow(labelOptions.shadow);
- }
- if (!dataLabel.added) {
- dataLabel.add(dataLabelsGroup);
- }
- if (labelOptions.textPath && !labelOptions.useHTML) {
- dataLabel.setTextPath((point.getDataLabelPath &&
- point.getDataLabelPath(dataLabel)) || point.graphic, labelOptions.textPath);
- if (point.dataLabelPath &&
- !labelOptions.textPath.enabled) {
- // clean the DOM
- point.dataLabelPath = point.dataLabelPath.destroy();
- }
- }
- // Now the data label is created and placed at 0,0, so we
- // need to align it
- series.alignDataLabel(point, dataLabel, labelOptions, null, isNew);
- }
- });
- });
- }
- fireEvent(this, 'afterDrawDataLabels');
- };
- /**
- * Align each individual data label.
- *
- * @private
- * @function Highcharts.Series#alignDataLabel
- * @param {Highcharts.Point} point
- * @param {Highcharts.SVGElement} dataLabel
- * @param {Highcharts.DataLabelsOptions} options
- * @param {Highcharts.BBoxObject} alignTo
- * @param {boolean} [isNew]
- * @return {void}
- */
- Series.prototype.alignDataLabel = function (point, dataLabel, options, alignTo, isNew) {
- var series = this,
- chart = this.chart,
- inverted = this.isCartesian && chart.inverted,
- enabledDataSorting = this.enabledDataSorting,
- plotX = pick(point.dlBox && point.dlBox.centerX,
- point.plotX, -9999),
- plotY = pick(point.plotY, -9999),
- bBox = dataLabel.getBBox(),
- baseline,
- rotation = options.rotation,
- normRotation,
- negRotation,
- align = options.align,
- rotCorr, // rotation correction
- isInsidePlot = chart.isInsidePlot(plotX,
- Math.round(plotY),
- inverted),
- // Math.round for rounding errors (#2683), alignTo to allow column
- // labels (#2700)
- alignAttr, // the final position;
- justify = pick(options.overflow, (enabledDataSorting ? 'none' : 'justify')) === 'justify', visible = this.visible &&
- point.visible !== false &&
- (point.series.forceDL ||
- (enabledDataSorting && !justify) ||
- isInsidePlot ||
- (
- // If the data label is inside the align box, it is enough
- // that parts of the align box is inside the plot area
- // (#12370)
- options.inside && alignTo && chart.isInsidePlot(plotX, inverted ?
- alignTo.x + 1 :
- alignTo.y + alignTo.height - 1, inverted))), setStartPos = function (alignOptions) {
- if (enabledDataSorting && series.xAxis && !justify) {
- series.setDataLabelStartPos(point, dataLabel, isNew, isInsidePlot, alignOptions);
- }
- };
- if (visible) {
- baseline = chart.renderer.fontMetrics(chart.styledMode ? void 0 : options.style.fontSize, dataLabel).b;
- // The alignment box is a singular point
- alignTo = extend({
- x: inverted ? this.yAxis.len - plotY : plotX,
- y: Math.round(inverted ? this.xAxis.len - plotX : plotY),
- width: 0,
- height: 0
- }, alignTo);
- // Add the text size for alignment calculation
- extend(options, {
- width: bBox.width,
- height: bBox.height
- });
- // Allow a hook for changing alignment in the last moment, then do the
- // alignment
- if (rotation) {
- justify = false; // Not supported for rotated text
- rotCorr = chart.renderer.rotCorr(baseline, rotation); // #3723
- alignAttr = {
- x: (alignTo.x +
- (options.x || 0) +
- alignTo.width / 2 +
- rotCorr.x),
- y: (alignTo.y +
- (options.y || 0) +
- { top: 0, middle: 0.5, bottom: 1 }[options.verticalAlign] *
- alignTo.height)
- };
- setStartPos(alignAttr); // data sorting
- dataLabel[isNew ? 'attr' : 'animate'](alignAttr)
- .attr({
- align: align
- });
- // Compensate for the rotated label sticking out on the sides
- normRotation = (rotation + 720) % 360;
- negRotation = normRotation > 180 && normRotation < 360;
- if (align === 'left') {
- alignAttr.y -= negRotation ? bBox.height : 0;
- }
- else if (align === 'center') {
- alignAttr.x -= bBox.width / 2;
- alignAttr.y -= bBox.height / 2;
- }
- else if (align === 'right') {
- alignAttr.x -= bBox.width;
- alignAttr.y -= negRotation ? 0 : bBox.height;
- }
- dataLabel.placed = true;
- dataLabel.alignAttr = alignAttr;
- }
- else {
- setStartPos(alignTo); // data sorting
- dataLabel.align(options, null, alignTo);
- alignAttr = dataLabel.alignAttr;
- }
- // Handle justify or crop
- if (justify && alignTo.height >= 0) { // #8830
- this.justifyDataLabel(dataLabel, options, alignAttr, bBox, alignTo, isNew);
- // Now check that the data label is within the plot area
- }
- else if (pick(options.crop, true)) {
- visible =
- chart.isInsidePlot(alignAttr.x, alignAttr.y) &&
- chart.isInsidePlot(alignAttr.x + bBox.width, alignAttr.y + bBox.height);
- }
- // When we're using a shape, make it possible with a connector or an
- // arrow pointing to thie point
- if (options.shape && !rotation) {
- dataLabel[isNew ? 'attr' : 'animate']({
- anchorX: inverted ?
- chart.plotWidth - point.plotY :
- point.plotX,
- anchorY: inverted ?
- chart.plotHeight - point.plotX :
- point.plotY
- });
- }
- }
- // To use alignAttr property in hideOverlappingLabels
- if (isNew && enabledDataSorting) {
- dataLabel.placed = false;
- }
- // Show or hide based on the final aligned position
- if (!visible && (!enabledDataSorting || justify)) {
- dataLabel.hide(true);
- dataLabel.placed = false; // don't animate back in
- }
- };
- /**
- * Set starting position for data label sorting animation.
- *
- * @private
- * @function Highcharts.Series#setDataLabelStartPos
- * @param {Highcharts.SVGElement} dataLabel
- * @param {Highcharts.ColumnPoint} point
- * @param {boolean | undefined} [isNew]
- * @param {boolean} [isInside]
- * @param {Highcharts.AlignObject} [alignOptions]
- *
- * @return {void}
- */
- Series.prototype.setDataLabelStartPos = function (point, dataLabel, isNew, isInside, alignOptions) {
- var chart = this.chart,
- inverted = chart.inverted,
- xAxis = this.xAxis,
- reversed = xAxis.reversed,
- labelCenter = inverted ? dataLabel.height / 2 : dataLabel.width / 2,
- pointWidth = point.pointWidth,
- halfWidth = pointWidth ? pointWidth / 2 : 0,
- startXPos,
- startYPos;
- startXPos = inverted ?
- alignOptions.x :
- (reversed ?
- -labelCenter - halfWidth :
- xAxis.width - labelCenter + halfWidth);
- startYPos = inverted ?
- (reversed ?
- this.yAxis.height - labelCenter + halfWidth :
- -labelCenter - halfWidth) : alignOptions.y;
- dataLabel.startXPos = startXPos;
- dataLabel.startYPos = startYPos;
- // We need to handle visibility in case of sorting point outside plot area
- if (!isInside) {
- dataLabel
- .attr({ opacity: 1 })
- .animate({ opacity: 0 }, void 0, dataLabel.hide);
- }
- else if (dataLabel.visibility === 'hidden') {
- dataLabel.show();
- dataLabel
- .attr({ opacity: 0 })
- .animate({ opacity: 1 });
- }
- // Save start position on first render, but do not change position
- if (!chart.hasRendered) {
- return;
- }
- // Set start position
- if (isNew) {
- dataLabel.attr({ x: dataLabel.startXPos, y: dataLabel.startYPos });
- }
- dataLabel.placed = true;
- };
- /**
- * If data labels fall partly outside the plot area, align them back in, in a
- * way that doesn't hide the point.
- *
- * @private
- * @function Highcharts.Series#justifyDataLabel
- * @param {Highcharts.SVGElement} dataLabel
- * @param {Highcharts.DataLabelsOptions} options
- * @param {Highcharts.SVGAttributes} alignAttr
- * @param {Highcharts.BBoxObject} bBox
- * @param {Highcharts.BBoxObject} [alignTo]
- * @param {boolean} [isNew]
- * @return {boolean|undefined}
- */
- Series.prototype.justifyDataLabel = function (dataLabel, options, alignAttr, bBox, alignTo, isNew) {
- var chart = this.chart,
- align = options.align,
- verticalAlign = options.verticalAlign,
- off,
- justified,
- padding = dataLabel.box ? 0 : (dataLabel.padding || 0);
- var _a = options.x,
- x = _a === void 0 ? 0 : _a,
- _b = options.y,
- y = _b === void 0 ? 0 : _b;
- // Off left
- off = alignAttr.x + padding;
- if (off < 0) {
- if (align === 'right' && x >= 0) {
- options.align = 'left';
- options.inside = true;
- }
- else {
- x -= off;
- }
- justified = true;
- }
- // Off right
- off = alignAttr.x + bBox.width - padding;
- if (off > chart.plotWidth) {
- if (align === 'left' && x <= 0) {
- options.align = 'right';
- options.inside = true;
- }
- else {
- x += chart.plotWidth - off;
- }
- justified = true;
- }
- // Off top
- off = alignAttr.y + padding;
- if (off < 0) {
- if (verticalAlign === 'bottom' && y >= 0) {
- options.verticalAlign = 'top';
- options.inside = true;
- }
- else {
- y -= off;
- }
- justified = true;
- }
- // Off bottom
- off = alignAttr.y + bBox.height - padding;
- if (off > chart.plotHeight) {
- if (verticalAlign === 'top' && y <= 0) {
- options.verticalAlign = 'bottom';
- options.inside = true;
- }
- else {
- y += chart.plotHeight - off;
- }
- justified = true;
- }
- if (justified) {
- options.x = x;
- options.y = y;
- dataLabel.placed = !isNew;
- dataLabel.align(options, void 0, alignTo);
- }
- return justified;
- };
- if (seriesTypes.pie) {
- seriesTypes.pie.prototype.dataLabelPositioners = {
- // Based on the value computed in Highcharts' distribute algorithm.
- radialDistributionY: function (point) {
- return point.top + point.distributeBox.pos;
- },
- // get the x - use the natural x position for labels near the
- // top and bottom, to prevent the top and botton slice
- // connectors from touching each other on either side
- // Based on the value computed in Highcharts' distribute algorithm.
- radialDistributionX: function (series, point, y, naturalY) {
- return series.getX(y < point.top + 2 || y > point.bottom - 2 ?
- naturalY :
- y, point.half, point);
- },
- // dataLabels.distance determines the x position of the label
- justify: function (point, radius, seriesCenter) {
- return seriesCenter[0] + (point.half ? -1 : 1) *
- (radius + point.labelDistance);
- },
- // Left edges of the left-half labels touch the left edge of the plot
- // area. Right edges of the right-half labels touch the right edge of
- // the plot area.
- alignToPlotEdges: function (dataLabel, half, plotWidth, plotLeft) {
- var dataLabelWidth = dataLabel.getBBox().width;
- return half ? dataLabelWidth + plotLeft :
- plotWidth - dataLabelWidth - plotLeft;
- },
- // Connectors of each side end in the same x position. Labels are
- // aligned to them. Left edge of the widest left-half label touches the
- // left edge of the plot area. Right edge of the widest right-half label
- // touches the right edge of the plot area.
- alignToConnectors: function (points, half, plotWidth, plotLeft) {
- var maxDataLabelWidth = 0,
- dataLabelWidth;
- // find widest data label
- points.forEach(function (point) {
- dataLabelWidth = point.dataLabel.getBBox().width;
- if (dataLabelWidth > maxDataLabelWidth) {
- maxDataLabelWidth = dataLabelWidth;
- }
- });
- return half ? maxDataLabelWidth + plotLeft :
- plotWidth - maxDataLabelWidth - plotLeft;
- }
- };
- /**
- * Override the base drawDataLabels method by pie specific functionality
- *
- * @private
- * @function Highcharts.seriesTypes.pie#drawDataLabels
- * @return {void}
- */
- seriesTypes.pie.prototype.drawDataLabels = function () {
- var series = this,
- data = series.data,
- point,
- chart = series.chart,
- options = series.options.dataLabels || {},
- connectorPadding = options.connectorPadding,
- connectorWidth,
- plotWidth = chart.plotWidth,
- plotHeight = chart.plotHeight,
- plotLeft = chart.plotLeft,
- maxWidth = Math.round(chart.chartWidth / 3),
- connector,
- seriesCenter = series.center,
- radius = seriesCenter[2] / 2,
- centerY = seriesCenter[1],
- dataLabel,
- dataLabelWidth,
- // labelPos,
- labelPosition,
- labelHeight,
- // divide the points into right and left halves for anti collision
- halves = [
- [],
- [] // left
- ],
- x,
- y,
- visibility,
- j,
- overflow = [0, 0, 0, 0], // top, right, bottom, left
- dataLabelPositioners = series.dataLabelPositioners,
- pointDataLabelsOptions;
- // get out if not enabled
- if (!series.visible ||
- (!options.enabled &&
- !series._hasPointLabels)) {
- return;
- }
- // Reset all labels that have been shortened
- data.forEach(function (point) {
- if (point.dataLabel && point.visible && point.dataLabel.shortened) {
- point.dataLabel
- .attr({
- width: 'auto'
- }).css({
- width: 'auto',
- textOverflow: 'clip'
- });
- point.dataLabel.shortened = false;
- }
- });
- // run parent method
- Series.prototype.drawDataLabels.apply(series);
- data.forEach(function (point) {
- if (point.dataLabel) {
- if (point.visible) { // #407, #2510
- // Arrange points for detection collision
- halves[point.half].push(point);
- // Reset positions (#4905)
- point.dataLabel._pos = null;
- // Avoid long labels squeezing the pie size too far down
- if (!defined(options.style.width) &&
- !defined(point.options.dataLabels &&
- point.options.dataLabels.style &&
- point.options.dataLabels.style.width)) {
- if (point.dataLabel.getBBox().width > maxWidth) {
- point.dataLabel.css({
- // Use a fraction of the maxWidth to avoid
- // wrapping close to the end of the string.
- width: Math.round(maxWidth * 0.7) + 'px'
- });
- point.dataLabel.shortened = true;
- }
- }
- }
- else {
- point.dataLabel = point.dataLabel.destroy();
- // Workaround to make pies destroy multiple datalabels
- // correctly. This logic needs rewriting to support multiple
- // datalabels fully.
- if (point.dataLabels && point.dataLabels.length === 1) {
- delete point.dataLabels;
- }
- }
- }
- });
- /* Loop over the points in each half, starting from the top and bottom
- * of the pie to detect overlapping labels.
- */
- halves.forEach(function (points, i) {
- var top,
- bottom,
- length = points.length,
- positions = [],
- naturalY,
- sideOverflow,
- size,
- distributionLength;
- if (!length) {
- return;
- }
- // Sort by angle
- series.sortByAngle(points, i - 0.5);
- // Only do anti-collision when we have dataLabels outside the pie
- // and have connectors. (#856)
- if (series.maxLabelDistance > 0) {
- top = Math.max(0, centerY - radius - series.maxLabelDistance);
- bottom = Math.min(centerY + radius + series.maxLabelDistance, chart.plotHeight);
- points.forEach(function (point) {
- // check if specific points' label is outside the pie
- if (point.labelDistance > 0 && point.dataLabel) {
- // point.top depends on point.labelDistance value
- // Used for calculation of y value in getX method
- point.top = Math.max(0, centerY - radius - point.labelDistance);
- point.bottom = Math.min(centerY + radius + point.labelDistance, chart.plotHeight);
- size = point.dataLabel.getBBox().height || 21;
- // point.positionsIndex is needed for getting index of
- // parameter related to specific point inside positions
- // array - not every point is in positions array.
- point.distributeBox = {
- target: point.labelPosition.natural.y -
- point.top + size / 2,
- size: size,
- rank: point.y
- };
- positions.push(point.distributeBox);
- }
- });
- distributionLength = bottom + size - top;
- H.distribute(positions, distributionLength, distributionLength / 5);
- }
- // Now the used slots are sorted, fill them up sequentially
- for (j = 0; j < length; j++) {
- point = points[j];
- // labelPos = point.labelPos;
- labelPosition = point.labelPosition;
- dataLabel = point.dataLabel;
- visibility = point.visible === false ? 'hidden' : 'inherit';
- naturalY = labelPosition.natural.y;
- y = naturalY;
- if (positions && defined(point.distributeBox)) {
- if (typeof point.distributeBox.pos === 'undefined') {
- visibility = 'hidden';
- }
- else {
- labelHeight = point.distributeBox.size;
- // Find label's y position
- y = dataLabelPositioners
- .radialDistributionY(point);
- }
- }
- // It is needed to delete point.positionIndex for
- // dynamically added points etc.
- delete point.positionIndex; // @todo unused
- // Find label's x position
- // justify is undocumented in the API - preserve support for it
- if (options.justify) {
- x = dataLabelPositioners.justify(point, radius, seriesCenter);
- }
- else {
- switch (options.alignTo) {
- case 'connectors':
- x = dataLabelPositioners.alignToConnectors(points, i, plotWidth, plotLeft);
- break;
- case 'plotEdges':
- x = dataLabelPositioners.alignToPlotEdges(dataLabel, i, plotWidth, plotLeft);
- break;
- default:
- x = dataLabelPositioners.radialDistributionX(series, point, y, naturalY);
- }
- }
- // Record the placement and visibility
- dataLabel._attr = {
- visibility: visibility,
- align: labelPosition.alignment
- };
- pointDataLabelsOptions = point.options.dataLabels || {};
- dataLabel._pos = {
- x: (x +
- pick(pointDataLabelsOptions.x, options.x) + // (#12985)
- ({
- left: connectorPadding,
- right: -connectorPadding
- }[labelPosition.alignment] || 0)),
- // 10 is for the baseline (label vs text)
- y: (y +
- pick(pointDataLabelsOptions.y, options.y) - // (#12985)
- 10)
- };
- // labelPos.x = x;
- // labelPos.y = y;
- labelPosition.final.x = x;
- labelPosition.final.y = y;
- // Detect overflowing data labels
- if (pick(options.crop, true)) {
- dataLabelWidth = dataLabel.getBBox().width;
- sideOverflow = null;
- // Overflow left
- if (x - dataLabelWidth < connectorPadding &&
- i === 1 // left half
- ) {
- sideOverflow = Math.round(dataLabelWidth - x + connectorPadding);
- overflow[3] = Math.max(sideOverflow, overflow[3]);
- // Overflow right
- }
- else if (x + dataLabelWidth > plotWidth - connectorPadding &&
- i === 0 // right half
- ) {
- sideOverflow = Math.round(x + dataLabelWidth - plotWidth + connectorPadding);
- overflow[1] = Math.max(sideOverflow, overflow[1]);
- }
- // Overflow top
- if (y - labelHeight / 2 < 0) {
- overflow[0] = Math.max(Math.round(-y + labelHeight / 2), overflow[0]);
- // Overflow left
- }
- else if (y + labelHeight / 2 > plotHeight) {
- overflow[2] = Math.max(Math.round(y + labelHeight / 2 - plotHeight), overflow[2]);
- }
- dataLabel.sideOverflow = sideOverflow;
- }
- } // for each point
- }); // for each half
- // Do not apply the final placement and draw the connectors until we
- // have verified that labels are not spilling over.
- if (arrayMax(overflow) === 0 ||
- this.verifyDataLabelOverflow(overflow)) {
- // Place the labels in the final position
- this.placeDataLabels();
- this.points.forEach(function (point) {
- // #8864: every connector can have individual options
- pointDataLabelsOptions =
- merge(options, point.options.dataLabels);
- connectorWidth =
- pick(pointDataLabelsOptions.connectorWidth, 1);
- // Draw the connector
- if (connectorWidth) {
- var isNew;
- connector = point.connector;
- dataLabel = point.dataLabel;
- if (dataLabel &&
- dataLabel._pos &&
- point.visible &&
- point.labelDistance > 0) {
- visibility = dataLabel._attr.visibility;
- isNew = !connector;
- if (isNew) {
- point.connector = connector = chart.renderer
- .path()
- .addClass('highcharts-data-label-connector ' +
- ' highcharts-color-' + point.colorIndex +
- (point.className ?
- ' ' + point.className :
- ''))
- .add(series.dataLabelsGroup);
- if (!chart.styledMode) {
- connector.attr({
- 'stroke-width': connectorWidth,
- 'stroke': (pointDataLabelsOptions.connectorColor ||
- point.color ||
- '#666666')
- });
- }
- }
- connector[isNew ? 'attr' : 'animate']({
- d: point.getConnectorPath()
- });
- connector.attr('visibility', visibility);
- }
- else if (connector) {
- point.connector = connector.destroy();
- }
- }
- });
- }
- };
- /**
- * Extendable method for getting the path of the connector between the data
- * label and the pie slice.
- *
- * @private
- * @function Highcharts.seriesTypes.pie#connectorPath
- *
- * @param {*} labelPos
- *
- * @return {Highcharts.SVGPathArray}
- */
- // TODO: depracated - remove it
- /*
- seriesTypes.pie.prototype.connectorPath = function (labelPos) {
- var x = labelPos.x,
- y = labelPos.y;
- return pick(this.options.dataLabels.softConnector, true) ? [
- 'M',
- // end of the string at the label
- x + (labelPos[6] === 'left' ? 5 : -5), y,
- 'C',
- x, y, // first break, next to the label
- 2 * labelPos[2] - labelPos[4], 2 * labelPos[3] - labelPos[5],
- labelPos[2], labelPos[3], // second break
- 'L',
- labelPos[4], labelPos[5] // base
- ] : [
- 'M',
- // end of the string at the label
- x + (labelPos[6] === 'left' ? 5 : -5), y,
- 'L',
- labelPos[2], labelPos[3], // second break
- 'L',
- labelPos[4], labelPos[5] // base
- ];
- };
- */
- /**
- * Perform the final placement of the data labels after we have verified
- * that they fall within the plot area.
- *
- * @private
- * @function Highcharts.seriesTypes.pie#placeDataLabels
- * @return {void}
- */
- seriesTypes.pie.prototype.placeDataLabels = function () {
- this.points.forEach(function (point) {
- var dataLabel = point.dataLabel,
- _pos;
- if (dataLabel && point.visible) {
- _pos = dataLabel._pos;
- if (_pos) {
- // Shorten data labels with ellipsis if they still overflow
- // after the pie has reached minSize (#223).
- if (dataLabel.sideOverflow) {
- dataLabel._attr.width =
- Math.max(dataLabel.getBBox().width -
- dataLabel.sideOverflow, 0);
- dataLabel.css({
- width: dataLabel._attr.width + 'px',
- textOverflow: ((this.options.dataLabels.style || {})
- .textOverflow ||
- 'ellipsis')
- });
- dataLabel.shortened = true;
- }
- dataLabel.attr(dataLabel._attr);
- dataLabel[dataLabel.moved ? 'animate' : 'attr'](_pos);
- dataLabel.moved = true;
- }
- else if (dataLabel) {
- dataLabel.attr({ y: -9999 });
- }
- }
- // Clear for update
- delete point.distributeBox;
- }, this);
- };
- seriesTypes.pie.prototype.alignDataLabel = noop;
- /**
- * Verify whether the data labels are allowed to draw, or we should run more
- * translation and data label positioning to keep them inside the plot area.
- * Returns true when data labels are ready to draw.
- *
- * @private
- * @function Highcharts.seriesTypes.pie#verifyDataLabelOverflow
- * @param {Array<number>} overflow
- * @return {boolean}
- */
- seriesTypes.pie.prototype.verifyDataLabelOverflow = function (overflow) {
- var center = this.center,
- options = this.options,
- centerOption = options.center,
- minSize = options.minSize || 80,
- newSize = minSize,
- // If a size is set, return true and don't try to shrink the pie
- // to fit the labels.
- ret = options.size !== null;
- if (!ret) {
- // Handle horizontal size and center
- if (centerOption[0] !== null) { // Fixed center
- newSize = Math.max(center[2] -
- Math.max(overflow[1], overflow[3]), minSize);
- }
- else { // Auto center
- newSize = Math.max(
- // horizontal overflow
- center[2] - overflow[1] - overflow[3], minSize);
- // horizontal center
- center[0] += (overflow[3] - overflow[1]) / 2;
- }
- // Handle vertical size and center
- if (centerOption[1] !== null) { // Fixed center
- newSize = clamp(newSize, minSize, center[2] - Math.max(overflow[0], overflow[2]));
- }
- else { // Auto center
- newSize = clamp(newSize, minSize,
- // vertical overflow
- center[2] - overflow[0] - overflow[2]);
- // vertical center
- center[1] += (overflow[0] - overflow[2]) / 2;
- }
- // If the size must be decreased, we need to run translate and
- // drawDataLabels again
- if (newSize < center[2]) {
- center[2] = newSize;
- center[3] = Math.min(// #3632
- relativeLength(options.innerSize || 0, newSize), newSize);
- this.translate(center);
- if (this.drawDataLabels) {
- this.drawDataLabels();
- }
- // Else, return true to indicate that the pie and its labels is
- // within the plot area
- }
- else {
- ret = true;
- }
- }
- return ret;
- };
- }
- if (seriesTypes.column) {
- /**
- * Override the basic data label alignment by adjusting for the position of
- * the column.
- *
- * @private
- * @function Highcharts.seriesTypes.column#alignDataLabel
- * @param {Highcharts.Point} point
- * @param {Highcharts.SVGElement} dataLabel
- * @param {Highcharts.DataLabelsOptions} options
- * @param {Highcharts.BBoxObject} alignTo
- * @param {boolean} [isNew]
- * @return {void}
- */
- seriesTypes.column.prototype.alignDataLabel = function (point, dataLabel, options, alignTo, isNew) {
- var inverted = this.chart.inverted,
- series = point.series,
- // data label box for alignment
- dlBox = point.dlBox || point.shapeArgs,
- below = pick(point.below, // range series
- point.plotY >
- pick(this.translatedThreshold,
- series.yAxis.len)),
- // draw it inside the box?
- inside = pick(options.inside, !!this.options.stacking),
- overshoot;
- // Align to the column itself, or the top of it
- if (dlBox) { // Area range uses this method but not alignTo
- alignTo = merge(dlBox);
- if (alignTo.y < 0) {
- alignTo.height += alignTo.y;
- alignTo.y = 0;
- }
- // If parts of the box overshoots outside the plot area, modify the
- // box to center the label inside
- overshoot = alignTo.y + alignTo.height - series.yAxis.len;
- if (overshoot > 0 && overshoot < alignTo.height) {
- alignTo.height -= overshoot;
- }
- if (inverted) {
- alignTo = {
- x: series.yAxis.len - alignTo.y - alignTo.height,
- y: series.xAxis.len - alignTo.x - alignTo.width,
- width: alignTo.height,
- height: alignTo.width
- };
- }
- // Compute the alignment box
- if (!inside) {
- if (inverted) {
- alignTo.x += below ? 0 : alignTo.width;
- alignTo.width = 0;
- }
- else {
- alignTo.y += below ? alignTo.height : 0;
- alignTo.height = 0;
- }
- }
- }
- // When alignment is undefined (typically columns and bars), display the
- // individual point below or above the point depending on the threshold
- options.align = pick(options.align, !inverted || inside ? 'center' : below ? 'right' : 'left');
- options.verticalAlign = pick(options.verticalAlign, inverted || inside ? 'middle' : below ? 'top' : 'bottom');
- // Call the parent method
- Series.prototype.alignDataLabel.call(this, point, dataLabel, options, alignTo, isNew);
- // If label was justified and we have contrast, set it:
- if (options.inside && point.contrastColor) {
- dataLabel.css({
- color: point.contrastColor
- });
- }
- };
- }
- });
- _registerModule(_modules, 'Extensions/OverlappingDataLabels.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Utilities.js']], function (Chart, U) {
- /* *
- *
- * Highcharts module to hide overlapping data labels.
- * This module is included in Highcharts.
- *
- * (c) 2009-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- fireEvent = U.fireEvent,
- isArray = U.isArray,
- isNumber = U.isNumber,
- objectEach = U.objectEach,
- pick = U.pick;
- /* eslint-disable no-invalid-this */
- // Collect potensial overlapping data labels. Stack labels probably don't need
- // to be considered because they are usually accompanied by data labels that lie
- // inside the columns.
- addEvent(Chart, 'render', function collectAndHide() {
- var labels = [];
- // Consider external label collectors
- (this.labelCollectors || []).forEach(function (collector) {
- labels = labels.concat(collector());
- });
- (this.yAxis || []).forEach(function (yAxis) {
- if (yAxis.stacking &&
- yAxis.options.stackLabels &&
- !yAxis.options.stackLabels.allowOverlap) {
- objectEach(yAxis.stacking.stacks, function (stack) {
- objectEach(stack, function (stackItem) {
- labels.push(stackItem.label);
- });
- });
- }
- });
- (this.series || []).forEach(function (series) {
- var dlOptions = series.options.dataLabels;
- if (series.visible &&
- !(dlOptions.enabled === false && !series._hasPointLabels)) { // #3866
- (series.nodes || series.points).forEach(function (point) {
- if (point.visible) {
- var dataLabels = (isArray(point.dataLabels) ?
- point.dataLabels :
- (point.dataLabel ? [point.dataLabel] : []));
- dataLabels.forEach(function (label) {
- var options = label.options;
- label.labelrank = pick(options.labelrank, point.labelrank, point.shapeArgs && point.shapeArgs.height); // #4118
- if (!options.allowOverlap) {
- labels.push(label);
- }
- });
- }
- });
- }
- });
- this.hideOverlappingLabels(labels);
- });
- /**
- * Hide overlapping labels. Labels are moved and faded in and out on zoom to
- * provide a smooth visual imression.
- *
- * @private
- * @function Highcharts.Chart#hideOverlappingLabels
- * @param {Array<Highcharts.SVGElement>} labels
- * Rendered data labels
- * @requires modules/overlapping-datalabels
- */
- Chart.prototype.hideOverlappingLabels = function (labels) {
- var chart = this,
- len = labels.length,
- ren = chart.renderer,
- label,
- i,
- j,
- label1,
- label2,
- box1,
- box2,
- isLabelAffected = false,
- isIntersectRect = function (box1,
- box2) {
- return !(box2.x >= box1.x + box1.width ||
- box2.x + box2.width <= box1.x ||
- box2.y >= box1.y + box1.height ||
- box2.y + box2.height <= box1.y);
- },
- // Get the box with its position inside the chart, as opposed to getBBox
- // that only reports the position relative to the parent.
- getAbsoluteBox = function (label) {
- var pos,
- parent,
- bBox,
- // Substract the padding if no background or border (#4333)
- padding = label.box ? 0 : (label.padding || 0),
- lineHeightCorrection = 0,
- xOffset = 0,
- boxWidth,
- alignValue;
- if (label &&
- (!label.alignAttr || label.placed)) {
- pos = label.alignAttr || {
- x: label.attr('x'),
- y: label.attr('y')
- };
- parent = label.parentGroup;
- // Get width and height if pure text nodes (stack labels)
- if (!label.width) {
- bBox = label.getBBox();
- label.width = bBox.width;
- label.height = bBox.height;
- // Labels positions are computed from top left corner, so
- // we need to substract the text height from text nodes too.
- lineHeightCorrection = ren
- .fontMetrics(null, label.element).h;
- }
- boxWidth = label.width - 2 * padding;
- alignValue = {
- left: '0',
- center: '0.5',
- right: '1'
- }[label.alignValue];
- if (alignValue) {
- xOffset = +alignValue * boxWidth;
- }
- else if (isNumber(label.x) && Math.round(label.x) !== label.translateX) {
- xOffset = label.x - label.translateX;
- }
- return {
- x: pos.x + (parent.translateX || 0) + padding -
- (xOffset || 0),
- y: pos.y + (parent.translateY || 0) + padding -
- lineHeightCorrection,
- width: label.width - 2 * padding,
- height: label.height - 2 * padding
- };
- }
- };
- for (i = 0; i < len; i++) {
- label = labels[i];
- if (label) {
- // Mark with initial opacity
- label.oldOpacity = label.opacity;
- label.newOpacity = 1;
- label.absoluteBox = getAbsoluteBox(label);
- }
- }
- // Prevent a situation in a gradually rising slope, that each label will
- // hide the previous one because the previous one always has lower rank.
- labels.sort(function (a, b) {
- return (b.labelrank || 0) - (a.labelrank || 0);
- });
- // Detect overlapping labels
- for (i = 0; i < len; i++) {
- label1 = labels[i];
- box1 = label1 && label1.absoluteBox;
- for (j = i + 1; j < len; ++j) {
- label2 = labels[j];
- box2 = label2 && label2.absoluteBox;
- if (box1 &&
- box2 &&
- label1 !== label2 && // #6465, polar chart with connectEnds
- label1.newOpacity !== 0 &&
- label2.newOpacity !== 0) {
- if (isIntersectRect(box1, box2)) {
- (label1.labelrank < label2.labelrank ? label1 : label2)
- .newOpacity = 0;
- }
- }
- }
- }
- // Hide or show
- labels.forEach(function (label) {
- var complete,
- newOpacity;
- if (label) {
- newOpacity = label.newOpacity;
- if (label.oldOpacity !== newOpacity) {
- // Make sure the label is completely hidden to avoid catching
- // clicks (#4362)
- if (label.alignAttr && label.placed) { // data labels
- label[newOpacity ? 'removeClass' : 'addClass']('highcharts-data-label-hidden');
- complete = function () {
- if (!chart.styledMode) {
- label.css({ pointerEvents: newOpacity ? 'auto' : 'none' });
- }
- label.visibility = newOpacity ? 'inherit' : 'hidden';
- };
- isLabelAffected = true;
- // Animate or set the opacity
- label.alignAttr.opacity = newOpacity;
- label[label.isOld ? 'animate' : 'attr'](label.alignAttr, null, complete);
- fireEvent(chart, 'afterHideOverlappingLabel');
- }
- else { // other labels, tick labels
- label.attr({
- opacity: newOpacity
- });
- }
- }
- label.isOld = true;
- }
- });
- if (isLabelAffected) {
- fireEvent(chart, 'afterHideAllOverlappingLabels');
- }
- };
- });
- _registerModule(_modules, 'Core/Interaction.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Legend.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js']], function (Chart, H, Legend, O, Point, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- extend = U.extend,
- fireEvent = U.fireEvent,
- isArray = U.isArray,
- isFunction = U.isFunction,
- isNumber = U.isNumber,
- isObject = U.isObject,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick;
- /**
- * @interface Highcharts.PointEventsOptionsObject
- */ /**
- * Fires when the point is selected either programmatically or following a click
- * on the point. One parameter, `event`, is passed to the function. Returning
- * `false` cancels the operation.
- * @name Highcharts.PointEventsOptionsObject#select
- * @type {Highcharts.PointSelectCallbackFunction|undefined}
- */ /**
- * Fires when the point is unselected either programmatically or following a
- * click on the point. One parameter, `event`, is passed to the function.
- * Returning `false` cancels the operation.
- * @name Highcharts.PointEventsOptionsObject#unselect
- * @type {Highcharts.PointUnselectCallbackFunction|undefined}
- */
- /**
- * Information about the select/unselect event.
- *
- * @interface Highcharts.PointInteractionEventObject
- * @extends global.Event
- */ /**
- * @name Highcharts.PointInteractionEventObject#accumulate
- * @type {boolean}
- */
- /**
- * Gets fired when the point is selected either programmatically or following a
- * click on the point.
- *
- * @callback Highcharts.PointSelectCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {Highcharts.PointInteractionEventObject} event
- * Event that occured.
- */
- /**
- * Fires when the point is unselected either programmatically or following a
- * click on the point.
- *
- * @callback Highcharts.PointUnselectCallbackFunction
- *
- * @param {Highcharts.Point} this
- * Point where the event occured.
- *
- * @param {Highcharts.PointInteractionEventObject} event
- * Event that occured.
- */
- var hasTouch = H.hasTouch,
- Series = H.Series,
- seriesTypes = H.seriesTypes,
- svg = H.svg,
- TrackerMixin;
- /* eslint-disable valid-jsdoc */
- /**
- * TrackerMixin for points and graphs.
- *
- * @private
- * @mixin Highcharts.TrackerMixin
- */
- TrackerMixin = H.TrackerMixin = {
- /**
- * Draw the tracker for a point.
- *
- * @private
- * @function Highcharts.TrackerMixin.drawTrackerPoint
- * @param {Highcharts.Series} this
- * @fires Highcharts.Series#event:afterDrawTracker
- */
- drawTrackerPoint: function () {
- var series = this,
- chart = series.chart,
- pointer = chart.pointer,
- onMouseOver = function (e) {
- var point = pointer.getPointFromEvent(e);
- // undefined on graph in scatterchart
- if (typeof point !== 'undefined') {
- pointer.isDirectTouch = true;
- point.onMouseOver(e);
- }
- }, dataLabels;
- // Add reference to the point
- series.points.forEach(function (point) {
- dataLabels = (isArray(point.dataLabels) ?
- point.dataLabels :
- (point.dataLabel ? [point.dataLabel] : []));
- if (point.graphic) {
- point.graphic.element.point = point;
- }
- dataLabels.forEach(function (dataLabel) {
- if (dataLabel.div) {
- dataLabel.div.point = point;
- }
- else {
- dataLabel.element.point = point;
- }
- });
- });
- // Add the event listeners, we need to do this only once
- if (!series._hasTracking) {
- series.trackerGroups.forEach(function (key) {
- if (series[key]) {
- // we don't always have dataLabelsGroup
- series[key]
- .addClass('highcharts-tracker')
- .on('mouseover', onMouseOver)
- .on('mouseout', function (e) {
- pointer.onTrackerMouseOut(e);
- });
- if (hasTouch) {
- series[key].on('touchstart', onMouseOver);
- }
- if (!chart.styledMode && series.options.cursor) {
- series[key]
- .css(css)
- .css({ cursor: series.options.cursor });
- }
- }
- });
- series._hasTracking = true;
- }
- fireEvent(this, 'afterDrawTracker');
- },
- /**
- * Draw the tracker object that sits above all data labels and markers to
- * track mouse events on the graph or points. For the line type charts
- * the tracker uses the same graphPath, but with a greater stroke width
- * for better control.
- *
- * @private
- * @function Highcharts.TrackerMixin.drawTrackerGraph
- * @param {Highcharts.Series} this
- * @fires Highcharts.Series#event:afterDrawTracker
- */
- drawTrackerGraph: function () {
- var series = this,
- options = series.options,
- trackByArea = options.trackByArea,
- trackerPath = [].concat(trackByArea ?
- series.areaPath :
- series.graphPath),
- // trackerPathLength = trackerPath.length,
- chart = series.chart,
- pointer = chart.pointer,
- renderer = chart.renderer,
- snap = chart.options.tooltip.snap,
- tracker = series.tracker,
- i,
- onMouseOver = function (e) {
- if (chart.hoverSeries !== series) {
- series.onMouseOver();
- }
- },
- /*
- * Empirical lowest possible opacities for TRACKER_FILL for an
- * element to stay invisible but clickable
- * IE6: 0.002
- * IE7: 0.002
- * IE8: 0.002
- * IE9: 0.00000000001 (unlimited)
- * IE10: 0.0001 (exporting only)
- * FF: 0.00000000001 (unlimited)
- * Chrome: 0.000001
- * Safari: 0.000001
- * Opera: 0.00000000001 (unlimited)
- */
- TRACKER_FILL = 'rgba(192,192,192,' + (svg ? 0.0001 : 0.002) + ')';
- // Draw the tracker
- if (tracker) {
- tracker.attr({ d: trackerPath });
- }
- else if (series.graph) { // create
- series.tracker = renderer.path(trackerPath)
- .attr({
- visibility: series.visible ? 'visible' : 'hidden',
- zIndex: 2
- })
- .addClass(trackByArea ?
- 'highcharts-tracker-area' :
- 'highcharts-tracker-line')
- .add(series.group);
- if (!chart.styledMode) {
- series.tracker.attr({
- 'stroke-linecap': 'round',
- 'stroke-linejoin': 'round',
- stroke: TRACKER_FILL,
- fill: trackByArea ? TRACKER_FILL : 'none',
- 'stroke-width': series.graph.strokeWidth() +
- (trackByArea ? 0 : 2 * snap)
- });
- }
- // The tracker is added to the series group, which is clipped, but
- // is covered by the marker group. So the marker group also needs to
- // capture events.
- [series.tracker, series.markerGroup].forEach(function (tracker) {
- tracker.addClass('highcharts-tracker')
- .on('mouseover', onMouseOver)
- .on('mouseout', function (e) {
- pointer.onTrackerMouseOut(e);
- });
- if (options.cursor && !chart.styledMode) {
- tracker.css({ cursor: options.cursor });
- }
- if (hasTouch) {
- tracker.on('touchstart', onMouseOver);
- }
- });
- }
- fireEvent(this, 'afterDrawTracker');
- }
- };
- /* End TrackerMixin */
- // Add tracking event listener to the series group, so the point graphics
- // themselves act as trackers
- if (seriesTypes.column) {
- /**
- * @private
- * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.column#drawTracker
- */
- seriesTypes.column.prototype.drawTracker = TrackerMixin.drawTrackerPoint;
- }
- if (seriesTypes.pie) {
- /**
- * @private
- * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.pie#drawTracker
- */
- seriesTypes.pie.prototype.drawTracker = TrackerMixin.drawTrackerPoint;
- }
- if (seriesTypes.scatter) {
- /**
- * @private
- * @borrows Highcharts.TrackerMixin.drawTrackerPoint as Highcharts.seriesTypes.scatter#drawTracker
- */
- seriesTypes.scatter.prototype.drawTracker = TrackerMixin.drawTrackerPoint;
- }
- // Extend Legend for item events.
- extend(Legend.prototype, {
- /**
- * @private
- * @function Highcharts.Legend#setItemEvents
- * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item
- * @param {Highcharts.SVGElement} legendItem
- * @param {boolean} [useHTML=false]
- * @fires Highcharts.Point#event:legendItemClick
- * @fires Highcharts.Series#event:legendItemClick
- */
- setItemEvents: function (item, legendItem, useHTML) {
- var legend = this,
- boxWrapper = legend.chart.renderer.boxWrapper,
- isPoint = item instanceof Point,
- activeClass = 'highcharts-legend-' +
- (isPoint ? 'point' : 'series') + '-active',
- styledMode = legend.chart.styledMode,
- // When `useHTML`, the symbol is rendered in other group, so
- // we need to apply events listeners to both places
- legendItems = useHTML ?
- [legendItem,
- item.legendSymbol] :
- [item.legendGroup];
- // Set the events on the item group, or in case of useHTML, the item
- // itself (#1249)
- legendItems.forEach(function (element) {
- if (element) {
- element
- .on('mouseover', function () {
- if (item.visible) {
- legend.allItems.forEach(function (inactiveItem) {
- if (item !== inactiveItem) {
- inactiveItem.setState('inactive', !isPoint);
- }
- });
- }
- item.setState('hover');
- // A CSS class to dim or hide other than the hovered
- // series.
- // Works only if hovered series is visible (#10071).
- if (item.visible) {
- boxWrapper.addClass(activeClass);
- }
- if (!styledMode) {
- legendItem.css(legend.options.itemHoverStyle);
- }
- })
- .on('mouseout', function () {
- if (!legend.chart.styledMode) {
- legendItem.css(merge(item.visible ?
- legend.itemStyle :
- legend.itemHiddenStyle));
- }
- legend.allItems.forEach(function (inactiveItem) {
- if (item !== inactiveItem) {
- inactiveItem.setState('', !isPoint);
- }
- });
- // A CSS class to dim or hide other than the hovered
- // series.
- boxWrapper.removeClass(activeClass);
- item.setState();
- })
- .on('click', function (event) {
- var strLegendItemClick = 'legendItemClick',
- fnLegendItemClick = function () {
- if (item.setVisible) {
- item.setVisible();
- }
- // Reset inactive state
- legend.allItems.forEach(function (inactiveItem) {
- if (item !== inactiveItem) {
- inactiveItem.setState(item.visible ? 'inactive' : '', !isPoint);
- }
- });
- };
- // A CSS class to dim or hide other than the hovered
- // series. Event handling in iOS causes the activeClass
- // to be added prior to click in some cases (#7418).
- boxWrapper.removeClass(activeClass);
- // Pass over the click/touch event. #4.
- event = {
- browserEvent: event
- };
- // click the name or symbol
- if (item.firePointEvent) { // point
- item.firePointEvent(strLegendItemClick, event, fnLegendItemClick);
- }
- else {
- fireEvent(item, strLegendItemClick, event, fnLegendItemClick);
- }
- });
- }
- });
- },
- /**
- * @private
- * @function Highcharts.Legend#createCheckboxForItem
- * @param {Highcharts.BubbleLegend|Point|Highcharts.Series} item
- * @fires Highcharts.Series#event:checkboxClick
- */
- createCheckboxForItem: function (item) {
- var legend = this;
- item.checkbox = createElement('input', {
- type: 'checkbox',
- className: 'highcharts-legend-checkbox',
- checked: item.selected,
- defaultChecked: item.selected // required by IE7
- }, legend.options.itemCheckboxStyle, legend.chart.container);
- addEvent(item.checkbox, 'click', function (event) {
- var target = event.target;
- fireEvent(item.series || item, 'checkboxClick', {
- checked: target.checked,
- item: item
- }, function () {
- item.select();
- });
- });
- }
- });
- // Extend the Chart object with interaction
- extend(Chart.prototype, /** @lends Chart.prototype */ {
- /**
- * Display the zoom button, so users can reset zoom to the default view
- * settings.
- *
- * @function Highcharts.Chart#showResetZoom
- *
- * @fires Highcharts.Chart#event:afterShowResetZoom
- * @fires Highcharts.Chart#event:beforeShowResetZoom
- */
- showResetZoom: function () {
- var chart = this,
- lang = defaultOptions.lang,
- btnOptions = chart.options.chart.resetZoomButton,
- theme = btnOptions.theme,
- states = theme.states,
- alignTo = (btnOptions.relativeTo === 'chart' ||
- btnOptions.relativeTo === 'spaceBox' ?
- null :
- 'plotBox');
- /**
- * @private
- */
- function zoomOut() {
- chart.zoomOut();
- }
- fireEvent(this, 'beforeShowResetZoom', null, function () {
- chart.resetZoomButton = chart.renderer
- .button(lang.resetZoom, null, null, zoomOut, theme, states && states.hover)
- .attr({
- align: btnOptions.position.align,
- title: lang.resetZoomTitle
- })
- .addClass('highcharts-reset-zoom')
- .add()
- .align(btnOptions.position, false, alignTo);
- });
- fireEvent(this, 'afterShowResetZoom');
- },
- /**
- * Zoom the chart out after a user has zoomed in. See also
- * [Axis.setExtremes](/class-reference/Highcharts.Axis#setExtremes).
- *
- * @function Highcharts.Chart#zoomOut
- *
- * @fires Highcharts.Chart#event:selection
- */
- zoomOut: function () {
- fireEvent(this, 'selection', { resetSelection: true }, this.zoom);
- },
- /**
- * Zoom into a given portion of the chart given by axis coordinates.
- *
- * @private
- * @function Highcharts.Chart#zoom
- * @param {Highcharts.SelectEventObject} event
- */
- zoom: function (event) {
- var chart = this,
- hasZoomed,
- pointer = chart.pointer,
- displayButton = false,
- mouseDownPos = chart.inverted ? pointer.mouseDownX : pointer.mouseDownY,
- resetZoomButton;
- // If zoom is called with no arguments, reset the axes
- if (!event || event.resetSelection) {
- chart.axes.forEach(function (axis) {
- hasZoomed = axis.zoom();
- });
- pointer.initiated = false; // #6804
- }
- else { // else, zoom in on all axes
- event.xAxis.concat(event.yAxis).forEach(function (axisData) {
- var axis = axisData.axis,
- axisStartPos = chart.inverted ? axis.left : axis.top,
- axisEndPos = chart.inverted ?
- axisStartPos + axis.width : axisStartPos + axis.height,
- isXAxis = axis.isXAxis,
- isWithinPane = false;
- // Check if zoomed area is within the pane (#1289).
- // In case of multiple panes only one pane should be zoomed.
- if ((!isXAxis &&
- mouseDownPos >= axisStartPos &&
- mouseDownPos <= axisEndPos) ||
- isXAxis ||
- !defined(mouseDownPos)) {
- isWithinPane = true;
- }
- // don't zoom more than minRange
- if (pointer[isXAxis ? 'zoomX' : 'zoomY'] && isWithinPane) {
- hasZoomed = axis.zoom(axisData.min, axisData.max);
- if (axis.displayBtn) {
- displayButton = true;
- }
- }
- });
- }
- // Show or hide the Reset zoom button
- resetZoomButton = chart.resetZoomButton;
- if (displayButton && !resetZoomButton) {
- chart.showResetZoom();
- }
- else if (!displayButton && isObject(resetZoomButton)) {
- chart.resetZoomButton = resetZoomButton.destroy();
- }
- // Redraw
- if (hasZoomed) {
- chart.redraw(pick(chart.options.chart.animation, event && event.animation, chart.pointCount < 100));
- }
- },
- /**
- * Pan the chart by dragging the mouse across the pane. This function is
- * called on mouse move, and the distance to pan is computed from chartX
- * compared to the first chartX position in the dragging operation.
- *
- * @private
- * @function Highcharts.Chart#pan
- * @param {Highcharts.PointerEventObject} e
- * @param {string} panning
- */
- pan: function (e, panning) {
- var chart = this,
- hoverPoints = chart.hoverPoints,
- panningOptions,
- chartOptions = chart.options.chart,
- hasMapNavigation = chart.options.mapNavigation &&
- chart.options.mapNavigation.enabled,
- doRedraw,
- type;
- if (typeof panning === 'object') {
- panningOptions = panning;
- }
- else {
- panningOptions = {
- enabled: panning,
- type: 'x'
- };
- }
- if (chartOptions && chartOptions.panning) {
- chartOptions.panning = panningOptions;
- }
- type = panningOptions.type;
- fireEvent(this, 'pan', { originalEvent: e }, function () {
- // remove active points for shared tooltip
- if (hoverPoints) {
- hoverPoints.forEach(function (point) {
- point.setState();
- });
- }
- // panning axis mapping
- var xy = [1]; // x
- if (type === 'xy') {
- xy = [1, 0];
- }
- else if (type === 'y') {
- xy = [0];
- }
- xy.forEach(function (isX) {
- var axis = chart[isX ? 'xAxis' : 'yAxis'][0], horiz = axis.horiz, mousePos = e[horiz ? 'chartX' : 'chartY'], mouseDown = horiz ? 'mouseDownX' : 'mouseDownY', startPos = chart[mouseDown], halfPointRange = (axis.pointRange || 0) / 2, pointRangeDirection = (axis.reversed && !chart.inverted) ||
- (!axis.reversed && chart.inverted) ?
- -1 :
- 1, extremes = axis.getExtremes(), panMin = axis.toValue(startPos - mousePos, true) +
- halfPointRange * pointRangeDirection, panMax = axis.toValue(startPos + axis.len - mousePos, true) -
- halfPointRange * pointRangeDirection, flipped = panMax < panMin, newMin = flipped ? panMax : panMin, newMax = flipped ? panMin : panMax, hasVerticalPanning = axis.hasVerticalPanning(), paddedMin, paddedMax, spill, panningState = axis.panningState;
- // General calculations of panning state.
- // This is related to using vertical panning. (#11315).
- axis.series.forEach(function (series) {
- if (hasVerticalPanning &&
- !isX && (!panningState || panningState.isDirty)) {
- var processedData = series.getProcessedData(true),
- dataExtremes = series.getExtremes(processedData.yData,
- true);
- if (!panningState) {
- panningState = {
- startMin: Number.MAX_VALUE,
- startMax: -Number.MAX_VALUE
- };
- }
- if (isNumber(dataExtremes.dataMin) &&
- isNumber(dataExtremes.dataMax)) {
- panningState.startMin = Math.min(dataExtremes.dataMin, panningState.startMin);
- panningState.startMax = Math.max(dataExtremes.dataMax, panningState.startMax);
- }
- }
- });
- paddedMin = Math.min(H.pick(panningState === null || panningState === void 0 ? void 0 : panningState.startMin, extremes.dataMin), halfPointRange ?
- extremes.min :
- axis.toValue(axis.toPixels(extremes.min) -
- axis.minPixelPadding));
- paddedMax = Math.max(H.pick(panningState === null || panningState === void 0 ? void 0 : panningState.startMax, extremes.dataMax), halfPointRange ?
- extremes.max :
- axis.toValue(axis.toPixels(extremes.max) +
- axis.minPixelPadding));
- axis.panningState = panningState;
- // It is not necessary to calculate extremes on ordinal axis,
- // because they are already calculated, so we don't want to
- // override them.
- if (!axis.isOrdinal) {
- // If the new range spills over, either to the min or max,
- // adjust the new range.
- spill = paddedMin - newMin;
- if (spill > 0) {
- newMax += spill;
- newMin = paddedMin;
- }
- spill = newMax - paddedMax;
- if (spill > 0) {
- newMax = paddedMax;
- newMin -= spill;
- }
- // Set new extremes if they are actually new
- if (axis.series.length &&
- newMin !== extremes.min &&
- newMax !== extremes.max &&
- newMin >= paddedMin &&
- newMax <= paddedMax) {
- axis.setExtremes(newMin, newMax, false, false, { trigger: 'pan' });
- if (!chart.resetZoomButton &&
- !hasMapNavigation &&
- // Show reset zoom button only when both newMin and
- // newMax values are between padded axis range.
- newMin !== paddedMin &&
- newMax !== paddedMax &&
- type.match('y')) {
- chart.showResetZoom();
- axis.displayBtn = false;
- }
- doRedraw = true;
- }
- // set new reference for next run:
- chart[mouseDown] = mousePos;
- }
- });
- if (doRedraw) {
- chart.redraw(false);
- }
- css(chart.container, { cursor: 'move' });
- });
- }
- });
- // Extend the Point object with interaction
- extend(Point.prototype, /** @lends Highcharts.Point.prototype */ {
- /**
- * Toggle the selection status of a point.
- *
- * @see Highcharts.Chart#getSelectedPoints
- *
- * @sample highcharts/members/point-select/
- * Select a point from a button
- * @sample highcharts/chart/events-selection-points/
- * Select a range of points through a drag selection
- * @sample maps/series/data-id/
- * Select a point in Highmaps
- *
- * @function Highcharts.Point#select
- *
- * @param {boolean} [selected]
- * When `true`, the point is selected. When `false`, the point is
- * unselected. When `null` or `undefined`, the selection state is toggled.
- *
- * @param {boolean} [accumulate=false]
- * When `true`, the selection is added to other selected points.
- * When `false`, other selected points are deselected. Internally in
- * Highcharts, when
- * [allowPointSelect](https://api.highcharts.com/highcharts/plotOptions.series.allowPointSelect)
- * is `true`, selected points are accumulated on Control, Shift or Cmd
- * clicking the point.
- *
- * @fires Highcharts.Point#event:select
- * @fires Highcharts.Point#event:unselect
- */
- select: function (selected, accumulate) {
- var point = this,
- series = point.series,
- chart = series.chart;
- selected = pick(selected, !point.selected);
- this.selectedStaging = selected;
- // fire the event with the default handler
- point.firePointEvent(selected ? 'select' : 'unselect', { accumulate: accumulate }, function () {
- /**
- * Whether the point is selected or not.
- *
- * @see Point#select
- * @see Chart#getSelectedPoints
- *
- * @name Highcharts.Point#selected
- * @type {boolean}
- */
- point.selected = point.options.selected = selected;
- series.options.data[series.data.indexOf(point)] =
- point.options;
- point.setState(selected && 'select');
- // unselect all other points unless Ctrl or Cmd + click
- if (!accumulate) {
- chart.getSelectedPoints().forEach(function (loopPoint) {
- var loopSeries = loopPoint.series;
- if (loopPoint.selected && loopPoint !== point) {
- loopPoint.selected = loopPoint.options.selected =
- false;
- loopSeries.options.data[loopSeries.data.indexOf(loopPoint)] = loopPoint.options;
- // Programatically selecting a point should restore
- // normal state, but when click happened on other
- // point, set inactive state to match other points
- loopPoint.setState(chart.hoverPoints &&
- loopSeries.options.inactiveOtherPoints ?
- 'inactive' : '');
- loopPoint.firePointEvent('unselect');
- }
- });
- }
- });
- delete this.selectedStaging;
- },
- /**
- * Runs on mouse over the point. Called internally from mouse and touch
- * events.
- *
- * @function Highcharts.Point#onMouseOver
- *
- * @param {Highcharts.PointerEventObject} [e]
- * The event arguments.
- */
- onMouseOver: function (e) {
- var point = this,
- series = point.series,
- chart = series.chart,
- pointer = chart.pointer;
- e = e ?
- pointer.normalize(e) :
- // In cases where onMouseOver is called directly without an event
- pointer.getChartCoordinatesFromPoint(point, chart.inverted);
- pointer.runPointActions(e, point);
- },
- /**
- * Runs on mouse out from the point. Called internally from mouse and touch
- * events.
- *
- * @function Highcharts.Point#onMouseOut
- * @fires Highcharts.Point#event:mouseOut
- */
- onMouseOut: function () {
- var point = this,
- chart = point.series.chart;
- point.firePointEvent('mouseOut');
- if (!point.series.options.inactiveOtherPoints) {
- (chart.hoverPoints || []).forEach(function (p) {
- p.setState();
- });
- }
- chart.hoverPoints = chart.hoverPoint = null;
- },
- /**
- * Import events from the series' and point's options. Only do it on
- * demand, to save processing time on hovering.
- *
- * @private
- * @function Highcharts.Point#importEvents
- */
- importEvents: function () {
- if (!this.hasImportedEvents) {
- var point = this,
- options = merge(point.series.options.point,
- point.options),
- events = options.events;
- point.events = events;
- objectEach(events, function (event, eventType) {
- if (isFunction(event)) {
- addEvent(point, eventType, event);
- }
- });
- this.hasImportedEvents = true;
- }
- },
- /**
- * Set the point's state.
- *
- * @function Highcharts.Point#setState
- *
- * @param {Highcharts.PointStateValue|""} [state]
- * The new state, can be one of `'hover'`, `'select'`, `'inactive'`,
- * or `''` (an empty string), `'normal'` or `undefined` to set to
- * normal state.
- * @param {boolean} [move]
- * State for animation.
- *
- * @fires Highcharts.Point#event:afterSetState
- */
- setState: function (state, move) {
- var point = this,
- series = point.series,
- previousState = point.state,
- stateOptions = (series.options.states[state || 'normal'] ||
- {}),
- markerOptions = (defaultOptions.plotOptions[series.type].marker &&
- series.options.marker),
- normalDisabled = (markerOptions && markerOptions.enabled === false),
- markerStateOptions = ((markerOptions &&
- markerOptions.states &&
- markerOptions.states[state || 'normal']) || {}),
- stateDisabled = markerStateOptions.enabled === false,
- stateMarkerGraphic = series.stateMarkerGraphic,
- pointMarker = point.marker || {},
- chart = series.chart,
- halo = series.halo,
- haloOptions,
- markerAttribs,
- pointAttribs,
- pointAttribsAnimation,
- hasMarkers = (markerOptions && series.markerAttribs),
- newSymbol;
- state = state || ''; // empty string
- if (
- // already has this state
- (state === point.state && !move) ||
- // selected points don't respond to hover
- (point.selected && state !== 'select') ||
- // series' state options is disabled
- (stateOptions.enabled === false) ||
- // general point marker's state options is disabled
- (state && (stateDisabled ||
- (normalDisabled &&
- markerStateOptions.enabled === false))) ||
- // individual point marker's state options is disabled
- (state &&
- pointMarker.states &&
- pointMarker.states[state] &&
- pointMarker.states[state].enabled === false) // #1610
- ) {
- return;
- }
- point.state = state;
- if (hasMarkers) {
- markerAttribs = series.markerAttribs(point, state);
- }
- // Apply hover styles to the existing point
- if (point.graphic) {
- if (previousState) {
- point.graphic.removeClass('highcharts-point-' + previousState);
- }
- if (state) {
- point.graphic.addClass('highcharts-point-' + state);
- }
- if (!chart.styledMode) {
- pointAttribs = series.pointAttribs(point, state);
- pointAttribsAnimation = pick(chart.options.chart.animation, stateOptions.animation);
- // Some inactive points (e.g. slices in pie) should apply
- // oppacity also for it's labels
- if (series.options.inactiveOtherPoints && pointAttribs.opacity) {
- (point.dataLabels || []).forEach(function (label) {
- if (label) {
- label.animate({
- opacity: pointAttribs.opacity
- }, pointAttribsAnimation);
- }
- });
- if (point.connector) {
- point.connector.animate({
- opacity: pointAttribs.opacity
- }, pointAttribsAnimation);
- }
- }
- point.graphic.animate(pointAttribs, pointAttribsAnimation);
- }
- if (markerAttribs) {
- point.graphic.animate(markerAttribs, pick(
- // Turn off globally:
- chart.options.chart.animation, markerStateOptions.animation, markerOptions.animation));
- }
- // Zooming in from a range with no markers to a range with markers
- if (stateMarkerGraphic) {
- stateMarkerGraphic.hide();
- }
- }
- else {
- // if a graphic is not applied to each point in the normal state,
- // create a shared graphic for the hover state
- if (state && markerStateOptions) {
- newSymbol = pointMarker.symbol || series.symbol;
- // If the point has another symbol than the previous one, throw
- // away the state marker graphic and force a new one (#1459)
- if (stateMarkerGraphic &&
- stateMarkerGraphic.currentSymbol !== newSymbol) {
- stateMarkerGraphic = stateMarkerGraphic.destroy();
- }
- // Add a new state marker graphic
- if (markerAttribs) {
- if (!stateMarkerGraphic) {
- if (newSymbol) {
- series.stateMarkerGraphic = stateMarkerGraphic =
- chart.renderer
- .symbol(newSymbol, markerAttribs.x, markerAttribs.y, markerAttribs.width, markerAttribs.height)
- .add(series.markerGroup);
- stateMarkerGraphic.currentSymbol = newSymbol;
- }
- // Move the existing graphic
- }
- else {
- stateMarkerGraphic[move ? 'animate' : 'attr']({
- x: markerAttribs.x,
- y: markerAttribs.y
- });
- }
- }
- if (!chart.styledMode && stateMarkerGraphic) {
- stateMarkerGraphic.attr(series.pointAttribs(point, state));
- }
- }
- if (stateMarkerGraphic) {
- stateMarkerGraphic[state && point.isInside ? 'show' : 'hide'](); // #2450
- stateMarkerGraphic.element.point = point; // #4310
- }
- }
- // Show me your halo
- haloOptions = stateOptions.halo;
- var markerGraphic = (point.graphic || stateMarkerGraphic);
- var markerVisibility = (markerGraphic && markerGraphic.visibility || 'inherit');
- if (haloOptions &&
- haloOptions.size &&
- markerGraphic &&
- markerVisibility !== 'hidden' &&
- !point.isCluster) {
- if (!halo) {
- series.halo = halo = chart.renderer.path()
- // #5818, #5903, #6705
- .add(markerGraphic.parentGroup);
- }
- halo.show()[move ? 'animate' : 'attr']({
- d: point.haloPath(haloOptions.size)
- });
- halo.attr({
- 'class': 'highcharts-halo highcharts-color-' +
- pick(point.colorIndex, series.colorIndex) +
- (point.className ? ' ' + point.className : ''),
- 'visibility': markerVisibility,
- 'zIndex': -1 // #4929, #8276
- });
- halo.point = point; // #6055
- if (!chart.styledMode) {
- halo.attr(extend({
- 'fill': point.color || series.color,
- 'fill-opacity': haloOptions.opacity
- }, haloOptions.attributes));
- }
- }
- else if (halo && halo.point && halo.point.haloPath) {
- // Animate back to 0 on the current halo point (#6055)
- halo.animate({ d: halo.point.haloPath(0) }, null,
- // Hide after unhovering. The `complete` callback runs in the
- // halo's context (#7681).
- halo.hide);
- }
- fireEvent(point, 'afterSetState');
- },
- /**
- * Get the path definition for the halo, which is usually a shadow-like
- * circle around the currently hovered point.
- *
- * @function Highcharts.Point#haloPath
- *
- * @param {number} size
- * The radius of the circular halo.
- *
- * @return {Highcharts.SVGPathArray}
- * The path definition.
- */
- haloPath: function (size) {
- var series = this.series,
- chart = series.chart;
- return chart.renderer.symbols.circle(Math.floor(this.plotX) - size, this.plotY - size, size * 2, size * 2);
- }
- });
- // Extend the Series object with interaction
- extend(Series.prototype, /** @lends Highcharts.Series.prototype */ {
- /**
- * Runs on mouse over the series graphical items.
- *
- * @function Highcharts.Series#onMouseOver
- * @fires Highcharts.Series#event:mouseOver
- */
- onMouseOver: function () {
- var series = this,
- chart = series.chart,
- hoverSeries = chart.hoverSeries,
- pointer = chart.pointer;
- pointer.setHoverChartIndex();
- // set normal state to previous series
- if (hoverSeries && hoverSeries !== series) {
- hoverSeries.onMouseOut();
- }
- // trigger the event, but to save processing time,
- // only if defined
- if (series.options.events.mouseOver) {
- fireEvent(series, 'mouseOver');
- }
- // hover this
- series.setState('hover');
- /**
- * Contains the original hovered series.
- *
- * @name Highcharts.Chart#hoverSeries
- * @type {Highcharts.Series|null}
- */
- chart.hoverSeries = series;
- },
- /**
- * Runs on mouse out of the series graphical items.
- *
- * @function Highcharts.Series#onMouseOut
- *
- * @fires Highcharts.Series#event:mouseOut
- */
- onMouseOut: function () {
- // trigger the event only if listeners exist
- var series = this,
- options = series.options,
- chart = series.chart,
- tooltip = chart.tooltip,
- hoverPoint = chart.hoverPoint;
- // #182, set to null before the mouseOut event fires
- chart.hoverSeries = null;
- // trigger mouse out on the point, which must be in this series
- if (hoverPoint) {
- hoverPoint.onMouseOut();
- }
- // fire the mouse out event
- if (series && options.events.mouseOut) {
- fireEvent(series, 'mouseOut');
- }
- // hide the tooltip
- if (tooltip &&
- !series.stickyTracking &&
- (!tooltip.shared || series.noSharedTooltip)) {
- tooltip.hide();
- }
- // Reset all inactive states
- chart.series.forEach(function (s) {
- s.setState('', true);
- });
- },
- /**
- * Set the state of the series. Called internally on mouse interaction
- * operations, but it can also be called directly to visually
- * highlight a series.
- *
- * @function Highcharts.Series#setState
- *
- * @param {Highcharts.SeriesStateValue|""} [state]
- * The new state, can be either `'hover'`, `'inactive'`, `'select'`,
- * or `''` (an empty string), `'normal'` or `undefined` to set to
- * normal state.
- * @param {boolean} [inherit]
- * Determines if state should be inherited by points too.
- */
- setState: function (state, inherit) {
- var series = this,
- options = series.options,
- graph = series.graph,
- inactiveOtherPoints = options.inactiveOtherPoints,
- stateOptions = options.states,
- lineWidth = options.lineWidth,
- opacity = options.opacity,
- // By default a quick animation to hover/inactive,
- // slower to un-hover
- stateAnimation = pick((stateOptions[state || 'normal'] &&
- stateOptions[state || 'normal'].animation),
- series.chart.options.chart.animation),
- attribs,
- i = 0;
- state = state || '';
- if (series.state !== state) {
- // Toggle class names
- [
- series.group,
- series.markerGroup,
- series.dataLabelsGroup
- ].forEach(function (group) {
- if (group) {
- // Old state
- if (series.state) {
- group.removeClass('highcharts-series-' + series.state);
- }
- // New state
- if (state) {
- group.addClass('highcharts-series-' + state);
- }
- }
- });
- series.state = state;
- if (!series.chart.styledMode) {
- if (stateOptions[state] &&
- stateOptions[state].enabled === false) {
- return;
- }
- if (state) {
- lineWidth = (stateOptions[state].lineWidth ||
- lineWidth + (stateOptions[state].lineWidthPlus || 0)); // #4035
- opacity = pick(stateOptions[state].opacity, opacity);
- }
- if (graph && !graph.dashstyle) {
- attribs = {
- 'stroke-width': lineWidth
- };
- // Animate the graph stroke-width.
- graph.animate(attribs, stateAnimation);
- while (series['zone-graph-' + i]) {
- series['zone-graph-' + i].attr(attribs);
- i = i + 1;
- }
- }
- // For some types (pie, networkgraph, sankey) opacity is
- // resolved on a point level
- if (!inactiveOtherPoints) {
- [
- series.group,
- series.markerGroup,
- series.dataLabelsGroup,
- series.labelBySeries
- ].forEach(function (group) {
- if (group) {
- group.animate({
- opacity: opacity
- }, stateAnimation);
- }
- });
- }
- }
- }
- // Don't loop over points on a series that doesn't apply inactive state
- // to siblings markers (e.g. line, column)
- if (inherit && inactiveOtherPoints && series.points) {
- series.setAllPointsToState(state);
- }
- },
- /**
- * Set the state for all points in the series.
- *
- * @function Highcharts.Series#setAllPointsToState
- *
- * @private
- *
- * @param {string} [state]
- * Can be either `hover` or undefined to set to normal state.
- */
- setAllPointsToState: function (state) {
- this.points.forEach(function (point) {
- if (point.setState) {
- point.setState(state);
- }
- });
- },
- /**
- * Show or hide the series.
- *
- * @function Highcharts.Series#setVisible
- *
- * @param {boolean} [visible]
- * True to show the series, false to hide. If undefined, the visibility is
- * toggled.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart after the series is altered. If doing more
- * operations on the chart, it is a good idea to set redraw to false and
- * call {@link Chart#redraw|chart.redraw()} after.
- *
- * @fires Highcharts.Series#event:hide
- * @fires Highcharts.Series#event:show
- */
- setVisible: function (vis, redraw) {
- var series = this,
- chart = series.chart,
- legendItem = series.legendItem,
- showOrHide,
- ignoreHiddenSeries = chart.options.chart.ignoreHiddenSeries,
- oldVisibility = series.visible;
- // if called without an argument, toggle visibility
- series.visible =
- vis =
- series.options.visible =
- series.userOptions.visible =
- typeof vis === 'undefined' ? !oldVisibility : vis; // #5618
- showOrHide = vis ? 'show' : 'hide';
- // show or hide elements
- [
- 'group',
- 'dataLabelsGroup',
- 'markerGroup',
- 'tracker',
- 'tt'
- ].forEach(function (key) {
- if (series[key]) {
- series[key][showOrHide]();
- }
- });
- // hide tooltip (#1361)
- if (chart.hoverSeries === series ||
- (chart.hoverPoint && chart.hoverPoint.series) === series) {
- series.onMouseOut();
- }
- if (legendItem) {
- chart.legend.colorizeItem(series, vis);
- }
- // rescale or adapt to resized chart
- series.isDirty = true;
- // in a stack, all other series are affected
- if (series.options.stacking) {
- chart.series.forEach(function (otherSeries) {
- if (otherSeries.options.stacking && otherSeries.visible) {
- otherSeries.isDirty = true;
- }
- });
- }
- // show or hide linked series
- series.linkedSeries.forEach(function (otherSeries) {
- otherSeries.setVisible(vis, false);
- });
- if (ignoreHiddenSeries) {
- chart.isDirtyBox = true;
- }
- fireEvent(series, showOrHide);
- if (redraw !== false) {
- chart.redraw();
- }
- },
- /**
- * Show the series if hidden.
- *
- * @sample highcharts/members/series-hide/
- * Toggle visibility from a button
- *
- * @function Highcharts.Series#show
- * @fires Highcharts.Series#event:show
- */
- show: function () {
- this.setVisible(true);
- },
- /**
- * Hide the series if visible. If the
- * [chart.ignoreHiddenSeries](https://api.highcharts.com/highcharts/chart.ignoreHiddenSeries)
- * option is true, the chart is redrawn without this series.
- *
- * @sample highcharts/members/series-hide/
- * Toggle visibility from a button
- *
- * @function Highcharts.Series#hide
- * @fires Highcharts.Series#event:hide
- */
- hide: function () {
- this.setVisible(false);
- },
- /**
- * Select or unselect the series. This means its
- * {@link Highcharts.Series.selected|selected}
- * property is set, the checkbox in the legend is toggled and when selected,
- * the series is returned by the {@link Highcharts.Chart#getSelectedSeries}
- * function.
- *
- * @sample highcharts/members/series-select/
- * Select a series from a button
- *
- * @function Highcharts.Series#select
- *
- * @param {boolean} [selected]
- * True to select the series, false to unselect. If undefined, the selection
- * state is toggled.
- *
- * @fires Highcharts.Series#event:select
- * @fires Highcharts.Series#event:unselect
- */
- select: function (selected) {
- var series = this;
- series.selected =
- selected =
- this.options.selected = (typeof selected === 'undefined' ?
- !series.selected :
- selected);
- if (series.checkbox) {
- series.checkbox.checked = selected;
- }
- fireEvent(series, selected ? 'select' : 'unselect');
- },
- /**
- * @private
- * @borrows Highcharts.TrackerMixin.drawTrackerGraph as Highcharts.Series#drawTracker
- */
- drawTracker: TrackerMixin.drawTrackerGraph
- });
- });
- _registerModule(_modules, 'Core/Responsive.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Utilities.js']], function (Chart, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var find = U.find,
- isArray = U.isArray,
- isObject = U.isObject,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- splat = U.splat,
- uniqueKey = U.uniqueKey;
- /**
- * A callback function to gain complete control on when the responsive rule
- * applies.
- *
- * @callback Highcharts.ResponsiveCallbackFunction
- *
- * @param {Highcharts.Chart} this
- * Chart context.
- *
- * @return {boolean}
- * Return `true` if it applies.
- */
- /**
- * Allows setting a set of rules to apply for different screen or chart
- * sizes. Each rule specifies additional chart options.
- *
- * @sample {highstock} stock/demo/responsive/
- * Stock chart
- * @sample highcharts/responsive/axis/
- * Axis
- * @sample highcharts/responsive/legend/
- * Legend
- * @sample highcharts/responsive/classname/
- * Class name
- *
- * @since 5.0.0
- * @apioption responsive
- */
- /**
- * A set of rules for responsive settings. The rules are executed from
- * the top down.
- *
- * @sample {highcharts} highcharts/responsive/axis/
- * Axis changes
- * @sample {highstock} highcharts/responsive/axis/
- * Axis changes
- * @sample {highmaps} highcharts/responsive/axis/
- * Axis changes
- *
- * @type {Array<*>}
- * @since 5.0.0
- * @apioption responsive.rules
- */
- /**
- * A full set of chart options to apply as overrides to the general
- * chart options. The chart options are applied when the given rule
- * is active.
- *
- * A special case is configuration objects that take arrays, for example
- * [xAxis](#xAxis), [yAxis](#yAxis) or [series](#series). For these
- * collections, an `id` option is used to map the new option set to
- * an existing object. If an existing object of the same id is not found,
- * the item of the same indexupdated. So for example, setting `chartOptions`
- * with two series items without an `id`, will cause the existing chart's
- * two series to be updated with respective options.
- *
- * @sample {highstock} stock/demo/responsive/
- * Stock chart
- * @sample highcharts/responsive/axis/
- * Axis
- * @sample highcharts/responsive/legend/
- * Legend
- * @sample highcharts/responsive/classname/
- * Class name
- *
- * @type {Highcharts.Options}
- * @since 5.0.0
- * @apioption responsive.rules.chartOptions
- */
- /**
- * Under which conditions the rule applies.
- *
- * @since 5.0.0
- * @apioption responsive.rules.condition
- */
- /**
- * A callback function to gain complete control on when the responsive
- * rule applies. Return `true` if it applies. This opens for checking
- * against other metrics than the chart size, for example the document
- * size or other elements.
- *
- * @type {Highcharts.ResponsiveCallbackFunction}
- * @since 5.0.0
- * @context Highcharts.Chart
- * @apioption responsive.rules.condition.callback
- */
- /**
- * The responsive rule applies if the chart height is less than this.
- *
- * @type {number}
- * @since 5.0.0
- * @apioption responsive.rules.condition.maxHeight
- */
- /**
- * The responsive rule applies if the chart width is less than this.
- *
- * @sample highcharts/responsive/axis/
- * Max width is 500
- *
- * @type {number}
- * @since 5.0.0
- * @apioption responsive.rules.condition.maxWidth
- */
- /**
- * The responsive rule applies if the chart height is greater than this.
- *
- * @type {number}
- * @default 0
- * @since 5.0.0
- * @apioption responsive.rules.condition.minHeight
- */
- /**
- * The responsive rule applies if the chart width is greater than this.
- *
- * @type {number}
- * @default 0
- * @since 5.0.0
- * @apioption responsive.rules.condition.minWidth
- */
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Update the chart based on the current chart/document size and options for
- * responsiveness.
- *
- * @private
- * @function Highcharts.Chart#setResponsive
- * @param {boolean} [redraw=true]
- * @param {boolean} [reset=false]
- * Reset by un-applying all rules. Chart.update resets all rules before applying
- * updated options.
- */
- Chart.prototype.setResponsive = function (redraw, reset) {
- var options = this.options.responsive,
- ruleIds = [],
- currentResponsive = this.currentResponsive,
- currentRuleIds,
- undoOptions;
- if (!reset && options && options.rules) {
- options.rules.forEach(function (rule) {
- if (typeof rule._id === 'undefined') {
- rule._id = uniqueKey();
- }
- this.matchResponsiveRule(rule, ruleIds /* , redraw */);
- }, this);
- }
- // Merge matching rules
- var mergedOptions = merge.apply(0,
- ruleIds.map(function (ruleId) {
- return find(options.rules,
- function (rule) {
- return rule._id === ruleId;
- }).chartOptions;
- }));
- mergedOptions.isResponsiveOptions = true;
- // Stringified key for the rules that currently apply.
- ruleIds = (ruleIds.toString() || void 0);
- currentRuleIds = currentResponsive && currentResponsive.ruleIds;
- // Changes in what rules apply
- if (ruleIds !== currentRuleIds) {
- // Undo previous rules. Before we apply a new set of rules, we need to
- // roll back completely to base options (#6291).
- if (currentResponsive) {
- this.update(currentResponsive.undoOptions, redraw, true);
- }
- if (ruleIds) {
- // Get undo-options for matching rules
- undoOptions = this.currentOptions(mergedOptions);
- undoOptions.isResponsiveOptions = true;
- this.currentResponsive = {
- ruleIds: ruleIds,
- mergedOptions: mergedOptions,
- undoOptions: undoOptions
- };
- this.update(mergedOptions, redraw, true);
- }
- else {
- this.currentResponsive = void 0;
- }
- }
- };
- /**
- * Handle a single responsiveness rule.
- *
- * @private
- * @function Highcharts.Chart#matchResponsiveRule
- * @param {Highcharts.ResponsiveRulesOptions} rule
- * @param {Array<string>} matches
- */
- Chart.prototype.matchResponsiveRule = function (rule, matches) {
- var condition = rule.condition,
- fn = condition.callback || function () {
- return (this.chartWidth <= pick(condition.maxWidth,
- Number.MAX_VALUE) &&
- this.chartHeight <=
- pick(condition.maxHeight,
- Number.MAX_VALUE) &&
- this.chartWidth >= pick(condition.minWidth, 0) &&
- this.chartHeight >= pick(condition.minHeight, 0));
- };
- if (fn.call(this)) {
- matches.push(rule._id);
- }
- };
- /**
- * Get the current values for a given set of options. Used before we update
- * the chart with a new responsiveness rule.
- *
- * @todo Restore axis options (by id?). The matching of items in collections
- * bears resemblance to the oneToOne matching in Chart.update. Probably we can
- * refactor out that matching and reuse it in both functions.
- *
- * @private
- * @function Highcharts.Chart#currentOptions
- * @param {Highcharts.Options} options
- * @return {Highcharts.Options}
- */
- Chart.prototype.currentOptions = function (options) {
- var chart = this,
- ret = {};
- /**
- * Recurse over a set of options and its current values,
- * and store the current values in the ret object.
- */
- function getCurrent(options, curr, ret, depth) {
- var i;
- objectEach(options, function (val, key) {
- if (!depth &&
- chart.collectionsWithUpdate.indexOf(key) > -1) {
- val = splat(val);
- ret[key] = [];
- // Iterate over collections like series, xAxis or yAxis and map
- // the items by index.
- for (i = 0; i < Math.max(val.length, curr[key].length); i++) {
- // Item exists in current data (#6347)
- if (curr[key][i]) {
- // If the item is missing from the new data, we need to
- // save the whole config structure. Like when
- // responsively updating from a dual axis layout to a
- // single axis and back (#13544).
- if (val[i] === void 0) {
- ret[key][i] = curr[key][i];
- // Otherwise, proceed
- }
- else {
- ret[key][i] = {};
- getCurrent(val[i], curr[key][i], ret[key][i], depth + 1);
- }
- }
- }
- }
- else if (isObject(val)) {
- ret[key] = isArray(val) ? [] : {};
- getCurrent(val, curr[key] || {}, ret[key], depth + 1);
- }
- else if (typeof curr[key] === 'undefined') { // #10286
- ret[key] = null;
- }
- else {
- ret[key] = curr[key];
- }
- });
- }
- getCurrent(options, this.options, ret, 0);
- return ret;
- };
- });
- _registerModule(_modules, 'masters/highcharts.src.js', [_modules['Core/Globals.js']], function (Highcharts) {
- return Highcharts;
- });
- _registerModule(_modules, 'Gantt/Tree.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2016-2020 Highsoft AS
- *
- * Authors: Jon Arild Nygard
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /* eslint no-console: 0 */
- var extend = U.extend,
- isNumber = U.isNumber,
- pick = U.pick;
- /**
- * Creates an object map from parent id to childrens index.
- *
- * @private
- * @function Highcharts.Tree#getListOfParents
- *
- * @param {Array<*>} data
- * List of points set in options. `Array.parent` is parent id of point.
- *
- * @param {Array<string>} ids
- * List of all point ids.
- *
- * @return {Highcharts.Dictionary<Array<*>>}
- * Map from parent id to children index in data
- */
- var getListOfParents = function (data,
- ids) {
- var listOfParents = data.reduce(function (prev,
- curr) {
- var parent = pick(curr.parent, '');
- if (typeof prev[parent] === 'undefined') {
- prev[parent] = [];
- }
- prev[parent].push(curr);
- return prev;
- }, {}), parents = Object.keys(listOfParents);
- // If parent does not exist, hoist parent to root of tree.
- parents.forEach(function (parent, list) {
- var children = listOfParents[parent];
- if ((parent !== '') && (ids.indexOf(parent) === -1)) {
- children.forEach(function (child) {
- list[''].push(child);
- });
- delete list[parent];
- }
- });
- return listOfParents;
- };
- var getNode = function (id,
- parent,
- level,
- data,
- mapOfIdToChildren,
- options) {
- var descendants = 0,
- height = 0,
- after = options && options.after,
- before = options && options.before,
- node = {
- data: data,
- depth: level - 1,
- id: id,
- level: level,
- parent: parent
- },
- start,
- end,
- children;
- // Allow custom logic before the children has been created.
- if (typeof before === 'function') {
- before(node, options);
- }
- // Call getNode recursively on the children. Calulate the height of the
- // node, and the number of descendants.
- children = ((mapOfIdToChildren[id] || [])).map(function (child) {
- var node = getNode(child.id,
- id, (level + 1),
- child,
- mapOfIdToChildren,
- options),
- childStart = child.start,
- childEnd = (child.milestone === true ?
- childStart :
- child.end);
- // Start should be the lowest child.start.
- start = ((!isNumber(start) || childStart < start) ?
- childStart :
- start);
- // End should be the largest child.end.
- // If child is milestone, then use start as end.
- end = ((!isNumber(end) || childEnd > end) ?
- childEnd :
- end);
- descendants = descendants + 1 + node.descendants;
- height = Math.max(node.height + 1, height);
- return node;
- });
- // Calculate start and end for point if it is not already explicitly set.
- if (data) {
- data.start = pick(data.start, start);
- data.end = pick(data.end, end);
- }
- extend(node, {
- children: children,
- descendants: descendants,
- height: height
- });
- // Allow custom logic after the children has been created.
- if (typeof after === 'function') {
- after(node, options);
- }
- return node;
- };
- var getTree = function (data,
- options) {
- var ids = data.map(function (d) {
- return d.id;
- }), mapOfIdToChildren = getListOfParents(data, ids);
- return getNode('', null, 1, null, mapOfIdToChildren, options);
- };
- var Tree = {
- getListOfParents: getListOfParents,
- getNode: getNode,
- getTree: getTree
- };
- return Tree;
- });
- _registerModule(_modules, 'Core/Axis/TreeGridTick.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Authors: Jon Arild Nygard
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- defined = U.defined,
- isObject = U.isObject,
- isNumber = U.isNumber,
- pick = U.pick,
- wrap = U.wrap;
- /**
- * @private
- */
- var TreeGridTick;
- (function (TreeGridTick) {
- /* *
- *
- * Interfaces
- *
- * */
- /* *
- *
- * Variables
- *
- * */
- var applied = false;
- /* *
- *
- * Functions
- *
- * */
- /**
- * @private
- */
- function compose(TickClass) {
- if (!applied) {
- addEvent(TickClass, 'init', onInit);
- wrap(TickClass.prototype, 'getLabelPosition', wrapGetLabelPosition);
- wrap(TickClass.prototype, 'renderLabel', wrapRenderLabel);
- // backwards compatibility
- TickClass.prototype.collapse = function (redraw) {
- this.treeGrid.collapse(redraw);
- };
- TickClass.prototype.expand = function (redraw) {
- this.treeGrid.expand(redraw);
- };
- TickClass.prototype.toggleCollapse = function (redraw) {
- this.treeGrid.toggleCollapse(redraw);
- };
- applied = true;
- }
- }
- TreeGridTick.compose = compose;
- /**
- * @private
- */
- function onInit() {
- var tick = this;
- if (!tick.treeGrid) {
- tick.treeGrid = new Additions(tick);
- }
- }
- /**
- * @private
- */
- function onTickHover(label) {
- label.addClass('highcharts-treegrid-node-active');
- if (!label.renderer.styledMode) {
- label.css({
- textDecoration: 'underline'
- });
- }
- }
- /**
- * @private
- */
- function onTickHoverExit(label, options) {
- var css = defined(options.style) ? options.style : {};
- label.removeClass('highcharts-treegrid-node-active');
- if (!label.renderer.styledMode) {
- label.css({ textDecoration: css.textDecoration });
- }
- }
- /**
- * @private
- */
- function renderLabelIcon(tick, params) {
- var treeGrid = tick.treeGrid,
- isNew = !treeGrid.labelIcon,
- renderer = params.renderer,
- labelBox = params.xy,
- options = params.options,
- width = options.width,
- height = options.height,
- iconCenter = {
- x: labelBox.x - (width / 2) - options.padding,
- y: labelBox.y - (height / 2)
- },
- rotation = params.collapsed ? 90 : 180,
- shouldRender = params.show && isNumber(iconCenter.y);
- var icon = treeGrid.labelIcon;
- if (!icon) {
- treeGrid.labelIcon = icon = renderer
- .path(renderer.symbols[options.type](options.x, options.y, width, height))
- .addClass('highcharts-label-icon')
- .add(params.group);
- }
- // Set the new position, and show or hide
- if (!shouldRender) {
- icon.attr({ y: -9999 }); // #1338
- }
- // Presentational attributes
- if (!renderer.styledMode) {
- icon
- .attr({
- 'stroke-width': 1,
- 'fill': pick(params.color, '#666666')
- })
- .css({
- cursor: 'pointer',
- stroke: options.lineColor,
- strokeWidth: options.lineWidth
- });
- }
- // Update the icon positions
- icon[isNew ? 'attr' : 'animate']({
- translateX: iconCenter.x,
- translateY: iconCenter.y,
- rotation: rotation
- });
- }
- /**
- * @private
- */
- function wrapGetLabelPosition(proceed, x, y, label, horiz, labelOptions, tickmarkOffset, index, step) {
- var tick = this,
- lbOptions = pick(tick.options && tick.options.labels,
- labelOptions),
- pos = tick.pos,
- axis = tick.axis,
- options = axis.options,
- isTreeGrid = options.type === 'treegrid',
- result = proceed.apply(tick,
- [x,
- y,
- label,
- horiz,
- lbOptions,
- tickmarkOffset,
- index,
- step]);
- var symbolOptions,
- indentation,
- mapOfPosToGridNode,
- node,
- level;
- if (isTreeGrid) {
- symbolOptions = (lbOptions && isObject(lbOptions.symbol, true) ?
- lbOptions.symbol :
- {});
- indentation = (lbOptions && isNumber(lbOptions.indentation) ?
- lbOptions.indentation :
- 0);
- mapOfPosToGridNode = axis.treeGrid.mapOfPosToGridNode;
- node = mapOfPosToGridNode && mapOfPosToGridNode[pos];
- level = (node && node.depth) || 1;
- result.x += (
- // Add space for symbols
- ((symbolOptions.width) + (symbolOptions.padding * 2)) +
- // Apply indentation
- ((level - 1) * indentation));
- }
- return result;
- }
- /**
- * @private
- */
- function wrapRenderLabel(proceed) {
- var tick = this, pos = tick.pos, axis = tick.axis, label = tick.label, mapOfPosToGridNode = axis.treeGrid.mapOfPosToGridNode, options = axis.options, labelOptions = pick(tick.options && tick.options.labels, options && options.labels), symbolOptions = (labelOptions && isObject(labelOptions.symbol, true) ?
- labelOptions.symbol :
- {}), node = mapOfPosToGridNode && mapOfPosToGridNode[pos], level = node && node.depth, isTreeGrid = options.type === 'treegrid', shouldRender = axis.tickPositions.indexOf(pos) > -1, prefixClassName = 'highcharts-treegrid-node-', styledMode = axis.chart.styledMode;
- var collapsed,
- addClassName,
- removeClassName;
- if (isTreeGrid && node) {
- // Add class name for hierarchical styling.
- if (label &&
- label.element) {
- label.addClass(prefixClassName + 'level-' + level);
- }
- }
- proceed.apply(tick, Array.prototype.slice.call(arguments, 1));
- if (isTreeGrid &&
- label &&
- label.element &&
- node &&
- node.descendants &&
- node.descendants > 0) {
- collapsed = axis.treeGrid.isCollapsed(node);
- renderLabelIcon(tick, {
- color: !styledMode && label.styles && label.styles.color || '',
- collapsed: collapsed,
- group: label.parentGroup,
- options: symbolOptions,
- renderer: label.renderer,
- show: shouldRender,
- xy: label.xy
- });
- // Add class name for the node.
- addClassName = prefixClassName +
- (collapsed ? 'collapsed' : 'expanded');
- removeClassName = prefixClassName +
- (collapsed ? 'expanded' : 'collapsed');
- label
- .addClass(addClassName)
- .removeClass(removeClassName);
- if (!styledMode) {
- label.css({
- cursor: 'pointer'
- });
- }
- // Add events to both label text and icon
- [label, tick.treeGrid.labelIcon].forEach(function (object) {
- if (object && !object.attachedTreeGridEvents) {
- // On hover
- addEvent(object.element, 'mouseover', function () {
- onTickHover(label);
- });
- // On hover out
- addEvent(object.element, 'mouseout', function () {
- onTickHoverExit(label, labelOptions);
- });
- addEvent(object.element, 'click', function () {
- tick.treeGrid.toggleCollapse();
- });
- object.attachedTreeGridEvents = true;
- }
- });
- }
- }
- /* *
- *
- * Classes
- *
- * */
- /**
- * @private
- * @class
- */
- var Additions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- /**
- * @private
- */
- function Additions(tick) {
- this.tick = tick;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Collapse the grid cell. Used when axis is of type treegrid.
- *
- * @see gantt/treegrid-axis/collapsed-dynamically/demo.js
- *
- * @private
- * @function Highcharts.Tick#collapse
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call to
- * {@link Highcharts.Chart#redraw}
- */
- Additions.prototype.collapse = function (redraw) {
- var tick = this.tick,
- axis = tick.axis,
- brokenAxis = axis.brokenAxis;
- if (brokenAxis &&
- axis.treeGrid.mapOfPosToGridNode) {
- var pos = tick.pos,
- node = axis.treeGrid.mapOfPosToGridNode[pos],
- breaks = axis.treeGrid.collapse(node);
- brokenAxis.setBreaks(breaks, pick(redraw, true));
- }
- };
- /**
- * Expand the grid cell. Used when axis is of type treegrid.
- *
- * @see gantt/treegrid-axis/collapsed-dynamically/demo.js
- *
- * @private
- * @function Highcharts.Tick#expand
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call to
- * {@link Highcharts.Chart#redraw}
- */
- Additions.prototype.expand = function (redraw) {
- var tick = this.tick,
- axis = tick.axis,
- brokenAxis = axis.brokenAxis;
- if (brokenAxis &&
- axis.treeGrid.mapOfPosToGridNode) {
- var pos = tick.pos,
- node = axis.treeGrid.mapOfPosToGridNode[pos],
- breaks = axis.treeGrid.expand(node);
- brokenAxis.setBreaks(breaks, pick(redraw, true));
- }
- };
- /**
- * Toggle the collapse/expand state of the grid cell. Used when axis is
- * of type treegrid.
- *
- * @see gantt/treegrid-axis/collapsed-dynamically/demo.js
- *
- * @private
- * @function Highcharts.Tick#toggleCollapse
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart or wait for an explicit call to
- * {@link Highcharts.Chart#redraw}
- */
- Additions.prototype.toggleCollapse = function (redraw) {
- var tick = this.tick,
- axis = tick.axis,
- brokenAxis = axis.brokenAxis;
- if (brokenAxis &&
- axis.treeGrid.mapOfPosToGridNode) {
- var pos = tick.pos,
- node = axis.treeGrid.mapOfPosToGridNode[pos],
- breaks = axis.treeGrid.toggleCollapse(node);
- brokenAxis.setBreaks(breaks, pick(redraw, true));
- }
- };
- return Additions;
- }());
- TreeGridTick.Additions = Additions;
- })(TreeGridTick || (TreeGridTick = {}));
- return TreeGridTick;
- });
- _registerModule(_modules, 'Mixins/TreeSeries.js', [_modules['Core/Color.js'], _modules['Core/Utilities.js']], function (Color, U) {
- /* *
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var extend = U.extend,
- isArray = U.isArray,
- isNumber = U.isNumber,
- isObject = U.isObject,
- merge = U.merge,
- pick = U.pick;
- var isBoolean = function (x) {
- return typeof x === 'boolean';
- }, isFn = function (x) {
- return typeof x === 'function';
- };
- /* eslint-disable valid-jsdoc */
- /**
- * @todo Combine buildTree and buildNode with setTreeValues
- * @todo Remove logic from Treemap and make it utilize this mixin.
- * @private
- */
- var setTreeValues = function setTreeValues(tree,
- options) {
- var before = options.before,
- idRoot = options.idRoot,
- mapIdToNode = options.mapIdToNode,
- nodeRoot = mapIdToNode[idRoot],
- levelIsConstant = (isBoolean(options.levelIsConstant) ?
- options.levelIsConstant :
- true),
- points = options.points,
- point = points[tree.i],
- optionsPoint = point && point.options || {},
- childrenTotal = 0,
- children = [],
- value;
- extend(tree, {
- levelDynamic: tree.level - (levelIsConstant ? 0 : nodeRoot.level),
- name: pick(point && point.name, ''),
- visible: (idRoot === tree.id ||
- (isBoolean(options.visible) ? options.visible : false))
- });
- if (isFn(before)) {
- tree = before(tree, options);
- }
- // First give the children some values
- tree.children.forEach(function (child, i) {
- var newOptions = extend({},
- options);
- extend(newOptions, {
- index: i,
- siblings: tree.children.length,
- visible: tree.visible
- });
- child = setTreeValues(child, newOptions);
- children.push(child);
- if (child.visible) {
- childrenTotal += child.val;
- }
- });
- tree.visible = childrenTotal > 0 || tree.visible;
- // Set the values
- value = pick(optionsPoint.value, childrenTotal);
- extend(tree, {
- children: children,
- childrenTotal: childrenTotal,
- isLeaf: tree.visible && !childrenTotal,
- val: value
- });
- return tree;
- };
- /**
- * @private
- */
- var getColor = function getColor(node,
- options) {
- var index = options.index,
- mapOptionsToLevel = options.mapOptionsToLevel,
- parentColor = options.parentColor,
- parentColorIndex = options.parentColorIndex,
- series = options.series,
- colors = options.colors,
- siblings = options.siblings,
- points = series.points,
- getColorByPoint,
- chartOptionsChart = series.chart.options.chart,
- point,
- level,
- colorByPoint,
- colorIndexByPoint,
- color,
- colorIndex;
- /**
- * @private
- */
- function variation(color) {
- var colorVariation = level && level.colorVariation;
- if (colorVariation) {
- if (colorVariation.key === 'brightness') {
- return Color.parse(color).brighten(colorVariation.to * (index / siblings)).get();
- }
- }
- return color;
- }
- if (node) {
- point = points[node.i];
- level = mapOptionsToLevel[node.level] || {};
- getColorByPoint = point && level.colorByPoint;
- if (getColorByPoint) {
- colorIndexByPoint = point.index % (colors ?
- colors.length :
- chartOptionsChart.colorCount);
- colorByPoint = colors && colors[colorIndexByPoint];
- }
- // Select either point color, level color or inherited color.
- if (!series.chart.styledMode) {
- color = pick(point && point.options.color, level && level.color, colorByPoint, parentColor && variation(parentColor), series.color);
- }
- colorIndex = pick(point && point.options.colorIndex, level && level.colorIndex, colorIndexByPoint, parentColorIndex, options.colorIndex);
- }
- return {
- color: color,
- colorIndex: colorIndex
- };
- };
- /**
- * Creates a map from level number to its given options.
- *
- * @private
- * @function getLevelOptions
- * @param {object} params
- * Object containing parameters.
- * - `defaults` Object containing default options. The default options
- * are merged with the userOptions to get the final options for a
- * specific level.
- * - `from` The lowest level number.
- * - `levels` User options from series.levels.
- * - `to` The highest level number.
- * @return {Highcharts.Dictionary<object>|null}
- * Returns a map from level number to its given options.
- */
- var getLevelOptions = function getLevelOptions(params) {
- var result = null,
- defaults,
- converted,
- i,
- from,
- to,
- levels;
- if (isObject(params)) {
- result = {};
- from = isNumber(params.from) ? params.from : 1;
- levels = params.levels;
- converted = {};
- defaults = isObject(params.defaults) ? params.defaults : {};
- if (isArray(levels)) {
- converted = levels.reduce(function (obj, item) {
- var level,
- levelIsConstant,
- options;
- if (isObject(item) && isNumber(item.level)) {
- options = merge({}, item);
- levelIsConstant = (isBoolean(options.levelIsConstant) ?
- options.levelIsConstant :
- defaults.levelIsConstant);
- // Delete redundant properties.
- delete options.levelIsConstant;
- delete options.level;
- // Calculate which level these options apply to.
- level = item.level + (levelIsConstant ? 0 : from - 1);
- if (isObject(obj[level])) {
- extend(obj[level], options);
- }
- else {
- obj[level] = options;
- }
- }
- return obj;
- }, {});
- }
- to = isNumber(params.to) ? params.to : 1;
- for (i = 0; i <= to; i++) {
- result[i] = merge({}, defaults, isObject(converted[i]) ? converted[i] : {});
- }
- }
- return result;
- };
- /**
- * Update the rootId property on the series. Also makes sure that it is
- * accessible to exporting.
- *
- * @private
- * @function updateRootId
- *
- * @param {object} series
- * The series to operate on.
- *
- * @return {string}
- * Returns the resulting rootId after update.
- */
- var updateRootId = function (series) {
- var rootId,
- options;
- if (isObject(series)) {
- // Get the series options.
- options = isObject(series.options) ? series.options : {};
- // Calculate the rootId.
- rootId = pick(series.rootNode, options.rootId, '');
- // Set rootId on series.userOptions to pick it up in exporting.
- if (isObject(series.userOptions)) {
- series.userOptions.rootId = rootId;
- }
- // Set rootId on series to pick it up on next update.
- series.rootNode = rootId;
- }
- return rootId;
- };
- var result = {
- getColor: getColor,
- getLevelOptions: getLevelOptions,
- setTreeValues: setTreeValues,
- updateRootId: updateRootId
- };
- return result;
- });
- _registerModule(_modules, 'Core/Axis/GridAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Axis/Tick.js'], _modules['Core/Utilities.js']], function (Axis, H, O, Tick, U) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Authors: Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var dateFormat = O.dateFormat;
- var addEvent = U.addEvent,
- defined = U.defined,
- erase = U.erase,
- find = U.find,
- isArray = U.isArray,
- isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- timeUnits = U.timeUnits,
- wrap = U.wrap;
- var argsToArray = function (args) {
- return Array.prototype.slice.call(args, 1);
- }, isObject = function (x) {
- // Always use strict mode
- return U.isObject(x, true);
- }, Chart = H.Chart;
- var applyGridOptions = function applyGridOptions(axis) {
- var options = axis.options;
- // Center-align by default
- if (!options.labels) {
- options.labels = {};
- }
- options.labels.align = pick(options.labels.align, 'center');
- // @todo: Check against tickLabelPlacement between/on etc
- /* Prevents adding the last tick label if the axis is not a category
- axis.
- Since numeric labels are normally placed at starts and ends of a
- range of value, and this module makes the label point at the value,
- an "extra" label would appear. */
- if (!axis.categories) {
- options.showLastLabel = false;
- }
- // Prevents rotation of labels when squished, as rotating them would not
- // help.
- axis.labelRotation = 0;
- options.labels.rotation = 0;
- };
- /**
- * For a datetime axis, the scale will automatically adjust to the
- * appropriate unit. This member gives the default string
- * representations used for each unit. For intermediate values,
- * different units may be used, for example the `day` unit can be used
- * on midnight and `hour` unit be used for intermediate values on the
- * same axis.
- * For grid axes (like in Gantt charts),
- * it is possible to declare as a list to provide different
- * formats depending on available space.
- * For an overview of the replacement codes, see
- * [dateFormat](/class-reference/Highcharts#dateFormat).
- *
- * Defaults to:
- * ```js
- * {
- hour: {
- list: ['%H:%M', '%H']
- },
- day: {
- list: ['%A, %e. %B', '%a, %e. %b', '%E']
- },
- week: {
- list: ['Week %W', 'W%W']
- },
- month: {
- list: ['%B', '%b', '%o']
- }
- },
- * ```
- *
- * @sample {gantt} gantt/demo/left-axis-table
- * Gantt Chart with custom axis date format.
- *
- * @product gantt
- * @apioption xAxis.dateTimeLabelFormats
- */
- /**
- * Set grid options for the axis labels. Requires Highcharts Gantt.
- *
- * @since 6.2.0
- * @product gantt
- * @apioption xAxis.grid
- */
- /**
- * Enable grid on the axis labels. Defaults to true for Gantt charts.
- *
- * @type {boolean}
- * @default true
- * @since 6.2.0
- * @product gantt
- * @apioption xAxis.grid.enabled
- */
- /**
- * Set specific options for each column (or row for horizontal axes) in the
- * grid. Each extra column/row is its own axis, and the axis options can be set
- * here.
- *
- * @sample gantt/demo/left-axis-table
- * Left axis as a table
- *
- * @type {Array<Highcharts.XAxisOptions>}
- * @apioption xAxis.grid.columns
- */
- /**
- * Set border color for the label grid lines.
- *
- * @type {Highcharts.ColorString}
- * @apioption xAxis.grid.borderColor
- */
- /**
- * Set border width of the label grid lines.
- *
- * @type {number}
- * @default 1
- * @apioption xAxis.grid.borderWidth
- */
- /**
- * Set cell height for grid axis labels. By default this is calculated from font
- * size. This option only applies to horizontal axes.
- *
- * @sample gantt/grid-axis/cellheight
- * Gant chart with custom cell height
- * @type {number}
- * @apioption xAxis.grid.cellHeight
- */
- ''; // detach doclets above
- /**
- * Get the largest label width and height.
- *
- * @private
- * @function Highcharts.Axis#getMaxLabelDimensions
- *
- * @param {Highcharts.Dictionary<Highcharts.Tick>} ticks
- * All the ticks on one axis.
- *
- * @param {Array<number|string>} tickPositions
- * All the tick positions on one axis.
- *
- * @return {Highcharts.SizeObject}
- * Object containing the properties height and width.
- *
- * @todo Move this to the generic axis implementation, as it is used there.
- */
- Axis.prototype.getMaxLabelDimensions = function (ticks, tickPositions) {
- var dimensions = {
- width: 0,
- height: 0
- };
- tickPositions.forEach(function (pos) {
- var tick = ticks[pos],
- tickHeight = 0,
- tickWidth = 0,
- label;
- if (isObject(tick)) {
- label = isObject(tick.label) ? tick.label : {};
- // Find width and height of tick
- tickHeight = label.getBBox ? label.getBBox().height : 0;
- if (label.textStr) {
- // Set the tickWidth same as the label width after ellipsis
- // applied #10281
- tickWidth = Math.round(label.getBBox().width);
- }
- // Update the result if width and/or height are larger
- dimensions.height = Math.max(tickHeight, dimensions.height);
- dimensions.width = Math.max(tickWidth, dimensions.width);
- }
- });
- return dimensions;
- };
- // Adds week date format
- H.dateFormats.W = function (timestamp) {
- var d = new this.Date(timestamp);
- var firstDay = (this.get('Day',
- d) + 6) % 7;
- var thursday = new this.Date(d.valueOf());
- this.set('Date', thursday, this.get('Date', d) - firstDay + 3);
- var firstThursday = new this.Date(this.get('FullYear',
- thursday), 0, 1);
- if (this.get('Day', firstThursday) !== 4) {
- this.set('Month', d, 0);
- this.set('Date', d, 1 + (11 - this.get('Day', firstThursday)) % 7);
- }
- return (1 +
- Math.floor((thursday.valueOf() - firstThursday.valueOf()) / 604800000)).toString();
- };
- // First letter of the day of the week, e.g. 'M' for 'Monday'.
- H.dateFormats.E = function (timestamp) {
- return dateFormat('%a', timestamp, true).charAt(0);
- };
- /* eslint-disable no-invalid-this */
- addEvent(Chart, 'afterSetChartSize', function () {
- this.axes.forEach(function (axis) {
- (axis.grid && axis.grid.columns || []).forEach(function (column) {
- column.setAxisSize();
- column.setAxisTranslation();
- });
- });
- });
- // Center tick labels in cells.
- addEvent(Tick, 'afterGetLabelPosition', function (e) {
- var tick = this,
- label = tick.label,
- axis = tick.axis,
- reversed = axis.reversed,
- chart = axis.chart,
- options = axis.options,
- gridOptions = options.grid || {},
- labelOpts = axis.options.labels,
- align = labelOpts.align,
- // verticalAlign is currently not supported for axis.labels.
- verticalAlign = 'middle', // labelOpts.verticalAlign,
- side = GridAxis.Side[axis.side],
- tickmarkOffset = e.tickmarkOffset,
- tickPositions = axis.tickPositions,
- tickPos = tick.pos - tickmarkOffset,
- nextTickPos = (isNumber(tickPositions[e.index + 1]) ?
- tickPositions[e.index + 1] - tickmarkOffset :
- axis.max + tickmarkOffset),
- tickSize = axis.tickSize('tick'),
- tickWidth = tickSize ? tickSize[0] : 0,
- crispCorr = tickSize ? tickSize[1] / 2 : 0,
- labelHeight,
- lblMetrics,
- lines,
- bottom,
- top,
- left,
- right;
- // Only center tick labels in grid axes
- if (gridOptions.enabled === true) {
- // Calculate top and bottom positions of the cell.
- if (side === 'top') {
- bottom = axis.top + axis.offset;
- top = bottom - tickWidth;
- }
- else if (side === 'bottom') {
- top = chart.chartHeight - axis.bottom + axis.offset;
- bottom = top + tickWidth;
- }
- else {
- bottom = axis.top + axis.len - axis.translate(reversed ? nextTickPos : tickPos);
- top = axis.top + axis.len - axis.translate(reversed ? tickPos : nextTickPos);
- }
- // Calculate left and right positions of the cell.
- if (side === 'right') {
- left = chart.chartWidth - axis.right + axis.offset;
- right = left + tickWidth;
- }
- else if (side === 'left') {
- right = axis.left + axis.offset;
- left = right - tickWidth;
- }
- else {
- left = Math.round(axis.left + axis.translate(reversed ? nextTickPos : tickPos)) - crispCorr;
- right = Math.round(axis.left + axis.translate(reversed ? tickPos : nextTickPos)) - crispCorr;
- }
- tick.slotWidth = right - left;
- // Calculate the positioning of the label based on
- // alignment.
- e.pos.x = (align === 'left' ?
- left :
- align === 'right' ?
- right :
- left + ((right - left) / 2) // default to center
- );
- e.pos.y = (verticalAlign === 'top' ?
- top :
- verticalAlign === 'bottom' ?
- bottom :
- top + ((bottom - top) / 2) // default to middle
- );
- lblMetrics = chart.renderer.fontMetrics(labelOpts.style.fontSize, label.element);
- labelHeight = label.getBBox().height;
- // Adjustment to y position to align the label correctly.
- // Would be better to have a setter or similar for this.
- if (!labelOpts.useHTML) {
- lines = Math.round(labelHeight / lblMetrics.h);
- e.pos.y += (
- // Center the label
- // TODO: why does this actually center the label?
- ((lblMetrics.b - (lblMetrics.h - lblMetrics.f)) / 2) +
- // Adjust for height of additional lines.
- -(((lines - 1) * lblMetrics.h) / 2));
- }
- else {
- e.pos.y += (
- // Readjust yCorr in htmlUpdateTransform
- lblMetrics.b +
- // Adjust for height of html label
- -(labelHeight / 2));
- }
- e.pos.x += (axis.horiz && labelOpts.x || 0);
- }
- });
- /* eslint-enable no-invalid-this */
- /**
- * Additions for grid axes.
- * @private
- * @class
- */
- var GridAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function GridAxisAdditions(axis) {
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Checks if an axis is the outer axis in its dimension. Since
- * axes are placed outwards in order, the axis with the highest
- * index is the outermost axis.
- *
- * Example: If there are multiple x-axes at the top of the chart,
- * this function returns true if the axis supplied is the last
- * of the x-axes.
- *
- * @private
- *
- * @return {boolean}
- * True if the axis is the outermost axis in its dimension; false if
- * not.
- */
- GridAxisAdditions.prototype.isOuterAxis = function () {
- var axis = this.axis;
- var chart = axis.chart;
- var columnIndex = axis.grid.columnIndex;
- var columns = (axis.linkedParent && axis.linkedParent.grid.columns ||
- axis.grid.columns);
- var parentAxis = columnIndex ? axis.linkedParent : axis;
- var thisIndex = -1,
- lastIndex = 0;
- chart[axis.coll].forEach(function (otherAxis, index) {
- if (otherAxis.side === axis.side && !otherAxis.options.isInternal) {
- lastIndex = index;
- if (otherAxis === parentAxis) {
- // Get the index of the axis in question
- thisIndex = index;
- }
- }
- });
- return (lastIndex === thisIndex &&
- (isNumber(columnIndex) ? columns.length === columnIndex : true));
- };
- return GridAxisAdditions;
- }());
- /**
- * Axis with grid support.
- * @private
- * @class
- */
- var GridAxis = /** @class */ (function () {
- function GridAxis() {
- }
- /* *
- *
- * Static Functions
- *
- * */
- /* eslint-disable valid-jsdoc */
- /**
- * Extends axis class with grid support.
- * @private
- */
- GridAxis.compose = function (AxisClass) {
- Axis.keepProps.push('grid');
- wrap(AxisClass.prototype, 'unsquish', GridAxis.wrapUnsquish);
- // Add event handlers
- addEvent(AxisClass, 'init', GridAxis.onInit);
- addEvent(AxisClass, 'afterGetOffset', GridAxis.onAfterGetOffset);
- addEvent(AxisClass, 'afterGetTitlePosition', GridAxis.onAfterGetTitlePosition);
- addEvent(AxisClass, 'afterInit', GridAxis.onAfterInit);
- addEvent(AxisClass, 'afterRender', GridAxis.onAfterRender);
- addEvent(AxisClass, 'afterSetAxisTranslation', GridAxis.onAfterSetAxisTranslation);
- addEvent(AxisClass, 'afterSetOptions', GridAxis.onAfterSetOptions);
- addEvent(AxisClass, 'afterSetOptions', GridAxis.onAfterSetOptions2);
- addEvent(AxisClass, 'afterSetScale', GridAxis.onAfterSetScale);
- addEvent(AxisClass, 'afterTickSize', GridAxis.onAfterTickSize);
- addEvent(AxisClass, 'trimTicks', GridAxis.onTrimTicks);
- addEvent(AxisClass, 'destroy', GridAxis.onDestroy);
- };
- /**
- * Handle columns and getOffset.
- * @private
- */
- GridAxis.onAfterGetOffset = function () {
- var grid = this.grid;
- (grid && grid.columns || []).forEach(function (column) {
- column.getOffset();
- });
- };
- /**
- * @private
- */
- GridAxis.onAfterGetTitlePosition = function (e) {
- var axis = this;
- var options = axis.options;
- var gridOptions = options.grid || {};
- if (gridOptions.enabled === true) {
- // compute anchor points for each of the title align options
- var title = axis.axisTitle,
- axisHeight = axis.height,
- horiz = axis.horiz,
- axisLeft = axis.left,
- offset = axis.offset,
- opposite = axis.opposite,
- _a = axis.options.title,
- axisTitleOptions = _a === void 0 ? {} : _a,
- axisTop = axis.top,
- axisWidth = axis.width;
- var tickSize = axis.tickSize();
- var titleWidth = title && title.getBBox().width;
- var xOption = axisTitleOptions.x || 0;
- var yOption = axisTitleOptions.y || 0;
- var titleMargin = pick(axisTitleOptions.margin,
- horiz ? 5 : 10);
- var titleFontSize = axis.chart.renderer.fontMetrics(axisTitleOptions.style &&
- axisTitleOptions.style.fontSize,
- title).f;
- var crispCorr = tickSize ? tickSize[0] / 2 : 0;
- // TODO account for alignment
- // the position in the perpendicular direction of the axis
- var offAxis = ((horiz ? axisTop + axisHeight : axisLeft) +
- (horiz ? 1 : -1) * // horizontal axis reverses the margin
- (opposite ? -1 : 1) * // so does opposite axes
- crispCorr +
- (axis.side === GridAxis.Side.bottom ? titleFontSize : 0));
- e.titlePosition.x = horiz ?
- axisLeft - titleWidth / 2 - titleMargin + xOption :
- offAxis + (opposite ? axisWidth : 0) + offset + xOption;
- e.titlePosition.y = horiz ?
- (offAxis -
- (opposite ? axisHeight : 0) +
- (opposite ? titleFontSize : -titleFontSize) / 2 +
- offset +
- yOption) :
- axisTop - titleMargin + yOption;
- }
- };
- /**
- * @private
- */
- GridAxis.onAfterInit = function () {
- var axis = this;
- var chart = axis.chart,
- _a = axis.options.grid,
- gridOptions = _a === void 0 ? {} : _a,
- userOptions = axis.userOptions;
- if (gridOptions.enabled) {
- applyGridOptions(axis);
- /* eslint-disable no-invalid-this */
- // TODO: wrap the axis instead
- wrap(axis, 'labelFormatter', function (proceed) {
- var _a = this,
- axis = _a.axis,
- value = _a.value;
- var tickPos = axis.tickPositions;
- var series = (axis.isLinked ?
- axis.linkedParent :
- axis).series[0];
- var isFirst = value === tickPos[0];
- var isLast = value === tickPos[tickPos.length - 1];
- var point = series && find(series.options.data,
- function (p) {
- return p[axis.isXAxis ? 'x' : 'y'] === value;
- });
- // Make additional properties available for the
- // formatter
- this.isFirst = isFirst;
- this.isLast = isLast;
- this.point = point;
- // Call original labelFormatter
- return proceed.call(this);
- });
- /* eslint-enable no-invalid-this */
- }
- if (gridOptions.columns) {
- var columns = axis.grid.columns = [],
- columnIndex = axis.grid.columnIndex = 0;
- // Handle columns, each column is a grid axis
- while (++columnIndex < gridOptions.columns.length) {
- var columnOptions = merge(userOptions,
- gridOptions.columns[gridOptions.columns.length - columnIndex - 1], {
- linkedTo: 0,
- // Force to behave like category axis
- type: 'category',
- // Disable by default the scrollbar on the grid axis
- scrollbar: {
- enabled: false
- }
- });
- delete columnOptions.grid.columns; // Prevent recursion
- var column = new Axis(axis.chart,
- columnOptions);
- column.grid.isColumn = true;
- column.grid.columnIndex = columnIndex;
- // Remove column axis from chart axes array, and place it
- // in the columns array.
- erase(chart.axes, column);
- erase(chart[axis.coll], column);
- columns.push(column);
- }
- }
- };
- /**
- * Draw an extra line on the far side of the outermost axis,
- * creating floor/roof/wall of a grid. And some padding.
- * ```
- * Make this:
- * (axis.min) __________________________ (axis.max)
- * | | | | |
- * Into this:
- * (axis.min) __________________________ (axis.max)
- * ___|____|____|____|____|__
- * ```
- * @private
- */
- GridAxis.onAfterRender = function () {
- var axis = this;
- var grid = axis.grid;
- var options = axis.options;
- var renderer = axis.chart.renderer;
- var gridOptions = options.grid || {};
- var yStartIndex,
- yEndIndex,
- xStartIndex,
- xEndIndex;
- if (gridOptions.enabled === true) {
- // @todo acutual label padding (top, bottom, left, right)
- axis.maxLabelDimensions = axis.getMaxLabelDimensions(axis.ticks, axis.tickPositions);
- // Remove right wall before rendering if updating
- if (axis.rightWall) {
- axis.rightWall.destroy();
- }
- /*
- Draw an extra axis line on outer axes
- >
- Make this: |______|______|______|___
- > _________________________
- Into this: |______|______|______|__|
- */
- if (axis.grid && axis.grid.isOuterAxis() && axis.axisLine) {
- var lineWidth = options.lineWidth;
- if (lineWidth) {
- var linePath = axis.getLinePath(lineWidth);
- var startPoint = linePath[0];
- var endPoint = linePath[1];
- // Negate distance if top or left axis
- // Subtract 1px to draw the line at the end of the tick
- var tickLength = (axis.tickSize('tick') || [1])[0];
- var distance = (tickLength - 1) * ((axis.side === GridAxis.Side.top ||
- axis.side === GridAxis.Side.left) ? -1 : 1);
- // If axis is horizontal, reposition line path vertically
- if (startPoint[0] === 'M' && endPoint[0] === 'L') {
- if (axis.horiz) {
- startPoint[2] += distance;
- endPoint[2] += distance;
- }
- else {
- // If axis is vertical, reposition line path
- // horizontally
- startPoint[1] += distance;
- endPoint[1] += distance;
- }
- }
- if (!axis.grid.axisLineExtra) {
- axis.grid.axisLineExtra = renderer
- .path(linePath)
- .attr({
- zIndex: 7
- })
- .addClass('highcharts-axis-line')
- .add(axis.axisGroup);
- if (!renderer.styledMode) {
- axis.grid.axisLineExtra.attr({
- stroke: options.lineColor,
- 'stroke-width': lineWidth
- });
- }
- }
- else {
- axis.grid.axisLineExtra.animate({
- d: linePath
- });
- }
- // show or hide the line depending on
- // options.showEmpty
- axis.axisLine[axis.showAxis ? 'show' : 'hide'](true);
- }
- }
- (grid && grid.columns || []).forEach(function (column) {
- column.render();
- });
- }
- };
- /**
- * @private
- */
- GridAxis.onAfterSetAxisTranslation = function () {
- var axis = this;
- var tickInfo = axis.tickPositions && axis.tickPositions.info;
- var options = axis.options;
- var gridOptions = options.grid || {};
- var userLabels = axis.userOptions.labels || {};
- if (axis.horiz) {
- if (gridOptions.enabled === true) {
- axis.series.forEach(function (series) {
- series.options.pointRange = 0;
- });
- }
- // Lower level time ticks, like hours or minutes, represent
- // points in time and not ranges. These should be aligned
- // left in the grid cell by default. The same applies to
- // years of higher order.
- if (tickInfo &&
- options.dateTimeLabelFormats &&
- options.labels &&
- !defined(userLabels.align) &&
- (options.dateTimeLabelFormats[tickInfo.unitName].range === false ||
- tickInfo.count > 1 // years
- )) {
- options.labels.align = 'left';
- if (!defined(userLabels.x)) {
- options.labels.x = 3;
- }
- }
- }
- };
- /**
- * Creates a left and right wall on horizontal axes:
- * - Places leftmost tick at the start of the axis, to create a left
- * wall
- * - Ensures that the rightmost tick is at the end of the axis, to
- * create a right wall.
- * @private
- */
- GridAxis.onAfterSetOptions = function (e) {
- var options = this.options,
- userOptions = e.userOptions,
- gridAxisOptions,
- gridOptions = ((options && isObject(options.grid)) ? options.grid : {});
- if (gridOptions.enabled === true) {
- // Merge the user options into default grid axis options so
- // that when a user option is set, it takes presedence.
- gridAxisOptions = merge(true, {
- className: ('highcharts-grid-axis ' + (userOptions.className || '')),
- dateTimeLabelFormats: {
- hour: {
- list: ['%H:%M', '%H']
- },
- day: {
- list: ['%A, %e. %B', '%a, %e. %b', '%E']
- },
- week: {
- list: ['Week %W', 'W%W']
- },
- month: {
- list: ['%B', '%b', '%o']
- }
- },
- grid: {
- borderWidth: 1
- },
- labels: {
- padding: 2,
- style: {
- fontSize: '13px'
- }
- },
- margin: 0,
- title: {
- text: null,
- reserveSpace: false,
- rotation: 0
- },
- // In a grid axis, only allow one unit of certain types,
- // for example we shouln't have one grid cell spanning
- // two days.
- units: [[
- 'millisecond',
- [1, 10, 100]
- ], [
- 'second',
- [1, 10]
- ], [
- 'minute',
- [1, 5, 15]
- ], [
- 'hour',
- [1, 6]
- ], [
- 'day',
- [1]
- ], [
- 'week',
- [1]
- ], [
- 'month',
- [1]
- ], [
- 'year',
- null
- ]]
- }, userOptions);
- // X-axis specific options
- if (this.coll === 'xAxis') {
- // For linked axes, tickPixelInterval is used only if
- // the tickPositioner below doesn't run or returns
- // undefined (like multiple years)
- if (defined(userOptions.linkedTo) &&
- !defined(userOptions.tickPixelInterval)) {
- gridAxisOptions.tickPixelInterval = 350;
- }
- // For the secondary grid axis, use the primary axis'
- // tick intervals and return ticks one level higher.
- if (
- // Check for tick pixel interval in options
- !defined(userOptions.tickPixelInterval) &&
- // Only for linked axes
- defined(userOptions.linkedTo) &&
- !defined(userOptions.tickPositioner) &&
- !defined(userOptions.tickInterval)) {
- gridAxisOptions.tickPositioner = function (min, max) {
- var parentInfo = (this.linkedParent &&
- this.linkedParent.tickPositions &&
- this.linkedParent.tickPositions.info);
- if (parentInfo) {
- var unitIdx,
- count,
- unitName,
- i,
- units = gridAxisOptions.units,
- unitRange;
- for (i = 0; i < units.length; i++) {
- if (units[i][0] ===
- parentInfo.unitName) {
- unitIdx = i;
- break;
- }
- }
- // Get the first allowed count on the next
- // unit.
- if (units[unitIdx + 1]) {
- unitName = units[unitIdx + 1][0];
- count =
- (units[unitIdx + 1][1] || [1])[0];
- // In case the base X axis shows years, make
- // the secondary axis show ten times the
- // years (#11427)
- }
- else if (parentInfo.unitName === 'year') {
- unitName = 'year';
- count = parentInfo.count * 10;
- }
- unitRange = timeUnits[unitName];
- this.tickInterval = unitRange * count;
- return this.getTimeTicks({
- unitRange: unitRange,
- count: count,
- unitName: unitName
- }, min, max, this.options.startOfWeek);
- }
- };
- }
- }
- // Now merge the combined options into the axis options
- merge(true, this.options, gridAxisOptions);
- if (this.horiz) {
- /* _________________________
- Make this: ___|_____|_____|_____|__|
- ^ ^
- _________________________
- Into this: |_____|_____|_____|_____|
- ^ ^ */
- options.minPadding = pick(userOptions.minPadding, 0);
- options.maxPadding = pick(userOptions.maxPadding, 0);
- }
- // If borderWidth is set, then use its value for tick and
- // line width.
- if (isNumber(options.grid.borderWidth)) {
- options.tickWidth = options.lineWidth = gridOptions.borderWidth;
- }
- }
- };
- /**
- * @private
- */
- GridAxis.onAfterSetOptions2 = function (e) {
- var axis = this;
- var userOptions = e.userOptions;
- var gridOptions = userOptions && userOptions.grid || {};
- var columns = gridOptions.columns;
- // Add column options to the parent axis. Children has their column
- // options set on init in onGridAxisAfterInit.
- if (gridOptions.enabled && columns) {
- merge(true, axis.options, columns[columns.length - 1]);
- }
- };
- /**
- * Handle columns and setScale.
- * @private
- */
- GridAxis.onAfterSetScale = function () {
- var axis = this;
- (axis.grid.columns || []).forEach(function (column) {
- column.setScale();
- });
- };
- /**
- * Draw vertical axis ticks extra long to create cell floors and roofs.
- * Overrides the tickLength for vertical axes.
- * @private
- */
- GridAxis.onAfterTickSize = function (e) {
- var defaultLeftAxisOptions = Axis.defaultLeftAxisOptions;
- var _a = this,
- horiz = _a.horiz,
- maxLabelDimensions = _a.maxLabelDimensions,
- _b = _a.options.grid,
- gridOptions = _b === void 0 ? {} : _b;
- if (gridOptions.enabled && maxLabelDimensions) {
- var labelPadding = (Math.abs(defaultLeftAxisOptions.labels.x) * 2);
- var distance = horiz ?
- gridOptions.cellHeight || labelPadding + maxLabelDimensions.height :
- labelPadding + maxLabelDimensions.width;
- if (isArray(e.tickSize)) {
- e.tickSize[0] = distance;
- }
- else {
- e.tickSize = [distance, 0];
- }
- }
- };
- /**
- * @private
- */
- GridAxis.onDestroy = function (e) {
- var grid = this.grid;
- (grid.columns || []).forEach(function (column) {
- column.destroy(e.keepEvents);
- });
- grid.columns = void 0;
- };
- /**
- * Wraps axis init to draw cell walls on vertical axes.
- * @private
- */
- GridAxis.onInit = function (e) {
- var axis = this;
- var userOptions = e.userOptions || {};
- var gridOptions = userOptions.grid || {};
- if (gridOptions.enabled && defined(gridOptions.borderColor)) {
- userOptions.tickColor = userOptions.lineColor = gridOptions.borderColor;
- }
- if (!axis.grid) {
- axis.grid = new GridAxisAdditions(axis);
- }
- };
- /**
- * Makes tick labels which are usually ignored in a linked axis
- * displayed if they are within range of linkedParent.min.
- * ```
- * _____________________________
- * | | | | |
- * Make this: | | 2 | 3 | 4 |
- * |___|_______|_______|_______|
- * ^
- * _____________________________
- * | | | | |
- * Into this: | 1 | 2 | 3 | 4 |
- * |___|_______|_______|_______|
- * ^
- * ```
- * @private
- * @todo Does this function do what the drawing says? Seems to affect
- * ticks and not the labels directly?
- */
- GridAxis.onTrimTicks = function () {
- var axis = this;
- var options = axis.options;
- var gridOptions = options.grid || {};
- var categoryAxis = axis.categories;
- var tickPositions = axis.tickPositions;
- var firstPos = tickPositions[0];
- var lastPos = tickPositions[tickPositions.length - 1];
- var linkedMin = axis.linkedParent && axis.linkedParent.min;
- var linkedMax = axis.linkedParent && axis.linkedParent.max;
- var min = linkedMin || axis.min;
- var max = linkedMax || axis.max;
- var tickInterval = axis.tickInterval;
- var endMoreThanMin = (firstPos < min &&
- firstPos + tickInterval > min);
- var startLessThanMax = (lastPos > max &&
- lastPos - tickInterval < max);
- if (gridOptions.enabled === true &&
- !categoryAxis &&
- (axis.horiz || axis.isLinked)) {
- if (endMoreThanMin && !options.startOnTick) {
- tickPositions[0] = min;
- }
- if (startLessThanMax && !options.endOnTick) {
- tickPositions[tickPositions.length - 1] = max;
- }
- }
- };
- /**
- * Avoid altering tickInterval when reserving space.
- * @private
- */
- GridAxis.wrapUnsquish = function (proceed) {
- var axis = this;
- var _a = axis.options.grid,
- gridOptions = _a === void 0 ? {} : _a;
- if (gridOptions.enabled === true && axis.categories) {
- return axis.tickInterval;
- }
- return proceed.apply(axis, argsToArray(arguments));
- };
- return GridAxis;
- }());
- (function (GridAxis) {
- /**
- * Enum for which side the axis is on. Maps to axis.side.
- * @private
- */
- var Side;
- (function (Side) {
- Side[Side["top"] = 0] = "top";
- Side[Side["right"] = 1] = "right";
- Side[Side["bottom"] = 2] = "bottom";
- Side[Side["left"] = 3] = "left";
- })(Side = GridAxis.Side || (GridAxis.Side = {}));
- })(GridAxis || (GridAxis = {}));
- GridAxis.compose(Axis);
- return GridAxis;
- });
- _registerModule(_modules, 'Core/Axis/BrokenAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js'], _modules['Extensions/Stacking.js']], function (Axis, H, U, StackItem) {
- /* *
- *
- * (c) 2009-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- find = U.find,
- fireEvent = U.fireEvent,
- isArray = U.isArray,
- isNumber = U.isNumber,
- pick = U.pick;
- var Series = H.Series;
- /* eslint-disable valid-jsdoc */
- /**
- * Provides support for broken axes.
- * @private
- * @class
- */
- var BrokenAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function BrokenAxisAdditions(axis) {
- this.hasBreaks = false;
- this.axis = axis;
- }
- /* *
- *
- * Static Functions
- *
- * */
- /**
- * @private
- */
- BrokenAxisAdditions.isInBreak = function (brk, val) {
- var ret,
- repeat = brk.repeat || Infinity,
- from = brk.from,
- length = brk.to - brk.from,
- test = (val >= from ?
- (val - from) % repeat :
- repeat - ((from - val) % repeat));
- if (!brk.inclusive) {
- ret = test < length && test !== 0;
- }
- else {
- ret = test <= length;
- }
- return ret;
- };
- /**
- * @private
- */
- BrokenAxisAdditions.lin2Val = function (val) {
- var axis = this;
- var brokenAxis = axis.brokenAxis;
- var breakArray = brokenAxis && brokenAxis.breakArray;
- if (!breakArray) {
- return val;
- }
- var nval = val,
- brk,
- i;
- for (i = 0; i < breakArray.length; i++) {
- brk = breakArray[i];
- if (brk.from >= nval) {
- break;
- }
- else if (brk.to < nval) {
- nval += brk.len;
- }
- else if (BrokenAxisAdditions.isInBreak(brk, nval)) {
- nval += brk.len;
- }
- }
- return nval;
- };
- /**
- * @private
- */
- BrokenAxisAdditions.val2Lin = function (val) {
- var axis = this;
- var brokenAxis = axis.brokenAxis;
- var breakArray = brokenAxis && brokenAxis.breakArray;
- if (!breakArray) {
- return val;
- }
- var nval = val,
- brk,
- i;
- for (i = 0; i < breakArray.length; i++) {
- brk = breakArray[i];
- if (brk.to <= val) {
- nval -= brk.len;
- }
- else if (brk.from >= val) {
- break;
- }
- else if (BrokenAxisAdditions.isInBreak(brk, val)) {
- nval -= (val - brk.from);
- break;
- }
- }
- return nval;
- };
- /* *
- *
- * Functions
- *
- * */
- /**
- * Returns the first break found where the x is larger then break.from and
- * smaller then break.to.
- *
- * @param {number} x
- * The number which should be within a break.
- *
- * @param {Array<Highcharts.XAxisBreaksOptions>} breaks
- * The array of breaks to search within.
- *
- * @return {Highcharts.XAxisBreaksOptions|undefined}
- * Returns the first break found that matches, returns false if no break is
- * found.
- */
- BrokenAxisAdditions.prototype.findBreakAt = function (x, breaks) {
- return find(breaks, function (b) {
- return b.from < x && x < b.to;
- });
- };
- /**
- * @private
- */
- BrokenAxisAdditions.prototype.isInAnyBreak = function (val, testKeep) {
- var brokenAxis = this;
- var axis = brokenAxis.axis;
- var breaks = axis.options.breaks,
- i = breaks && breaks.length,
- inbrk,
- keep,
- ret;
- if (i) {
- while (i--) {
- if (BrokenAxisAdditions.isInBreak(breaks[i], val)) {
- inbrk = true;
- if (!keep) {
- keep = pick(breaks[i].showPoints, !axis.isXAxis);
- }
- }
- }
- if (inbrk && testKeep) {
- ret = inbrk && !keep;
- }
- else {
- ret = inbrk;
- }
- }
- return ret;
- };
- /**
- * Dynamically set or unset breaks in an axis. This function in lighter than
- * usin Axis.update, and it also preserves animation.
- *
- * @private
- * @function Highcharts.Axis#setBreaks
- *
- * @param {Array<Highcharts.XAxisBreaksOptions>} [breaks]
- * The breaks to add. When `undefined` it removes existing breaks.
- *
- * @param {boolean} [redraw=true]
- * Whether to redraw the chart immediately.
- *
- * @return {void}
- */
- BrokenAxisAdditions.prototype.setBreaks = function (breaks, redraw) {
- var brokenAxis = this;
- var axis = brokenAxis.axis;
- var hasBreaks = (isArray(breaks) && !!breaks.length);
- axis.isDirty = brokenAxis.hasBreaks !== hasBreaks;
- brokenAxis.hasBreaks = hasBreaks;
- axis.options.breaks = axis.userOptions.breaks = breaks;
- axis.forceRedraw = true; // Force recalculation in setScale
- // Recalculate series related to the axis.
- axis.series.forEach(function (series) {
- series.isDirty = true;
- });
- if (!hasBreaks && axis.val2lin === BrokenAxisAdditions.val2Lin) {
- // Revert to prototype functions
- delete axis.val2lin;
- delete axis.lin2val;
- }
- if (hasBreaks) {
- axis.userOptions.ordinal = false;
- axis.lin2val = BrokenAxisAdditions.lin2Val;
- axis.val2lin = BrokenAxisAdditions.val2Lin;
- axis.setExtremes = function (newMin, newMax, redraw, animation, eventArguments) {
- // If trying to set extremes inside a break, extend min to
- // after, and max to before the break ( #3857 )
- if (brokenAxis.hasBreaks) {
- var axisBreak,
- breaks = this.options.breaks;
- while ((axisBreak = brokenAxis.findBreakAt(newMin, breaks))) {
- newMin = axisBreak.to;
- }
- while ((axisBreak = brokenAxis.findBreakAt(newMax, breaks))) {
- newMax = axisBreak.from;
- }
- // If both min and max is within the same break.
- if (newMax < newMin) {
- newMax = newMin;
- }
- }
- Axis.prototype.setExtremes.call(this, newMin, newMax, redraw, animation, eventArguments);
- };
- axis.setAxisTranslation = function (saveOld) {
- Axis.prototype.setAxisTranslation.call(this, saveOld);
- brokenAxis.unitLength = null;
- if (brokenAxis.hasBreaks) {
- var breaks = axis.options.breaks || [],
- // Temporary one:
- breakArrayT = [],
- breakArray = [],
- length = 0,
- inBrk,
- repeat,
- min = axis.userMin || axis.min,
- max = axis.userMax || axis.max,
- pointRangePadding = pick(axis.pointRangePadding, 0),
- start,
- i;
- // Min & max check (#4247)
- breaks.forEach(function (brk) {
- repeat = brk.repeat || Infinity;
- if (BrokenAxisAdditions.isInBreak(brk, min)) {
- min +=
- (brk.to % repeat) -
- (min % repeat);
- }
- if (BrokenAxisAdditions.isInBreak(brk, max)) {
- max -=
- (max % repeat) -
- (brk.from % repeat);
- }
- });
- // Construct an array holding all breaks in the axis
- breaks.forEach(function (brk) {
- start = brk.from;
- repeat = brk.repeat || Infinity;
- while (start - repeat > min) {
- start -= repeat;
- }
- while (start < min) {
- start += repeat;
- }
- for (i = start; i < max; i += repeat) {
- breakArrayT.push({
- value: i,
- move: 'in'
- });
- breakArrayT.push({
- value: i + (brk.to - brk.from),
- move: 'out',
- size: brk.breakSize
- });
- }
- });
- breakArrayT.sort(function (a, b) {
- return ((a.value === b.value) ?
- ((a.move === 'in' ? 0 : 1) -
- (b.move === 'in' ? 0 : 1)) :
- a.value - b.value);
- });
- // Simplify the breaks
- inBrk = 0;
- start = min;
- breakArrayT.forEach(function (brk) {
- inBrk += (brk.move === 'in' ? 1 : -1);
- if (inBrk === 1 && brk.move === 'in') {
- start = brk.value;
- }
- if (inBrk === 0) {
- breakArray.push({
- from: start,
- to: brk.value,
- len: brk.value - start - (brk.size || 0)
- });
- length += brk.value - start - (brk.size || 0);
- }
- });
- /**
- * HC <= 8 backwards compatibility, used by demo samples.
- * @deprecated
- * @private
- * @requires modules/broken-axis
- */
- axis.breakArray = brokenAxis.breakArray = breakArray;
- // Used with staticScale, and below the actual axis length,
- // when breaks are substracted.
- brokenAxis.unitLength = max - min - length + pointRangePadding;
- fireEvent(axis, 'afterBreaks');
- if (axis.staticScale) {
- axis.transA = axis.staticScale;
- }
- else if (brokenAxis.unitLength) {
- axis.transA *=
- (max - axis.min + pointRangePadding) /
- brokenAxis.unitLength;
- }
- if (pointRangePadding) {
- axis.minPixelPadding =
- axis.transA * axis.minPointOffset;
- }
- axis.min = min;
- axis.max = max;
- }
- };
- }
- if (pick(redraw, true)) {
- axis.chart.redraw();
- }
- };
- return BrokenAxisAdditions;
- }());
- /**
- * Axis with support of broken data rows.
- * @private
- * @class
- */
- var BrokenAxis = /** @class */ (function () {
- function BrokenAxis() {
- }
- /**
- * Adds support for broken axes.
- * @private
- */
- BrokenAxis.compose = function (AxisClass, SeriesClass) {
- AxisClass.keepProps.push('brokenAxis');
- var seriesProto = Series.prototype;
- /**
- * @private
- */
- seriesProto.drawBreaks = function (axis, keys) {
- var series = this,
- points = series.points,
- breaks,
- threshold,
- eventName,
- y;
- if (axis && // #5950
- axis.brokenAxis &&
- axis.brokenAxis.hasBreaks) {
- var brokenAxis_1 = axis.brokenAxis;
- keys.forEach(function (key) {
- breaks = brokenAxis_1 && brokenAxis_1.breakArray || [];
- threshold = axis.isXAxis ?
- axis.min :
- pick(series.options.threshold, axis.min);
- points.forEach(function (point) {
- y = pick(point['stack' + key.toUpperCase()], point[key]);
- breaks.forEach(function (brk) {
- if (isNumber(threshold) && isNumber(y)) {
- eventName = false;
- if ((threshold < brk.from && y > brk.to) ||
- (threshold > brk.from && y < brk.from)) {
- eventName = 'pointBreak';
- }
- else if ((threshold < brk.from && y > brk.from && y < brk.to) ||
- (threshold > brk.from && y > brk.to && y < brk.from)) {
- eventName = 'pointInBreak';
- }
- if (eventName) {
- fireEvent(axis, eventName, { point: point, brk: brk });
- }
- }
- });
- });
- });
- }
- };
- /**
- * Extend getGraphPath by identifying gaps in the data so that we can
- * draw a gap in the line or area. This was moved from ordinal axis
- * module to broken axis module as of #5045.
- *
- * @private
- * @function Highcharts.Series#gappedPath
- *
- * @return {Highcharts.SVGPathArray}
- * Gapped path
- */
- seriesProto.gappedPath = function () {
- var currentDataGrouping = this.currentDataGrouping,
- groupingSize = currentDataGrouping && currentDataGrouping.gapSize,
- gapSize = this.options.gapSize,
- points = this.points.slice(),
- i = points.length - 1,
- yAxis = this.yAxis,
- stack;
- /**
- * Defines when to display a gap in the graph, together with the
- * [gapUnit](plotOptions.series.gapUnit) option.
- *
- * In case when `dataGrouping` is enabled, points can be grouped
- * into a larger time span. This can make the grouped points to have
- * a greater distance than the absolute value of `gapSize` property,
- * which will result in disappearing graph completely. To prevent
- * this situation the mentioned distance between grouped points is
- * used instead of previously defined `gapSize`.
- *
- * In practice, this option is most often used to visualize gaps in
- * time series. In a stock chart, intraday data is available for
- * daytime hours, while gaps will appear in nights and weekends.
- *
- * @see [gapUnit](plotOptions.series.gapUnit)
- * @see [xAxis.breaks](#xAxis.breaks)
- *
- * @sample {highstock} stock/plotoptions/series-gapsize/
- * Setting the gap size to 2 introduces gaps for weekends
- * in daily datasets.
- *
- * @type {number}
- * @default 0
- * @product highstock
- * @requires modules/broken-axis
- * @apioption plotOptions.series.gapSize
- */
- /**
- * Together with [gapSize](plotOptions.series.gapSize), this option
- * defines where to draw gaps in the graph.
- *
- * When the `gapUnit` is `"relative"` (default), a gap size of 5
- * means that if the distance between two points is greater than
- * 5 times that of the two closest points, the graph will be broken.
- *
- * When the `gapUnit` is `"value"`, the gap is based on absolute
- * axis values, which on a datetime axis is milliseconds. This also
- * applies to the navigator series that inherits gap options from
- * the base series.
- *
- * @see [gapSize](plotOptions.series.gapSize)
- *
- * @type {string}
- * @default relative
- * @since 5.0.13
- * @product highstock
- * @validvalue ["relative", "value"]
- * @requires modules/broken-axis
- * @apioption plotOptions.series.gapUnit
- */
- if (gapSize && i > 0) { // #5008
- // Gap unit is relative
- if (this.options.gapUnit !== 'value') {
- gapSize *= this.basePointRange;
- }
- // Setting a new gapSize in case dataGrouping is enabled (#7686)
- if (groupingSize &&
- groupingSize > gapSize &&
- // Except when DG is forced (e.g. from other series)
- // and has lower granularity than actual points (#11351)
- groupingSize >= this.basePointRange) {
- gapSize = groupingSize;
- }
- // extension for ordinal breaks
- var current = void 0,
- next = void 0;
- while (i--) {
- // Reassign next if it is not visible
- if (!(next && next.visible !== false)) {
- next = points[i + 1];
- }
- current = points[i];
- // Skip iteration if one of the points is not visible
- if (next.visible === false || current.visible === false) {
- continue;
- }
- if (next.x - current.x > gapSize) {
- var xRange = (current.x + next.x) / 2;
- points.splice(// insert after this one
- i + 1, 0, {
- isNull: true,
- x: xRange
- });
- // For stacked chart generate empty stack items, #6546
- if (yAxis.stacking && this.options.stacking) {
- stack = yAxis.stacking.stacks[this.stackKey][xRange] =
- new StackItem(yAxis, yAxis.options
- .stackLabels, false, xRange, this.stack);
- stack.total = 0;
- }
- }
- // Assign current to next for the upcoming iteration
- next = current;
- }
- }
- // Call base method
- return this.getGraphPath(points);
- };
- /* eslint-disable no-invalid-this */
- addEvent(AxisClass, 'init', function () {
- var axis = this;
- if (!axis.brokenAxis) {
- axis.brokenAxis = new BrokenAxisAdditions(axis);
- }
- });
- addEvent(AxisClass, 'afterInit', function () {
- if (typeof this.brokenAxis !== 'undefined') {
- this.brokenAxis.setBreaks(this.options.breaks, false);
- }
- });
- addEvent(AxisClass, 'afterSetTickPositions', function () {
- var axis = this;
- var brokenAxis = axis.brokenAxis;
- if (brokenAxis &&
- brokenAxis.hasBreaks) {
- var tickPositions = this.tickPositions,
- info = this.tickPositions.info,
- newPositions = [],
- i;
- for (i = 0; i < tickPositions.length; i++) {
- if (!brokenAxis.isInAnyBreak(tickPositions[i])) {
- newPositions.push(tickPositions[i]);
- }
- }
- this.tickPositions = newPositions;
- this.tickPositions.info = info;
- }
- });
- // Force Axis to be not-ordinal when breaks are defined
- addEvent(AxisClass, 'afterSetOptions', function () {
- if (this.brokenAxis && this.brokenAxis.hasBreaks) {
- this.options.ordinal = false;
- }
- });
- addEvent(SeriesClass, 'afterGeneratePoints', function () {
- var _a = this,
- isDirty = _a.isDirty,
- connectNulls = _a.options.connectNulls,
- points = _a.points,
- xAxis = _a.xAxis,
- yAxis = _a.yAxis;
- // Set, or reset visibility of the points. Axis.setBreaks marks the
- // series as isDirty
- if (isDirty) {
- var i = points.length;
- while (i--) {
- var point = points[i];
- // Respect nulls inside the break (#4275)
- var nullGap = point.y === null && connectNulls === false;
- var isPointInBreak = (!nullGap && ((xAxis &&
- xAxis.brokenAxis &&
- xAxis.brokenAxis.isInAnyBreak(point.x,
- true)) || (yAxis &&
- yAxis.brokenAxis &&
- yAxis.brokenAxis.isInAnyBreak(point.y,
- true))));
- // Set point.visible if in any break.
- // If not in break, reset visible to original value.
- point.visible = isPointInBreak ?
- false :
- point.options.visible !== false;
- }
- }
- });
- addEvent(SeriesClass, 'afterRender', function drawPointsWrapped() {
- this.drawBreaks(this.xAxis, ['x']);
- this.drawBreaks(this.yAxis, pick(this.pointArrayMap, ['y']));
- });
- };
- return BrokenAxis;
- }());
- BrokenAxis.compose(Axis, Series); // @todo remove automatism
- return BrokenAxis;
- });
- _registerModule(_modules, 'Core/Axis/TreeGridAxis.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Axis/Tick.js'], _modules['Gantt/Tree.js'], _modules['Core/Axis/TreeGridTick.js'], _modules['Mixins/TreeSeries.js'], _modules['Core/Utilities.js']], function (Axis, Tick, Tree, TreeGridTick, mixinTreeSeries, U) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Authors: Jon Arild Nygard
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var getLevelOptions = mixinTreeSeries.getLevelOptions;
- var addEvent = U.addEvent,
- find = U.find,
- fireEvent = U.fireEvent,
- isNumber = U.isNumber,
- isObject = U.isObject,
- isString = U.isString,
- merge = U.merge,
- pick = U.pick,
- wrap = U.wrap;
- /**
- * @private
- */
- var TreeGridAxis;
- (function (TreeGridAxis) {
- /* *
- *
- * Interfaces
- *
- * */
- /* *
- *
- * Variables
- *
- * */
- var applied = false;
- /* *
- *
- * Functions
- *
- * */
- /**
- * @private
- */
- function compose(AxisClass) {
- if (!applied) {
- wrap(AxisClass.prototype, 'generateTick', wrapGenerateTick);
- wrap(AxisClass.prototype, 'getMaxLabelDimensions', wrapGetMaxLabelDimensions);
- wrap(AxisClass.prototype, 'init', wrapInit);
- wrap(AxisClass.prototype, 'setTickInterval', wrapSetTickInterval);
- TreeGridTick.compose(Tick);
- applied = true;
- }
- }
- TreeGridAxis.compose = compose;
- /**
- * @private
- */
- function getBreakFromNode(node, max) {
- var from = node.collapseStart || 0,
- to = node.collapseEnd || 0;
- // In broken-axis, the axis.max is minimized until it is not within a
- // break. Therefore, if break.to is larger than axis.max, the axis.to
- // should not add the 0.5 axis.tickMarkOffset, to avoid adding a break
- // larger than axis.max.
- // TODO consider simplifying broken-axis and this might solve itself
- if (to >= max) {
- from -= 0.5;
- }
- return {
- from: from,
- to: to,
- showPoints: false
- };
- }
- /**
- * Creates a tree structure of the data, and the treegrid. Calculates
- * categories, and y-values of points based on the tree.
- *
- * @private
- * @function getTreeGridFromData
- *
- * @param {Array<Highcharts.GanttPointOptions>} data
- * All the data points to display in the axis.
- *
- * @param {boolean} uniqueNames
- * Wether or not the data node with the same name should share grid cell. If
- * true they do share cell. False by default.
- *
- * @param {number} numberOfSeries
- *
- * @return {object}
- * Returns an object containing categories, mapOfIdToNode,
- * mapOfPosToGridNode, and tree.
- *
- * @todo There should be only one point per line.
- * @todo It should be optional to have one category per point, or merge
- * cells
- * @todo Add unit-tests.
- */
- function getTreeGridFromData(data, uniqueNames, numberOfSeries) {
- var categories = [],
- collapsedNodes = [],
- mapOfIdToNode = {},
- mapOfPosToGridNode = {},
- posIterator = -1,
- uniqueNamesEnabled = typeof uniqueNames === 'boolean' ? uniqueNames : false,
- tree;
- // Build the tree from the series data.
- var treeParams = {
- // After the children has been created.
- after: function (node) {
- var gridNode = mapOfPosToGridNode[node.pos],
- height = 0,
- descendants = 0;
- gridNode.children.forEach(function (child) {
- descendants += (child.descendants || 0) + 1;
- height = Math.max((child.height || 0) + 1, height);
- });
- gridNode.descendants = descendants;
- gridNode.height = height;
- if (gridNode.collapsed) {
- collapsedNodes.push(gridNode);
- }
- },
- // Before the children has been created.
- before: function (node) {
- var data = isObject(node.data,
- true) ? node.data : {},
- name = isString(data.name) ? data.name : '',
- parentNode = mapOfIdToNode[node.parent],
- parentGridNode = (isObject(parentNode,
- true) ?
- mapOfPosToGridNode[parentNode.pos] :
- null),
- hasSameName = function (x) {
- return x.name === name;
- }, gridNode, pos;
- // If not unique names, look for sibling node with the same name
- if (uniqueNamesEnabled &&
- isObject(parentGridNode, true) &&
- !!(gridNode = find(parentGridNode.children, hasSameName))) {
- // If there is a gridNode with the same name, reuse position
- pos = gridNode.pos;
- // Add data node to list of nodes in the grid node.
- gridNode.nodes.push(node);
- }
- else {
- // If it is a new grid node, increment position.
- pos = posIterator++;
- }
- // Add new grid node to map.
- if (!mapOfPosToGridNode[pos]) {
- mapOfPosToGridNode[pos] = gridNode = {
- depth: parentGridNode ? parentGridNode.depth + 1 : 0,
- name: name,
- nodes: [node],
- children: [],
- pos: pos
- };
- // If not root, then add name to categories.
- if (pos !== -1) {
- categories.push(name);
- }
- // Add name to list of children.
- if (isObject(parentGridNode, true)) {
- parentGridNode.children.push(gridNode);
- }
- }
- // Add data node to map
- if (isString(node.id)) {
- mapOfIdToNode[node.id] = node;
- }
- // If one of the points are collapsed, then start the grid node
- // in collapsed state.
- if (gridNode &&
- data.collapsed === true) {
- gridNode.collapsed = true;
- }
- // Assign pos to data node
- node.pos = pos;
- }
- };
- var updateYValuesAndTickPos = function (map,
- numberOfSeries) {
- var setValues = function (gridNode,
- start,
- result) {
- var nodes = gridNode.nodes,
- end = start + (start === -1 ? 0 : numberOfSeries - 1),
- diff = (end - start) / 2,
- padding = 0.5,
- pos = start + diff;
- nodes.forEach(function (node) {
- var data = node.data;
- if (isObject(data, true)) {
- // Update point
- data.y = start + (data.seriesIndex || 0);
- // Remove the property once used
- delete data.seriesIndex;
- }
- node.pos = pos;
- });
- result[pos] = gridNode;
- gridNode.pos = pos;
- gridNode.tickmarkOffset = diff + padding;
- gridNode.collapseStart = end + padding;
- gridNode.children.forEach(function (child) {
- setValues(child, end + 1, result);
- end = (child.collapseEnd || 0) - padding;
- });
- // Set collapseEnd to the end of the last child node.
- gridNode.collapseEnd = end + padding;
- return result;
- };
- return setValues(map['-1'], -1, {});
- };
- // Create tree from data
- tree = Tree.getTree(data, treeParams);
- // Update y values of data, and set calculate tick positions.
- mapOfPosToGridNode = updateYValuesAndTickPos(mapOfPosToGridNode, numberOfSeries);
- // Return the resulting data.
- return {
- categories: categories,
- mapOfIdToNode: mapOfIdToNode,
- mapOfPosToGridNode: mapOfPosToGridNode,
- collapsedNodes: collapsedNodes,
- tree: tree
- };
- }
- /**
- * Builds the tree of categories and calculates its positions.
- * @private
- * @param {object} e Event object
- * @param {object} e.target The chart instance which the event was fired on.
- * @param {object[]} e.target.axes The axes of the chart.
- */
- function onBeforeRender(e) {
- var chart = e.target,
- axes = chart.axes;
- axes.filter(function (axis) {
- return axis.options.type === 'treegrid';
- }).forEach(function (axis) {
- var options = axis.options || {},
- labelOptions = options.labels,
- uniqueNames = options.uniqueNames,
- numberOfSeries = 0,
- isDirty,
- data,
- treeGrid,
- max = options.max;
- // Check whether any of series is rendering for the first time,
- // visibility has changed, or its data is dirty,
- // and only then update. #10570, #10580
- // Also check if mapOfPosToGridNode exists. #10887
- isDirty = (!axis.treeGrid.mapOfPosToGridNode ||
- axis.series.some(function (series) {
- return !series.hasRendered ||
- series.isDirtyData ||
- series.isDirty;
- }));
- if (isDirty) {
- // Concatenate data from all series assigned to this axis.
- data = axis.series.reduce(function (arr, s) {
- if (s.visible) {
- // Push all data to array
- (s.options.data || []).forEach(function (data) {
- if (isObject(data, true)) {
- // Set series index on data. Removed again
- // after use.
- data.seriesIndex = numberOfSeries;
- arr.push(data);
- }
- });
- // Increment series index
- if (uniqueNames === true) {
- numberOfSeries++;
- }
- }
- return arr;
- }, []);
- // If max is higher than set data - add a
- // dummy data to render categories #10779
- if (max && data.length < max) {
- for (var i = data.length; i <= max; i++) {
- data.push({
- // Use the zero-width character
- // to avoid conflict with uniqueNames
- name: i + '\u200B'
- });
- }
- }
- // setScale is fired after all the series is initialized,
- // which is an ideal time to update the axis.categories.
- treeGrid = getTreeGridFromData(data, uniqueNames || false, (uniqueNames === true) ? numberOfSeries : 1);
- // Assign values to the axis.
- axis.categories = treeGrid.categories;
- axis.treeGrid.mapOfPosToGridNode = treeGrid.mapOfPosToGridNode;
- axis.hasNames = true;
- axis.treeGrid.tree = treeGrid.tree;
- // Update yData now that we have calculated the y values
- axis.series.forEach(function (series) {
- var data = (series.options.data || []).map(function (d) {
- return isObject(d,
- true) ? merge(d) : d;
- });
- // Avoid destroying points when series is not visible
- if (series.visible) {
- series.setData(data, false);
- }
- });
- // Calculate the label options for each level in the tree.
- axis.treeGrid.mapOptionsToLevel =
- getLevelOptions({
- defaults: labelOptions,
- from: 1,
- levels: labelOptions && labelOptions.levels,
- to: axis.treeGrid.tree && axis.treeGrid.tree.height
- });
- // Setting initial collapsed nodes
- if (e.type === 'beforeRender') {
- axis.treeGrid.collapsedNodes = treeGrid.collapsedNodes;
- }
- }
- });
- }
- /**
- * Generates a tick for initial positioning.
- *
- * @private
- * @function Highcharts.GridAxis#generateTick
- *
- * @param {Function} proceed
- * The original generateTick function.
- *
- * @param {number} pos
- * The tick position in axis values.
- */
- function wrapGenerateTick(proceed, pos) {
- var axis = this,
- mapOptionsToLevel = axis.treeGrid.mapOptionsToLevel || {},
- isTreeGrid = axis.options.type === 'treegrid',
- ticks = axis.ticks;
- var tick = ticks[pos],
- levelOptions,
- options,
- gridNode;
- if (isTreeGrid &&
- axis.treeGrid.mapOfPosToGridNode) {
- gridNode = axis.treeGrid.mapOfPosToGridNode[pos];
- levelOptions = mapOptionsToLevel[gridNode.depth];
- if (levelOptions) {
- options = {
- labels: levelOptions
- };
- }
- if (!tick) {
- ticks[pos] = tick =
- new Tick(axis, pos, void 0, void 0, {
- category: gridNode.name,
- tickmarkOffset: gridNode.tickmarkOffset,
- options: options
- });
- }
- else {
- // update labels depending on tick interval
- tick.parameters.category = gridNode.name;
- tick.options = options;
- tick.addLabel();
- }
- }
- else {
- proceed.apply(axis, Array.prototype.slice.call(arguments, 1));
- }
- }
- /**
- * Override to add indentation to axis.maxLabelDimensions.
- *
- * @private
- * @function Highcharts.GridAxis#getMaxLabelDimensions
- *
- * @param {Function} proceed
- * The original function
- */
- function wrapGetMaxLabelDimensions(proceed) {
- var axis = this,
- options = axis.options,
- labelOptions = options && options.labels,
- indentation = (labelOptions && isNumber(labelOptions.indentation) ?
- labelOptions.indentation :
- 0),
- retVal = proceed.apply(axis,
- Array.prototype.slice.call(arguments, 1)),
- isTreeGrid = axis.options.type === 'treegrid';
- var treeDepth;
- if (isTreeGrid && axis.treeGrid.mapOfPosToGridNode) {
- treeDepth = axis.treeGrid.mapOfPosToGridNode[-1].height || 0;
- retVal.width += indentation * (treeDepth - 1);
- }
- return retVal;
- }
- /**
- * @private
- */
- function wrapInit(proceed, chart, userOptions) {
- var axis = this,
- isTreeGrid = userOptions.type === 'treegrid';
- if (!axis.treeGrid) {
- axis.treeGrid = new Additions(axis);
- }
- // Set default and forced options for TreeGrid
- if (isTreeGrid) {
- // Add event for updating the categories of a treegrid.
- // NOTE Preferably these events should be set on the axis.
- addEvent(chart, 'beforeRender', onBeforeRender);
- addEvent(chart, 'beforeRedraw', onBeforeRender);
- // Add new collapsed nodes on addseries
- addEvent(chart, 'addSeries', function (e) {
- if (e.options.data) {
- var treeGrid = getTreeGridFromData(e.options.data,
- userOptions.uniqueNames || false, 1);
- axis.treeGrid.collapsedNodes = (axis.treeGrid.collapsedNodes || []).concat(treeGrid.collapsedNodes);
- }
- });
- // Collapse all nodes in axis.treegrid.collapsednodes
- // where collapsed equals true.
- addEvent(axis, 'foundExtremes', function () {
- if (axis.treeGrid.collapsedNodes) {
- axis.treeGrid.collapsedNodes.forEach(function (node) {
- var breaks = axis.treeGrid.collapse(node);
- if (axis.brokenAxis) {
- axis.brokenAxis.setBreaks(breaks, false);
- // remove the node from the axis collapsedNodes
- if (axis.treeGrid.collapsedNodes) {
- axis.treeGrid.collapsedNodes = axis.treeGrid.collapsedNodes.filter(function (n) {
- return node.collapseStart !== n.collapseStart ||
- node.collapseEnd !== n.collapseEnd;
- });
- }
- }
- });
- }
- });
- // If staticScale is not defined on the yAxis
- // and chart height is set, set axis.isDirty
- // to ensure collapsing works (#12012)
- addEvent(axis, 'afterBreaks', function () {
- var _a;
- if (axis.coll === 'yAxis' && !axis.staticScale && ((_a = axis.chart.options.chart) === null || _a === void 0 ? void 0 : _a.height)) {
- axis.isDirty = true;
- }
- });
- userOptions = merge({
- // Default options
- grid: {
- enabled: true
- },
- // TODO: add support for align in treegrid.
- labels: {
- align: 'left',
- /**
- * Set options on specific levels in a tree grid axis. Takes
- * precedence over labels options.
- *
- * @sample {gantt} gantt/treegrid-axis/labels-levels
- * Levels on TreeGrid Labels
- *
- * @type {Array<*>}
- * @product gantt
- * @apioption yAxis.labels.levels
- *
- * @private
- */
- levels: [{
- /**
- * Specify the level which the options within this object
- * applies to.
- *
- * @type {number}
- * @product gantt
- * @apioption yAxis.labels.levels.level
- *
- * @private
- */
- level: void 0
- }, {
- level: 1,
- /**
- * @type {Highcharts.CSSObject}
- * @product gantt
- * @apioption yAxis.labels.levels.style
- *
- * @private
- */
- style: {
- /** @ignore-option */
- fontWeight: 'bold'
- }
- }],
- /**
- * The symbol for the collapse and expand icon in a
- * treegrid.
- *
- * @product gantt
- * @optionparent yAxis.labels.symbol
- *
- * @private
- */
- symbol: {
- /**
- * The symbol type. Points to a definition function in
- * the `Highcharts.Renderer.symbols` collection.
- *
- * @type {Highcharts.SymbolKeyValue}
- *
- * @private
- */
- type: 'triangle',
- x: -5,
- y: -5,
- height: 10,
- width: 10,
- padding: 5
- }
- },
- uniqueNames: false
- }, userOptions, {
- // Forced options
- reversed: true,
- // grid.columns is not supported in treegrid
- grid: {
- columns: void 0
- }
- });
- }
- // Now apply the original function with the original arguments,
- // which are sliced off this function's arguments
- proceed.apply(axis, [chart, userOptions]);
- if (isTreeGrid) {
- axis.hasNames = true;
- axis.options.showLastLabel = true;
- }
- }
- /**
- * Set the tick positions, tickInterval, axis min and max.
- *
- * @private
- * @function Highcharts.GridAxis#setTickInterval
- *
- * @param {Function} proceed
- * The original setTickInterval function.
- */
- function wrapSetTickInterval(proceed) {
- var axis = this,
- options = axis.options,
- isTreeGrid = options.type === 'treegrid';
- if (isTreeGrid) {
- axis.min = pick(axis.userMin, options.min, axis.dataMin);
- axis.max = pick(axis.userMax, options.max, axis.dataMax);
- fireEvent(axis, 'foundExtremes');
- // setAxisTranslation modifies the min and max according to
- // axis breaks.
- axis.setAxisTranslation(true);
- axis.tickmarkOffset = 0.5;
- axis.tickInterval = 1;
- axis.tickPositions = axis.treeGrid.mapOfPosToGridNode ?
- axis.treeGrid.getTickPositions() :
- [];
- }
- else {
- proceed.apply(axis, Array.prototype.slice.call(arguments, 1));
- }
- }
- /* *
- *
- * Classes
- *
- * */
- /**
- * @private
- * @class
- */
- var Additions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- /**
- * @private
- */
- function Additions(axis) {
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Calculates the new axis breaks to collapse a node.
- *
- * @private
- *
- * @param {Highcharts.Axis} axis
- * The axis to check against.
- *
- * @param {Highcharts.GridNode} node
- * The node to collapse.
- *
- * @param {number} pos
- * The tick position to collapse.
- *
- * @return {Array<object>}
- * Returns an array of the new breaks for the axis.
- */
- Additions.prototype.collapse = function (node) {
- var axis = this.axis,
- breaks = (axis.options.breaks || []),
- obj = getBreakFromNode(node,
- axis.max);
- breaks.push(obj);
- return breaks;
- };
- /**
- * Calculates the new axis breaks to expand a node.
- *
- * @private
- *
- * @param {Highcharts.Axis} axis
- * The axis to check against.
- *
- * @param {Highcharts.GridNode} node
- * The node to expand.
- *
- * @param {number} pos
- * The tick position to expand.
- *
- * @return {Array<object>}
- * Returns an array of the new breaks for the axis.
- */
- Additions.prototype.expand = function (node) {
- var axis = this.axis,
- breaks = (axis.options.breaks || []),
- obj = getBreakFromNode(node,
- axis.max);
- // Remove the break from the axis breaks array.
- return breaks.reduce(function (arr, b) {
- if (b.to !== obj.to || b.from !== obj.from) {
- arr.push(b);
- }
- return arr;
- }, []);
- };
- /**
- * Creates a list of positions for the ticks on the axis. Filters out
- * positions that are outside min and max, or is inside an axis break.
- *
- * @private
- *
- * @return {Array<number>}
- * List of positions.
- */
- Additions.prototype.getTickPositions = function () {
- var axis = this.axis;
- return Object.keys(axis.treeGrid.mapOfPosToGridNode || {}).reduce(function (arr, key) {
- var pos = +key;
- if (axis.min <= pos &&
- axis.max >= pos &&
- !(axis.brokenAxis && axis.brokenAxis.isInAnyBreak(pos))) {
- arr.push(pos);
- }
- return arr;
- }, []);
- };
- /**
- * Check if a node is collapsed.
- *
- * @private
- *
- * @param {Highcharts.Axis} axis
- * The axis to check against.
- *
- * @param {object} node
- * The node to check if is collapsed.
- *
- * @param {number} pos
- * The tick position to collapse.
- *
- * @return {boolean}
- * Returns true if collapsed, false if expanded.
- */
- Additions.prototype.isCollapsed = function (node) {
- var axis = this.axis,
- breaks = (axis.options.breaks || []),
- obj = getBreakFromNode(node,
- axis.max);
- return breaks.some(function (b) {
- return b.from === obj.from && b.to === obj.to;
- });
- };
- /**
- * Calculates the new axis breaks after toggling the collapse/expand
- * state of a node. If it is collapsed it will be expanded, and if it is
- * exapended it will be collapsed.
- *
- * @private
- *
- * @param {Highcharts.Axis} axis
- * The axis to check against.
- *
- * @param {Highcharts.GridNode} node
- * The node to toggle.
- *
- * @return {Array<object>}
- * Returns an array of the new breaks for the axis.
- */
- Additions.prototype.toggleCollapse = function (node) {
- return (this.isCollapsed(node) ?
- this.expand(node) :
- this.collapse(node));
- };
- return Additions;
- }());
- TreeGridAxis.Additions = Additions;
- })(TreeGridAxis || (TreeGridAxis = {}));
- // Make utility functions available for testing.
- Axis.prototype.utils = {
- getNode: Tree.getNode
- };
- TreeGridAxis.compose(Axis);
- return TreeGridAxis;
- });
- _registerModule(_modules, 'Extensions/CurrentDateIndication.js', [_modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Utilities.js'], _modules['Core/Axis/PlotLineOrBand.js']], function (H, O, U, PlotLineOrBand) {
- /* *
- *
- * (c) 2016-2020 Highsoft AS
- *
- * Author: Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var dateFormat = O.dateFormat;
- var addEvent = U.addEvent,
- merge = U.merge,
- wrap = U.wrap;
- var Axis = H.Axis;
- var defaultConfig = {
- /**
- * Show an indicator on the axis for the current date and time. Can be a
- * boolean or a configuration object similar to
- * [xAxis.plotLines](#xAxis.plotLines).
- *
- * @sample gantt/current-date-indicator/demo
- * Current date indicator enabled
- * @sample gantt/current-date-indicator/object-config
- * Current date indicator with custom options
- *
- * @declare Highcharts.AxisCurrentDateIndicatorOptions
- * @type {boolean|*}
- * @default true
- * @extends xAxis.plotLines
- * @excluding value
- * @product gantt
- * @apioption xAxis.currentDateIndicator
- */
- currentDateIndicator: true,
- color: '#ccd6eb',
- width: 2,
- /**
- * @declare Highcharts.AxisCurrentDateIndicatorLabelOptions
- */
- label: {
- /**
- * Format of the label. This options is passed as the fist argument to
- * [dateFormat](/class-reference/Highcharts#dateFormat) function.
- *
- * @type {string}
- * @default '%a, %b %d %Y, %H:%M'
- * @product gantt
- * @apioption xAxis.currentDateIndicator.label.format
- */
- format: '%a, %b %d %Y, %H:%M',
- formatter: function (value, format) {
- return dateFormat(format, value);
- },
- rotation: 0,
- /**
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @internal */
- fontSize: '10px'
- }
- }
- };
- /* eslint-disable no-invalid-this */
- addEvent(Axis, 'afterSetOptions', function () {
- var options = this.options,
- cdiOptions = options.currentDateIndicator;
- if (cdiOptions) {
- cdiOptions = typeof cdiOptions === 'object' ?
- merge(defaultConfig, cdiOptions) : merge(defaultConfig);
- cdiOptions.value = new Date();
- if (!options.plotLines) {
- options.plotLines = [];
- }
- options.plotLines.push(cdiOptions);
- }
- });
- addEvent(PlotLineOrBand, 'render', function () {
- // If the label already exists, update its text
- if (this.label) {
- this.label.attr({
- text: this.getLabelText(this.options.label)
- });
- }
- });
- wrap(PlotLineOrBand.prototype, 'getLabelText', function (defaultMethod, defaultLabelOptions) {
- var options = this.options;
- if (options.currentDateIndicator && options.label &&
- typeof options.label.formatter === 'function') {
- options.value = new Date();
- return options.label.formatter
- .call(this, options.value, options.label.format);
- }
- return defaultMethod.call(this, defaultLabelOptions);
- });
- });
- _registerModule(_modules, 'Extensions/StaticScale.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2016-2020 Torstein Honsi, Lars Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- defined = U.defined,
- isNumber = U.isNumber,
- pick = U.pick;
- var Chart = H.Chart;
- /* eslint-disable no-invalid-this */
- /**
- * For vertical axes only. Setting the static scale ensures that each tick unit
- * is translated into a fixed pixel height. For example, setting the static
- * scale to 24 results in each Y axis category taking up 24 pixels, and the
- * height of the chart adjusts. Adding or removing items will make the chart
- * resize.
- *
- * @sample gantt/xrange-series/demo/
- * X-range series with static scale
- *
- * @type {number}
- * @default 50
- * @since 6.2.0
- * @product gantt
- * @apioption yAxis.staticScale
- */
- addEvent(H.Axis, 'afterSetOptions', function () {
- var chartOptions = this.chart.options && this.chart.options.chart;
- if (!this.horiz &&
- isNumber(this.options.staticScale) &&
- (!chartOptions.height ||
- (chartOptions.scrollablePlotArea &&
- chartOptions.scrollablePlotArea.minHeight))) {
- this.staticScale = this.options.staticScale;
- }
- });
- Chart.prototype.adjustHeight = function () {
- if (this.redrawTrigger !== 'adjustHeight') {
- (this.axes || []).forEach(function (axis) {
- var chart = axis.chart,
- animate = !!chart.initiatedScale &&
- chart.options.animation,
- staticScale = axis.options.staticScale,
- height,
- diff;
- if (axis.staticScale && defined(axis.min)) {
- height = pick(axis.brokenAxis && axis.brokenAxis.unitLength, axis.max + axis.tickInterval - axis.min) * staticScale;
- // Minimum height is 1 x staticScale.
- height = Math.max(height, staticScale);
- diff = height - chart.plotHeight;
- if (Math.abs(diff) >= 1) {
- chart.plotHeight = height;
- chart.redrawTrigger = 'adjustHeight';
- chart.setSize(void 0, chart.chartHeight + diff, animate);
- }
- // Make sure clip rects have the right height before initial
- // animation.
- axis.series.forEach(function (series) {
- var clipRect = series.sharedClipKey &&
- chart[series.sharedClipKey];
- if (clipRect) {
- clipRect.attr({
- height: chart.plotHeight
- });
- }
- });
- }
- });
- this.initiatedScale = true;
- }
- this.redrawTrigger = null;
- };
- addEvent(Chart, 'render', Chart.prototype.adjustHeight);
- });
- _registerModule(_modules, 'Extensions/ArrowSymbols.js', [_modules['Core/Renderer/SVG/SVGRenderer.js']], function (SVGRenderer) {
- /* *
- *
- * (c) 2017 Highsoft AS
- * Authors: Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * Creates an arrow symbol. Like a triangle, except not filled.
- * ```
- * o
- * o
- * o
- * o
- * o
- * o
- * o
- * ```
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the arrow
- *
- * @param {number} y
- * y position of the arrow
- *
- * @param {number} w
- * width of the arrow
- *
- * @param {number} h
- * height of the arrow
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols.arrow = function (x, y, w, h) {
- return [
- ['M', x, y + h / 2],
- ['L', x + w, y],
- ['L', x, y + h / 2],
- ['L', x + w, y + h]
- ];
- };
- /**
- * Creates a half-width arrow symbol. Like a triangle, except not filled.
- * ```
- * o
- * o
- * o
- * o
- * o
- * ```
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the arrow
- *
- * @param {number} y
- * y position of the arrow
- *
- * @param {number} w
- * width of the arrow
- *
- * @param {number} h
- * height of the arrow
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols['arrow-half'] = function (x, y, w, h) {
- return SVGRenderer.prototype.symbols.arrow(x, y, w / 2, h);
- };
- /**
- * Creates a left-oriented triangle.
- * ```
- * o
- * ooooooo
- * ooooooooooooo
- * ooooooo
- * o
- * ```
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the triangle
- *
- * @param {number} y
- * y position of the triangle
- *
- * @param {number} w
- * width of the triangle
- *
- * @param {number} h
- * height of the triangle
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols['triangle-left'] = function (x, y, w, h) {
- return [
- ['M', x + w, y],
- ['L', x, y + h / 2],
- ['L', x + w, y + h],
- ['Z']
- ];
- };
- /**
- * Alias function for triangle-left.
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the arrow
- *
- * @param {number} y
- * y position of the arrow
- *
- * @param {number} w
- * width of the arrow
- *
- * @param {number} h
- * height of the arrow
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols['arrow-filled'] = SVGRenderer.prototype.symbols['triangle-left'];
- /**
- * Creates a half-width, left-oriented triangle.
- * ```
- * o
- * oooo
- * ooooooo
- * oooo
- * o
- * ```
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the triangle
- *
- * @param {number} y
- * y position of the triangle
- *
- * @param {number} w
- * width of the triangle
- *
- * @param {number} h
- * height of the triangle
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols['triangle-left-half'] = function (x, y, w, h) {
- return SVGRenderer.prototype.symbols['triangle-left'](x, y, w / 2, h);
- };
- /**
- * Alias function for triangle-left-half.
- *
- * @private
- * @function
- *
- * @param {number} x
- * x position of the arrow
- *
- * @param {number} y
- * y position of the arrow
- *
- * @param {number} w
- * width of the arrow
- *
- * @param {number} h
- * height of the arrow
- *
- * @return {Highcharts.SVGPathArray}
- * Path array
- */
- SVGRenderer.prototype.symbols['arrow-filled-half'] = SVGRenderer.prototype.symbols['triangle-left-half'];
- });
- _registerModule(_modules, 'Gantt/Connection.js', [_modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js']], function (H, O, Point, U) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Authors: Øystein Moseng, Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * The default pathfinder algorithm to use for a chart. It is possible to define
- * your own algorithms by adding them to the
- * `Highcharts.Pathfinder.prototype.algorithms`
- * object before the chart has been created.
- *
- * The default algorithms are as follows:
- *
- * `straight`: Draws a straight line between the connecting
- * points. Does not avoid other points when drawing.
- *
- * `simpleConnect`: Finds a path between the points using right angles
- * only. Takes only starting/ending points into
- * account, and will not avoid other points.
- *
- * `fastAvoid`: Finds a path between the points using right angles
- * only. Will attempt to avoid other points, but its
- * focus is performance over accuracy. Works well with
- * less dense datasets.
- *
- * @typedef {"fastAvoid"|"simpleConnect"|"straight"|string} Highcharts.PathfinderTypeValue
- */
- ''; // detach doclets above
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- defined = U.defined,
- error = U.error,
- extend = U.extend,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- splat = U.splat;
- var deg2rad = H.deg2rad,
- max = Math.max,
- min = Math.min;
- /*
- @todo:
- - Document how to write your own algorithms
- - Consider adding a Point.pathTo method that wraps creating a connection
- and rendering it
- */
- // Set default Pathfinder options
- extend(defaultOptions, {
- /**
- * The Pathfinder module allows you to define connections between any two
- * points, represented as lines - optionally with markers for the start
- * and/or end points. Multiple algorithms are available for calculating how
- * the connecting lines are drawn.
- *
- * Connector functionality requires Highcharts Gantt to be loaded. In Gantt
- * charts, the connectors are used to draw dependencies between tasks.
- *
- * @see [dependency](series.gantt.data.dependency)
- *
- * @sample gantt/pathfinder/demo
- * Pathfinder connections
- *
- * @declare Highcharts.ConnectorsOptions
- * @product gantt
- * @optionparent connectors
- */
- connectors: {
- /**
- * Enable connectors for this chart. Requires Highcharts Gantt.
- *
- * @type {boolean}
- * @default true
- * @since 6.2.0
- * @apioption connectors.enabled
- */
- /**
- * Set the default dash style for this chart's connecting lines.
- *
- * @type {string}
- * @default solid
- * @since 6.2.0
- * @apioption connectors.dashStyle
- */
- /**
- * Set the default color for this chart's Pathfinder connecting lines.
- * Defaults to the color of the point being connected.
- *
- * @type {Highcharts.ColorString}
- * @since 6.2.0
- * @apioption connectors.lineColor
- */
- /**
- * Set the default pathfinder margin to use, in pixels. Some Pathfinder
- * algorithms attempt to avoid obstacles, such as other points in the
- * chart. These algorithms use this margin to determine how close lines
- * can be to an obstacle. The default is to compute this automatically
- * from the size of the obstacles in the chart.
- *
- * To draw connecting lines close to existing points, set this to a low
- * number. For more space around existing points, set this number
- * higher.
- *
- * @sample gantt/pathfinder/algorithm-margin
- * Small algorithmMargin
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.algorithmMargin
- */
- /**
- * Set the default pathfinder algorithm to use for this chart. It is
- * possible to define your own algorithms by adding them to the
- * Highcharts.Pathfinder.prototype.algorithms object before the chart
- * has been created.
- *
- * The default algorithms are as follows:
- *
- * `straight`: Draws a straight line between the connecting
- * points. Does not avoid other points when drawing.
- *
- * `simpleConnect`: Finds a path between the points using right angles
- * only. Takes only starting/ending points into
- * account, and will not avoid other points.
- *
- * `fastAvoid`: Finds a path between the points using right angles
- * only. Will attempt to avoid other points, but its
- * focus is performance over accuracy. Works well with
- * less dense datasets.
- *
- * Default value: `straight` is used as default for most series types,
- * while `simpleConnect` is used as default for Gantt series, to show
- * dependencies between points.
- *
- * @sample gantt/pathfinder/demo
- * Different types used
- *
- * @type {Highcharts.PathfinderTypeValue}
- * @default undefined
- * @since 6.2.0
- */
- type: 'straight',
- /**
- * Set the default pixel width for this chart's Pathfinder connecting
- * lines.
- *
- * @since 6.2.0
- */
- lineWidth: 1,
- /**
- * Marker options for this chart's Pathfinder connectors. Note that
- * this option is overridden by the `startMarker` and `endMarker`
- * options.
- *
- * @declare Highcharts.ConnectorsMarkerOptions
- * @since 6.2.0
- */
- marker: {
- /**
- * Set the radius of the connector markers. The default is
- * automatically computed based on the algorithmMargin setting.
- *
- * Setting marker.width and marker.height will override this
- * setting.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.radius
- */
- /**
- * Set the width of the connector markers. If not supplied, this
- * is inferred from the marker radius.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.width
- */
- /**
- * Set the height of the connector markers. If not supplied, this
- * is inferred from the marker radius.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.height
- */
- /**
- * Set the color of the connector markers. By default this is the
- * same as the connector color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 6.2.0
- * @apioption connectors.marker.color
- */
- /**
- * Set the line/border color of the connector markers. By default
- * this is the same as the marker color.
- *
- * @type {Highcharts.ColorString}
- * @since 6.2.0
- * @apioption connectors.marker.lineColor
- */
- /**
- * Enable markers for the connectors.
- */
- enabled: false,
- /**
- * Horizontal alignment of the markers relative to the points.
- *
- * @type {Highcharts.AlignValue}
- */
- align: 'center',
- /**
- * Vertical alignment of the markers relative to the points.
- *
- * @type {Highcharts.VerticalAlignValue}
- */
- verticalAlign: 'middle',
- /**
- * Whether or not to draw the markers inside the points.
- */
- inside: false,
- /**
- * Set the line/border width of the pathfinder markers.
- */
- lineWidth: 1
- },
- /**
- * Marker options specific to the start markers for this chart's
- * Pathfinder connectors. Overrides the generic marker options.
- *
- * @declare Highcharts.ConnectorsStartMarkerOptions
- * @extends connectors.marker
- * @since 6.2.0
- */
- startMarker: {
- /**
- * Set the symbol of the connector start markers.
- */
- symbol: 'diamond'
- },
- /**
- * Marker options specific to the end markers for this chart's
- * Pathfinder connectors. Overrides the generic marker options.
- *
- * @declare Highcharts.ConnectorsEndMarkerOptions
- * @extends connectors.marker
- * @since 6.2.0
- */
- endMarker: {
- /**
- * Set the symbol of the connector end markers.
- */
- symbol: 'arrow-filled'
- }
- }
- });
- /**
- * Override Pathfinder connector options for a series. Requires Highcharts Gantt
- * to be loaded.
- *
- * @declare Highcharts.SeriesConnectorsOptionsObject
- * @extends connectors
- * @since 6.2.0
- * @excluding enabled, algorithmMargin
- * @product gantt
- * @apioption plotOptions.series.connectors
- */
- /**
- * Connect to a point. This option can be either a string, referring to the ID
- * of another point, or an object, or an array of either. If the option is an
- * array, each element defines a connection.
- *
- * @sample gantt/pathfinder/demo
- * Different connection types
- *
- * @declare Highcharts.XrangePointConnectorsOptionsObject
- * @type {string|Array<string|*>|*}
- * @extends plotOptions.series.connectors
- * @since 6.2.0
- * @excluding enabled
- * @product gantt
- * @requires highcharts-gantt
- * @apioption series.xrange.data.connect
- */
- /**
- * The ID of the point to connect to.
- *
- * @type {string}
- * @since 6.2.0
- * @product gantt
- * @apioption series.xrange.data.connect.to
- */
- /**
- * Get point bounding box using plotX/plotY and shapeArgs. If using
- * graphic.getBBox() directly, the bbox will be affected by animation.
- *
- * @private
- * @function
- *
- * @param {Highcharts.Point} point
- * The point to get BB of.
- *
- * @return {Highcharts.Dictionary<number>|null}
- * Result xMax, xMin, yMax, yMin.
- */
- function getPointBB(point) {
- var shapeArgs = point.shapeArgs,
- bb;
- // Prefer using shapeArgs (columns)
- if (shapeArgs) {
- return {
- xMin: shapeArgs.x,
- xMax: shapeArgs.x + shapeArgs.width,
- yMin: shapeArgs.y,
- yMax: shapeArgs.y + shapeArgs.height
- };
- }
- // Otherwise use plotX/plotY and bb
- bb = point.graphic && point.graphic.getBBox();
- return bb ? {
- xMin: point.plotX - bb.width / 2,
- xMax: point.plotX + bb.width / 2,
- yMin: point.plotY - bb.height / 2,
- yMax: point.plotY + bb.height / 2
- } : null;
- }
- /**
- * Calculate margin to place around obstacles for the pathfinder in pixels.
- * Returns a minimum of 1 pixel margin.
- *
- * @private
- * @function
- *
- * @param {Array<object>} obstacles
- * Obstacles to calculate margin from.
- *
- * @return {number}
- * The calculated margin in pixels. At least 1.
- */
- function calculateObstacleMargin(obstacles) {
- var len = obstacles.length,
- i = 0,
- j,
- obstacleDistance,
- distances = [],
- // Compute smallest distance between two rectangles
- distance = function (a,
- b,
- bbMargin) {
- // Count the distance even if we are slightly off
- var margin = pick(bbMargin, 10),
- yOverlap = a.yMax + margin > b.yMin - margin &&
- a.yMin - margin < b.yMax + margin,
- xOverlap = a.xMax + margin > b.xMin - margin &&
- a.xMin - margin < b.xMax + margin,
- xDistance = yOverlap ? (a.xMin > b.xMax ? a.xMin - b.xMax : b.xMin - a.xMax) : Infinity,
- yDistance = xOverlap ? (a.yMin > b.yMax ? a.yMin - b.yMax : b.yMin - a.yMax) : Infinity;
- // If the rectangles collide, try recomputing with smaller margin.
- // If they collide anyway, discard the obstacle.
- if (xOverlap && yOverlap) {
- return (margin ?
- distance(a, b, Math.floor(margin / 2)) :
- Infinity);
- }
- return min(xDistance, yDistance);
- };
- // Go over all obstacles and compare them to the others.
- for (; i < len; ++i) {
- // Compare to all obstacles ahead. We will already have compared this
- // obstacle to the ones before.
- for (j = i + 1; j < len; ++j) {
- obstacleDistance = distance(obstacles[i], obstacles[j]);
- // TODO: Magic number 80
- if (obstacleDistance < 80) { // Ignore large distances
- distances.push(obstacleDistance);
- }
- }
- }
- // Ensure we always have at least one value, even in very spaceous charts
- distances.push(80);
- return max(Math.floor(distances.sort(function (a, b) {
- return (a - b);
- })[
- // Discard first 10% of the relevant distances, and then grab
- // the smallest one.
- Math.floor(distances.length / 10)] / 2 - 1 // Divide the distance by 2 and subtract 1.
- ), 1 // 1 is the minimum margin
- );
- }
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Connection class. Used internally to represent a connection between two
- * points.
- *
- * @private
- * @class
- * @name Highcharts.Connection
- *
- * @param {Highcharts.Point} from
- * Connection runs from this Point.
- *
- * @param {Highcharts.Point} to
- * Connection runs to this Point.
- *
- * @param {Highcharts.ConnectorsOptions} [options]
- * Connection options.
- */
- var Connection = /** @class */ (function () {
- function Connection(from, to, options) {
- /* *
- *
- * Properties
- *
- * */
- this.chart = void 0;
- this.fromPoint = void 0;
- this.graphics = void 0;
- this.pathfinder = void 0;
- this.toPoint = void 0;
- this.init(from, to, options);
- }
- /**
- * Initialize the Connection object. Used as constructor only.
- *
- * @function Highcharts.Connection#init
- *
- * @param {Highcharts.Point} from
- * Connection runs from this Point.
- *
- * @param {Highcharts.Point} to
- * Connection runs to this Point.
- *
- * @param {Highcharts.ConnectorsOptions} [options]
- * Connection options.
- */
- Connection.prototype.init = function (from, to, options) {
- this.fromPoint = from;
- this.toPoint = to;
- this.options = options;
- this.chart = from.series.chart;
- this.pathfinder = this.chart.pathfinder;
- };
- /**
- * Add (or update) this connection's path on chart. Stores reference to the
- * created element on this.graphics.path.
- *
- * @function Highcharts.Connection#renderPath
- *
- * @param {Highcharts.SVGPathArray} path
- * Path to render, in array format. E.g. ['M', 0, 0, 'L', 10, 10]
- *
- * @param {Highcharts.SVGAttributes} [attribs]
- * SVG attributes for the path.
- *
- * @param {Partial<Highcharts.AnimationOptionsObject>} [animation]
- * Animation options for the rendering.
- */
- Connection.prototype.renderPath = function (path, attribs, animation) {
- var connection = this,
- chart = this.chart,
- styledMode = chart.styledMode,
- pathfinder = chart.pathfinder,
- animate = !chart.options.chart.forExport && animation !== false,
- pathGraphic = connection.graphics && connection.graphics.path,
- anim;
- // Add the SVG element of the pathfinder group if it doesn't exist
- if (!pathfinder.group) {
- pathfinder.group = chart.renderer.g()
- .addClass('highcharts-pathfinder-group')
- .attr({ zIndex: -1 })
- .add(chart.seriesGroup);
- }
- // Shift the group to compensate for plot area.
- // Note: Do this always (even when redrawing a path) to avoid issues
- // when updating chart in a way that changes plot metrics.
- pathfinder.group.translate(chart.plotLeft, chart.plotTop);
- // Create path if does not exist
- if (!(pathGraphic && pathGraphic.renderer)) {
- pathGraphic = chart.renderer.path()
- .add(pathfinder.group);
- if (!styledMode) {
- pathGraphic.attr({
- opacity: 0
- });
- }
- }
- // Set path attribs and animate to the new path
- pathGraphic.attr(attribs);
- anim = { d: path };
- if (!styledMode) {
- anim.opacity = 1;
- }
- pathGraphic[animate ? 'animate' : 'attr'](anim, animation);
- // Store reference on connection
- this.graphics = this.graphics || {};
- this.graphics.path = pathGraphic;
- };
- /**
- * Calculate and add marker graphics for connection to the chart. The
- * created/updated elements are stored on this.graphics.start and
- * this.graphics.end.
- *
- * @function Highcharts.Connection#addMarker
- *
- * @param {string} type
- * Marker type, either 'start' or 'end'.
- *
- * @param {Highcharts.ConnectorsMarkerOptions} options
- * All options for this marker. Not calculated or merged with other
- * options.
- *
- * @param {Highcharts.SVGPathArray} path
- * Connection path in array format. This is used to calculate the
- * rotation angle of the markers.
- */
- Connection.prototype.addMarker = function (type, options, path) {
- var connection = this,
- chart = connection.fromPoint.series.chart,
- pathfinder = chart.pathfinder,
- renderer = chart.renderer,
- point = (type === 'start' ?
- connection.fromPoint :
- connection.toPoint),
- anchor = point.getPathfinderAnchorPoint(options),
- markerVector,
- radians,
- rotation,
- box,
- width,
- height,
- pathVector,
- segment;
- if (!options.enabled) {
- return;
- }
- // Last vector before start/end of path, used to get angle
- if (type === 'start') {
- segment = path[1];
- }
- else { // 'end'
- segment = path[path.length - 2];
- }
- if (segment && segment[0] === 'M' || segment[0] === 'L') {
- pathVector = {
- x: segment[1],
- y: segment[2]
- };
- // Get angle between pathVector and anchor point and use it to
- // create marker position.
- radians = point.getRadiansToVector(pathVector, anchor);
- markerVector = point.getMarkerVector(radians, options.radius, anchor);
- // Rotation of marker is calculated from angle between pathVector
- // and markerVector.
- // (Note:
- // Used to recalculate radians between markerVector and pathVector,
- // but this should be the same as between pathVector and anchor.)
- rotation = -radians / deg2rad;
- if (options.width && options.height) {
- width = options.width;
- height = options.height;
- }
- else {
- width = height = options.radius * 2;
- }
- // Add graphics object if it does not exist
- connection.graphics = connection.graphics || {};
- box = {
- x: markerVector.x - (width / 2),
- y: markerVector.y - (height / 2),
- width: width,
- height: height,
- rotation: rotation,
- rotationOriginX: markerVector.x,
- rotationOriginY: markerVector.y
- };
- if (!connection.graphics[type]) {
- // Create new marker element
- connection.graphics[type] = renderer
- .symbol(options.symbol)
- .addClass('highcharts-point-connecting-path-' + type + '-marker')
- .attr(box)
- .add(pathfinder.group);
- if (!renderer.styledMode) {
- connection.graphics[type].attr({
- fill: options.color || connection.fromPoint.color,
- stroke: options.lineColor,
- 'stroke-width': options.lineWidth,
- opacity: 0
- })
- .animate({
- opacity: 1
- }, point.series.options.animation);
- }
- }
- else {
- connection.graphics[type].animate(box);
- }
- }
- };
- /**
- * Calculate and return connection path.
- * Note: Recalculates chart obstacles on demand if they aren't calculated.
- *
- * @function Highcharts.Connection#getPath
- *
- * @param {Highcharts.ConnectorsOptions} options
- * Connector options. Not calculated or merged with other options.
- *
- * @return {object|undefined}
- * Calculated SVG path data in array format.
- */
- Connection.prototype.getPath = function (options) {
- var pathfinder = this.pathfinder,
- chart = this.chart,
- algorithm = pathfinder.algorithms[options.type],
- chartObstacles = pathfinder.chartObstacles;
- if (typeof algorithm !== 'function') {
- error('"' + options.type + '" is not a Pathfinder algorithm.');
- return {
- path: [],
- obstacles: []
- };
- }
- // This function calculates obstacles on demand if they don't exist
- if (algorithm.requiresObstacles && !chartObstacles) {
- chartObstacles =
- pathfinder.chartObstacles =
- pathfinder.getChartObstacles(options);
- // If the algorithmMargin was computed, store the result in default
- // options.
- chart.options.connectors.algorithmMargin =
- options.algorithmMargin;
- // Cache some metrics too
- pathfinder.chartObstacleMetrics =
- pathfinder.getObstacleMetrics(chartObstacles);
- }
- // Get the SVG path
- return algorithm(
- // From
- this.fromPoint.getPathfinderAnchorPoint(options.startMarker),
- // To
- this.toPoint.getPathfinderAnchorPoint(options.endMarker), merge({
- chartObstacles: chartObstacles,
- lineObstacles: pathfinder.lineObstacles || [],
- obstacleMetrics: pathfinder.chartObstacleMetrics,
- hardBounds: {
- xMin: 0,
- xMax: chart.plotWidth,
- yMin: 0,
- yMax: chart.plotHeight
- },
- obstacleOptions: {
- margin: options.algorithmMargin
- },
- startDirectionX: pathfinder.getAlgorithmStartDirection(options.startMarker)
- }, options));
- };
- /**
- * (re)Calculate and (re)draw the connection.
- *
- * @function Highcharts.Connection#render
- */
- Connection.prototype.render = function () {
- var connection = this,
- fromPoint = connection.fromPoint,
- series = fromPoint.series,
- chart = series.chart,
- pathfinder = chart.pathfinder,
- pathResult,
- path,
- options = merge(chart.options.connectors,
- series.options.connectors,
- fromPoint.options.connectors,
- connection.options),
- attribs = {};
- // Set path attribs
- if (!chart.styledMode) {
- attribs.stroke = options.lineColor || fromPoint.color;
- attribs['stroke-width'] = options.lineWidth;
- if (options.dashStyle) {
- attribs.dashstyle = options.dashStyle;
- }
- }
- attribs['class'] = // eslint-disable-line dot-notation
- 'highcharts-point-connecting-path ' +
- 'highcharts-color-' + fromPoint.colorIndex;
- options = merge(attribs, options);
- // Set common marker options
- if (!defined(options.marker.radius)) {
- options.marker.radius = min(max(Math.ceil((options.algorithmMargin || 8) / 2) - 1, 1), 5);
- }
- // Get the path
- pathResult = connection.getPath(options);
- path = pathResult.path;
- // Always update obstacle storage with obstacles from this path.
- // We don't know if future calls will need this for their algorithm.
- if (pathResult.obstacles) {
- pathfinder.lineObstacles =
- pathfinder.lineObstacles || [];
- pathfinder.lineObstacles =
- pathfinder.lineObstacles.concat(pathResult.obstacles);
- }
- // Add the calculated path to the pathfinder group
- connection.renderPath(path, attribs, series.options.animation);
- // Render the markers
- connection.addMarker('start', merge(options.marker, options.startMarker), path);
- connection.addMarker('end', merge(options.marker, options.endMarker), path);
- };
- /**
- * Destroy connection by destroying the added graphics elements.
- *
- * @function Highcharts.Connection#destroy
- */
- Connection.prototype.destroy = function () {
- if (this.graphics) {
- objectEach(this.graphics, function (val) {
- val.destroy();
- });
- delete this.graphics;
- }
- };
- return Connection;
- }());
- // Add to Highcharts namespace
- H.Connection = Connection;
- // Add pathfinding capabilities to Points
- extend(Point.prototype, /** @lends Point.prototype */ {
- /**
- * Get coordinates of anchor point for pathfinder connection.
- *
- * @private
- * @function Highcharts.Point#getPathfinderAnchorPoint
- *
- * @param {Highcharts.ConnectorsMarkerOptions} markerOptions
- * Connection options for position on point.
- *
- * @return {Highcharts.PositionObject}
- * An object with x/y properties for the position. Coordinates are
- * in plot values, not relative to point.
- */
- getPathfinderAnchorPoint: function (markerOptions) {
- var bb = getPointBB(this),
- x,
- y;
- switch (markerOptions.align) { // eslint-disable-line default-case
- case 'right':
- x = 'xMax';
- break;
- case 'left':
- x = 'xMin';
- }
- switch (markerOptions.verticalAlign) { // eslint-disable-line default-case
- case 'top':
- y = 'yMin';
- break;
- case 'bottom':
- y = 'yMax';
- }
- return {
- x: x ? bb[x] : (bb.xMin + bb.xMax) / 2,
- y: y ? bb[y] : (bb.yMin + bb.yMax) / 2
- };
- },
- /**
- * Utility to get the angle from one point to another.
- *
- * @private
- * @function Highcharts.Point#getRadiansToVector
- *
- * @param {Highcharts.PositionObject} v1
- * The first vector, as an object with x/y properties.
- *
- * @param {Highcharts.PositionObject} v2
- * The second vector, as an object with x/y properties.
- *
- * @return {number}
- * The angle in degrees
- */
- getRadiansToVector: function (v1, v2) {
- var box;
- if (!defined(v2)) {
- box = getPointBB(this);
- if (box) {
- v2 = {
- x: (box.xMin + box.xMax) / 2,
- y: (box.yMin + box.yMax) / 2
- };
- }
- }
- return Math.atan2(v2.y - v1.y, v1.x - v2.x);
- },
- /**
- * Utility to get the position of the marker, based on the path angle and
- * the marker's radius.
- *
- * @private
- * @function Highcharts.Point#getMarkerVector
- *
- * @param {number} radians
- * The angle in radians from the point center to another vector.
- *
- * @param {number} markerRadius
- * The radius of the marker, to calculate the additional distance to
- * the center of the marker.
- *
- * @param {object} anchor
- * The anchor point of the path and marker as an object with x/y
- * properties.
- *
- * @return {object}
- * The marker vector as an object with x/y properties.
- */
- getMarkerVector: function (radians, markerRadius, anchor) {
- var twoPI = Math.PI * 2.0,
- theta = radians,
- bb = getPointBB(this),
- rectWidth = bb.xMax - bb.xMin,
- rectHeight = bb.yMax - bb.yMin,
- rAtan = Math.atan2(rectHeight,
- rectWidth),
- tanTheta = 1,
- leftOrRightRegion = false,
- rectHalfWidth = rectWidth / 2.0,
- rectHalfHeight = rectHeight / 2.0,
- rectHorizontalCenter = bb.xMin + rectHalfWidth,
- rectVerticalCenter = bb.yMin + rectHalfHeight,
- edgePoint = {
- x: rectHorizontalCenter,
- y: rectVerticalCenter
- },
- markerPoint = {},
- xFactor = 1,
- yFactor = 1;
- while (theta < -Math.PI) {
- theta += twoPI;
- }
- while (theta > Math.PI) {
- theta -= twoPI;
- }
- tanTheta = Math.tan(theta);
- if ((theta > -rAtan) && (theta <= rAtan)) {
- // Right side
- yFactor = -1;
- leftOrRightRegion = true;
- }
- else if (theta > rAtan && theta <= (Math.PI - rAtan)) {
- // Top side
- yFactor = -1;
- }
- else if (theta > (Math.PI - rAtan) || theta <= -(Math.PI - rAtan)) {
- // Left side
- xFactor = -1;
- leftOrRightRegion = true;
- }
- else {
- // Bottom side
- xFactor = -1;
- }
- // Correct the edgePoint according to the placement of the marker
- if (leftOrRightRegion) {
- edgePoint.x += xFactor * (rectHalfWidth);
- edgePoint.y += yFactor * (rectHalfWidth) * tanTheta;
- }
- else {
- edgePoint.x += xFactor * (rectHeight / (2.0 * tanTheta));
- edgePoint.y += yFactor * (rectHalfHeight);
- }
- if (anchor.x !== rectHorizontalCenter) {
- edgePoint.x = anchor.x;
- }
- if (anchor.y !== rectVerticalCenter) {
- edgePoint.y = anchor.y;
- }
- markerPoint.x = edgePoint.x + (markerRadius * Math.cos(theta));
- markerPoint.y = edgePoint.y - (markerRadius * Math.sin(theta));
- return markerPoint;
- }
- });
- /**
- * Warn if using legacy options. Copy the options over. Note that this will
- * still break if using the legacy options in chart.update, addSeries etc.
- * @private
- */
- function warnLegacy(chart) {
- if (chart.options.pathfinder ||
- chart.series.reduce(function (acc, series) {
- if (series.options) {
- merge(true, (series.options.connectors = series.options.connectors ||
- {}), series.options.pathfinder);
- }
- return acc || series.options && series.options.pathfinder;
- }, false)) {
- merge(true, (chart.options.connectors = chart.options.connectors || {}), chart.options.pathfinder);
- error('WARNING: Pathfinder options have been renamed. ' +
- 'Use "chart.connectors" or "series.connectors" instead.');
- }
- }
- return Connection;
- });
- _registerModule(_modules, 'Gantt/PathfinderAlgorithms.js', [_modules['Core/Utilities.js']], function (U) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Author: Øystein Moseng
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var extend = U.extend,
- pick = U.pick;
- var min = Math.min,
- max = Math.max,
- abs = Math.abs;
- /**
- * Get index of last obstacle before xMin. Employs a type of binary search, and
- * thus requires that obstacles are sorted by xMin value.
- *
- * @private
- * @function findLastObstacleBefore
- *
- * @param {Array<object>} obstacles
- * Array of obstacles to search in.
- *
- * @param {number} xMin
- * The xMin threshold.
- *
- * @param {number} [startIx]
- * Starting index to search from. Must be within array range.
- *
- * @return {number}
- * The index of the last obstacle element before xMin.
- */
- function findLastObstacleBefore(obstacles, xMin, startIx) {
- var left = startIx || 0, // left limit
- right = obstacles.length - 1, // right limit
- min = xMin - 0.0000001, // Make sure we include all obstacles at xMin
- cursor,
- cmp;
- while (left <= right) {
- cursor = (right + left) >> 1;
- cmp = min - obstacles[cursor].xMin;
- if (cmp > 0) {
- left = cursor + 1;
- }
- else if (cmp < 0) {
- right = cursor - 1;
- }
- else {
- return cursor;
- }
- }
- return left > 0 ? left - 1 : 0;
- }
- /**
- * Test if a point lays within an obstacle.
- *
- * @private
- * @function pointWithinObstacle
- *
- * @param {object} obstacle
- * Obstacle to test.
- *
- * @param {Highcharts.Point} point
- * Point with x/y props.
- *
- * @return {boolean}
- * Whether point is within the obstacle or not.
- */
- function pointWithinObstacle(obstacle, point) {
- return (point.x <= obstacle.xMax &&
- point.x >= obstacle.xMin &&
- point.y <= obstacle.yMax &&
- point.y >= obstacle.yMin);
- }
- /**
- * Find the index of an obstacle that wraps around a point.
- * Returns -1 if not found.
- *
- * @private
- * @function findObstacleFromPoint
- *
- * @param {Array<object>} obstacles
- * Obstacles to test.
- *
- * @param {Highcharts.Point} point
- * Point with x/y props.
- *
- * @return {number}
- * Ix of the obstacle in the array, or -1 if not found.
- */
- function findObstacleFromPoint(obstacles, point) {
- var i = findLastObstacleBefore(obstacles,
- point.x + 1) + 1;
- while (i--) {
- if (obstacles[i].xMax >= point.x &&
- // optimization using lazy evaluation
- pointWithinObstacle(obstacles[i], point)) {
- return i;
- }
- }
- return -1;
- }
- /**
- * Get SVG path array from array of line segments.
- *
- * @private
- * @function pathFromSegments
- *
- * @param {Array<object>} segments
- * The segments to build the path from.
- *
- * @return {Highcharts.SVGPathArray}
- * SVG path array as accepted by the SVG Renderer.
- */
- function pathFromSegments(segments) {
- var path = [];
- if (segments.length) {
- path.push(['M', segments[0].start.x, segments[0].start.y]);
- for (var i = 0; i < segments.length; ++i) {
- path.push(['L', segments[i].end.x, segments[i].end.y]);
- }
- }
- return path;
- }
- /**
- * Limits obstacle max/mins in all directions to bounds. Modifies input
- * obstacle.
- *
- * @private
- * @function limitObstacleToBounds
- *
- * @param {object} obstacle
- * Obstacle to limit.
- *
- * @param {object} bounds
- * Bounds to use as limit.
- *
- * @return {void}
- */
- function limitObstacleToBounds(obstacle, bounds) {
- obstacle.yMin = max(obstacle.yMin, bounds.yMin);
- obstacle.yMax = min(obstacle.yMax, bounds.yMax);
- obstacle.xMin = max(obstacle.xMin, bounds.xMin);
- obstacle.xMax = min(obstacle.xMax, bounds.xMax);
- }
- /**
- * Get an SVG path from a starting coordinate to an ending coordinate.
- * Draws a straight line.
- *
- * @function Highcharts.Pathfinder.algorithms.straight
- *
- * @param {Highcharts.PositionObject} start
- * Starting coordinate, object with x/y props.
- *
- * @param {Highcharts.PositionObject} end
- * Ending coordinate, object with x/y props.
- *
- * @return {object}
- * An object with the SVG path in Array form as accepted by the SVG
- * renderer, as well as an array of new obstacles making up this
- * path.
- */
- function straight(start, end) {
- return {
- path: [
- ['M', start.x, start.y],
- ['L', end.x, end.y]
- ],
- obstacles: [{ start: start, end: end }]
- };
- }
- /**
- * Find a path from a starting coordinate to an ending coordinate, using
- * right angles only, and taking only starting/ending obstacle into
- * consideration.
- *
- * @function Highcharts.Pathfinder.algorithms.simpleConnect
- *
- * @param {Highcharts.PositionObject} start
- * Starting coordinate, object with x/y props.
- *
- * @param {Highcharts.PositionObject} end
- * Ending coordinate, object with x/y props.
- *
- * @param {object} options
- * Options for the algorithm:
- * - chartObstacles: Array of chart obstacles to avoid
- * - startDirectionX: Optional. True if starting in the X direction.
- * If not provided, the algorithm starts in the direction that is
- * the furthest between start/end.
- *
- * @return {object}
- * An object with the SVG path in Array form as accepted by the SVG
- * renderer, as well as an array of new obstacles making up this
- * path.
- */
- var simpleConnect = extend(function (start,
- end,
- options) {
- var segments = [],
- endSegment,
- dir = pick(options.startDirectionX,
- abs(end.x - start.x) > abs(end.y - start.y)) ? 'x' : 'y',
- chartObstacles = options.chartObstacles,
- startObstacleIx = findObstacleFromPoint(chartObstacles,
- start),
- endObstacleIx = findObstacleFromPoint(chartObstacles,
- end),
- startObstacle,
- endObstacle,
- prevWaypoint,
- waypoint,
- waypoint2,
- useMax,
- endPoint;
- // eslint-disable-next-line valid-jsdoc
- /**
- * Return a clone of a point with a property set from a target object,
- * optionally with an offset
- * @private
- */
- function copyFromPoint(from, fromKey, to, toKey, offset) {
- var point = {
- x: from.x,
- y: from.y
- };
- point[fromKey] = to[toKey || fromKey] + (offset || 0);
- return point;
- }
- // eslint-disable-next-line valid-jsdoc
- /**
- * Return waypoint outside obstacle.
- * @private
- */
- function getMeOut(obstacle, point, direction) {
- var useMax = abs(point[direction] - obstacle[direction + 'Min']) >
- abs(point[direction] - obstacle[direction + 'Max']);
- return copyFromPoint(point, direction, obstacle, direction + (useMax ? 'Max' : 'Min'), useMax ? 1 : -1);
- }
- // Pull out end point
- if (endObstacleIx > -1) {
- endObstacle = chartObstacles[endObstacleIx];
- waypoint = getMeOut(endObstacle, end, dir);
- endSegment = {
- start: waypoint,
- end: end
- };
- endPoint = waypoint;
- }
- else {
- endPoint = end;
- }
- // If an obstacle envelops the start point, add a segment to get out,
- // and around it.
- if (startObstacleIx > -1) {
- startObstacle = chartObstacles[startObstacleIx];
- waypoint = getMeOut(startObstacle, start, dir);
- segments.push({
- start: start,
- end: waypoint
- });
- // If we are going back again, switch direction to get around start
- // obstacle.
- if (
- // Going towards max from start:
- waypoint[dir] >= start[dir] ===
- // Going towards min to end:
- waypoint[dir] >= endPoint[dir]) {
- dir = dir === 'y' ? 'x' : 'y';
- useMax = start[dir] < end[dir];
- segments.push({
- start: waypoint,
- end: copyFromPoint(waypoint, dir, startObstacle, dir + (useMax ? 'Max' : 'Min'), useMax ? 1 : -1)
- });
- // Switch direction again
- dir = dir === 'y' ? 'x' : 'y';
- }
- }
- // We are around the start obstacle. Go towards the end in one
- // direction.
- prevWaypoint = segments.length ?
- segments[segments.length - 1].end :
- start;
- waypoint = copyFromPoint(prevWaypoint, dir, endPoint);
- segments.push({
- start: prevWaypoint,
- end: waypoint
- });
- // Final run to end point in the other direction
- dir = dir === 'y' ? 'x' : 'y';
- waypoint2 = copyFromPoint(waypoint, dir, endPoint);
- segments.push({
- start: waypoint,
- end: waypoint2
- });
- // Finally add the endSegment
- segments.push(endSegment);
- return {
- path: pathFromSegments(segments),
- obstacles: segments
- };
- }, {
- requiresObstacles: true
- });
- /**
- * Find a path from a starting coordinate to an ending coordinate, taking
- * obstacles into consideration. Might not always find the optimal path,
- * but is fast, and usually good enough.
- *
- * @function Highcharts.Pathfinder.algorithms.fastAvoid
- *
- * @param {Highcharts.PositionObject} start
- * Starting coordinate, object with x/y props.
- *
- * @param {Highcharts.PositionObject} end
- * Ending coordinate, object with x/y props.
- *
- * @param {object} options
- * Options for the algorithm.
- * - chartObstacles: Array of chart obstacles to avoid
- * - lineObstacles: Array of line obstacles to jump over
- * - obstacleMetrics: Object with metrics of chartObstacles cached
- * - hardBounds: Hard boundaries to not cross
- * - obstacleOptions: Options for the obstacles, including margin
- * - startDirectionX: Optional. True if starting in the X direction.
- * If not provided, the algorithm starts in the
- * direction that is the furthest between
- * start/end.
- *
- * @return {object}
- * An object with the SVG path in Array form as accepted by the SVG
- * renderer, as well as an array of new obstacles making up this
- * path.
- */
- var fastAvoid = extend(function (start,
- end,
- options) {
- /*
- Algorithm rules/description
- - Find initial direction
- - Determine soft/hard max for each direction.
- - Move along initial direction until obstacle.
- - Change direction.
- - If hitting obstacle,
- first try to change length of previous line
- before changing direction again.
-
- Soft min/max x = start/destination x +/- widest obstacle + margin
- Soft min/max y = start/destination y +/- tallest obstacle + margin
-
- @todo:
- - Make retrospective,
- try changing prev segment to reduce
- corners
- - Fix logic for breaking out of end-points - not always picking
- the best direction currently
- - When going around the end obstacle we should not always go the
- shortest route,
- rather pick the one closer to the end point
- */
- var dirIsX = pick(options.startDirectionX,
- abs(end.x - start.x) > abs(end.y - start.y)),
- dir = dirIsX ? 'x' : 'y',
- segments,
- useMax,
- extractedEndPoint,
- endSegments = [],
- forceObstacleBreak = false, // Used in clearPathTo to keep track of
- // when to force break through an obstacle.
- // Boundaries to stay within. If beyond soft boundary, prefer to
- // change direction ASAP. If at hard max, always change immediately.
- metrics = options.obstacleMetrics,
- softMinX = min(start.x,
- end.x) - metrics.maxWidth - 10,
- softMaxX = max(start.x,
- end.x) + metrics.maxWidth + 10,
- softMinY = min(start.y,
- end.y) - metrics.maxHeight - 10,
- softMaxY = max(start.y,
- end.y) + metrics.maxHeight + 10,
- // Obstacles
- chartObstacles = options.chartObstacles,
- startObstacleIx = findLastObstacleBefore(chartObstacles,
- softMinX),
- endObstacleIx = findLastObstacleBefore(chartObstacles,
- softMaxX);
- // eslint-disable-next-line valid-jsdoc
- /**
- * How far can you go between two points before hitting an obstacle?
- * Does not work for diagonal lines (because it doesn't have to).
- * @private
- */
- function pivotPoint(fromPoint, toPoint, directionIsX) {
- var firstPoint,
- lastPoint,
- highestPoint,
- lowestPoint,
- i,
- searchDirection = fromPoint.x < toPoint.x ? 1 : -1;
- if (fromPoint.x < toPoint.x) {
- firstPoint = fromPoint;
- lastPoint = toPoint;
- }
- else {
- firstPoint = toPoint;
- lastPoint = fromPoint;
- }
- if (fromPoint.y < toPoint.y) {
- lowestPoint = fromPoint;
- highestPoint = toPoint;
- }
- else {
- lowestPoint = toPoint;
- highestPoint = fromPoint;
- }
- // Go through obstacle range in reverse if toPoint is before
- // fromPoint in the X-dimension.
- i = searchDirection < 0 ?
- // Searching backwards, start at last obstacle before last point
- min(findLastObstacleBefore(chartObstacles, lastPoint.x), chartObstacles.length - 1) :
- // Forwards. Since we're not sorted by xMax, we have to look
- // at all obstacles.
- 0;
- // Go through obstacles in this X range
- while (chartObstacles[i] && (searchDirection > 0 && chartObstacles[i].xMin <= lastPoint.x ||
- searchDirection < 0 && chartObstacles[i].xMax >= firstPoint.x)) {
- // If this obstacle is between from and to points in a straight
- // line, pivot at the intersection.
- if (chartObstacles[i].xMin <= lastPoint.x &&
- chartObstacles[i].xMax >= firstPoint.x &&
- chartObstacles[i].yMin <= highestPoint.y &&
- chartObstacles[i].yMax >= lowestPoint.y) {
- if (directionIsX) {
- return {
- y: fromPoint.y,
- x: fromPoint.x < toPoint.x ?
- chartObstacles[i].xMin - 1 :
- chartObstacles[i].xMax + 1,
- obstacle: chartObstacles[i]
- };
- }
- // else ...
- return {
- x: fromPoint.x,
- y: fromPoint.y < toPoint.y ?
- chartObstacles[i].yMin - 1 :
- chartObstacles[i].yMax + 1,
- obstacle: chartObstacles[i]
- };
- }
- i += searchDirection;
- }
- return toPoint;
- }
- /**
- * Decide in which direction to dodge or get out of an obstacle.
- * Considers desired direction, which way is shortest, soft and hard
- * bounds.
- *
- * (? Returns a string, either xMin, xMax, yMin or yMax.)
- *
- * @private
- * @function
- *
- * @param {object} obstacle
- * Obstacle to dodge/escape.
- *
- * @param {object} fromPoint
- * Point with x/y props that's dodging/escaping.
- *
- * @param {object} toPoint
- * Goal point.
- *
- * @param {boolean} dirIsX
- * Dodge in X dimension.
- *
- * @param {object} bounds
- * Hard and soft boundaries.
- *
- * @return {boolean}
- * Use max or not.
- */
- function getDodgeDirection(obstacle, fromPoint, toPoint, dirIsX, bounds) {
- var softBounds = bounds.soft, hardBounds = bounds.hard, dir = dirIsX ? 'x' : 'y', toPointMax = { x: fromPoint.x, y: fromPoint.y }, toPointMin = { x: fromPoint.x, y: fromPoint.y }, minPivot, maxPivot, maxOutOfSoftBounds = obstacle[dir + 'Max'] >=
- softBounds[dir + 'Max'], minOutOfSoftBounds = obstacle[dir + 'Min'] <=
- softBounds[dir + 'Min'], maxOutOfHardBounds = obstacle[dir + 'Max'] >=
- hardBounds[dir + 'Max'], minOutOfHardBounds = obstacle[dir + 'Min'] <=
- hardBounds[dir + 'Min'],
- // Find out if we should prefer one direction over the other if
- // we can choose freely
- minDistance = abs(obstacle[dir + 'Min'] - fromPoint[dir]), maxDistance = abs(obstacle[dir + 'Max'] - fromPoint[dir]),
- // If it's a small difference, pick the one leading towards dest
- // point. Otherwise pick the shortest distance
- useMax = abs(minDistance - maxDistance) < 10 ?
- fromPoint[dir] < toPoint[dir] :
- maxDistance < minDistance;
- // Check if we hit any obstacles trying to go around in either
- // direction.
- toPointMin[dir] = obstacle[dir + 'Min'];
- toPointMax[dir] = obstacle[dir + 'Max'];
- minPivot = pivotPoint(fromPoint, toPointMin, dirIsX)[dir] !==
- toPointMin[dir];
- maxPivot = pivotPoint(fromPoint, toPointMax, dirIsX)[dir] !==
- toPointMax[dir];
- useMax = minPivot ?
- (maxPivot ? useMax : true) :
- (maxPivot ? false : useMax);
- // useMax now contains our preferred choice, bounds not taken into
- // account. If both or neither direction is out of bounds we want to
- // use this.
- // Deal with soft bounds
- useMax = minOutOfSoftBounds ?
- (maxOutOfSoftBounds ? useMax : true) : // Out on min
- (maxOutOfSoftBounds ? false : useMax); // Not out on min
- // Deal with hard bounds
- useMax = minOutOfHardBounds ?
- (maxOutOfHardBounds ? useMax : true) : // Out on min
- (maxOutOfHardBounds ? false : useMax); // Not out on min
- return useMax;
- }
- // eslint-disable-next-line valid-jsdoc
- /**
- * Find a clear path between point.
- * @private
- */
- function clearPathTo(fromPoint, toPoint, dirIsX) {
- // Don't waste time if we've hit goal
- if (fromPoint.x === toPoint.x && fromPoint.y === toPoint.y) {
- return [];
- }
- var dir = dirIsX ? 'x' : 'y',
- pivot,
- segments,
- waypoint,
- waypointUseMax,
- envelopingObstacle,
- secondEnvelopingObstacle,
- envelopWaypoint,
- obstacleMargin = options.obstacleOptions.margin,
- bounds = {
- soft: {
- xMin: softMinX,
- xMax: softMaxX,
- yMin: softMinY,
- yMax: softMaxY
- },
- hard: options.hardBounds
- };
- // If fromPoint is inside an obstacle we have a problem. Break out
- // by just going to the outside of this obstacle. We prefer to go to
- // the nearest edge in the chosen direction.
- envelopingObstacle =
- findObstacleFromPoint(chartObstacles, fromPoint);
- if (envelopingObstacle > -1) {
- envelopingObstacle = chartObstacles[envelopingObstacle];
- waypointUseMax = getDodgeDirection(envelopingObstacle, fromPoint, toPoint, dirIsX, bounds);
- // Cut obstacle to hard bounds to make sure we stay within
- limitObstacleToBounds(envelopingObstacle, options.hardBounds);
- envelopWaypoint = dirIsX ? {
- y: fromPoint.y,
- x: envelopingObstacle[waypointUseMax ? 'xMax' : 'xMin'] +
- (waypointUseMax ? 1 : -1)
- } : {
- x: fromPoint.x,
- y: envelopingObstacle[waypointUseMax ? 'yMax' : 'yMin'] +
- (waypointUseMax ? 1 : -1)
- };
- // If we crashed into another obstacle doing this, we put the
- // waypoint between them instead
- secondEnvelopingObstacle = findObstacleFromPoint(chartObstacles, envelopWaypoint);
- if (secondEnvelopingObstacle > -1) {
- secondEnvelopingObstacle = chartObstacles[secondEnvelopingObstacle];
- // Cut obstacle to hard bounds
- limitObstacleToBounds(secondEnvelopingObstacle, options.hardBounds);
- // Modify waypoint to lay between obstacles
- envelopWaypoint[dir] = waypointUseMax ? max(envelopingObstacle[dir + 'Max'] - obstacleMargin + 1, (secondEnvelopingObstacle[dir + 'Min'] +
- envelopingObstacle[dir + 'Max']) / 2) :
- min((envelopingObstacle[dir + 'Min'] + obstacleMargin - 1), ((secondEnvelopingObstacle[dir + 'Max'] +
- envelopingObstacle[dir + 'Min']) / 2));
- // We are not going anywhere. If this happens for the first
- // time, do nothing. Otherwise, try to go to the extreme of
- // the obstacle pair in the current direction.
- if (fromPoint.x === envelopWaypoint.x &&
- fromPoint.y === envelopWaypoint.y) {
- if (forceObstacleBreak) {
- envelopWaypoint[dir] = waypointUseMax ?
- max(envelopingObstacle[dir + 'Max'], secondEnvelopingObstacle[dir + 'Max']) + 1 :
- min(envelopingObstacle[dir + 'Min'], secondEnvelopingObstacle[dir + 'Min']) - 1;
- }
- // Toggle on if off, and the opposite
- forceObstacleBreak = !forceObstacleBreak;
- }
- else {
- // This point is not identical to previous.
- // Clear break trigger.
- forceObstacleBreak = false;
- }
- }
- segments = [{
- start: fromPoint,
- end: envelopWaypoint
- }];
- }
- else { // If not enveloping, use standard pivot calculation
- pivot = pivotPoint(fromPoint, {
- x: dirIsX ? toPoint.x : fromPoint.x,
- y: dirIsX ? fromPoint.y : toPoint.y
- }, dirIsX);
- segments = [{
- start: fromPoint,
- end: {
- x: pivot.x,
- y: pivot.y
- }
- }];
- // Pivot before goal, use a waypoint to dodge obstacle
- if (pivot[dirIsX ? 'x' : 'y'] !== toPoint[dirIsX ? 'x' : 'y']) {
- // Find direction of waypoint
- waypointUseMax = getDodgeDirection(pivot.obstacle, pivot, toPoint, !dirIsX, bounds);
- // Cut waypoint to hard bounds
- limitObstacleToBounds(pivot.obstacle, options.hardBounds);
- waypoint = {
- x: dirIsX ?
- pivot.x :
- pivot.obstacle[waypointUseMax ? 'xMax' : 'xMin'] +
- (waypointUseMax ? 1 : -1),
- y: dirIsX ?
- pivot.obstacle[waypointUseMax ? 'yMax' : 'yMin'] +
- (waypointUseMax ? 1 : -1) :
- pivot.y
- };
- // We're changing direction here, store that to make sure we
- // also change direction when adding the last segment array
- // after handling waypoint.
- dirIsX = !dirIsX;
- segments = segments.concat(clearPathTo({
- x: pivot.x,
- y: pivot.y
- }, waypoint, dirIsX));
- }
- }
- // Get segments for the other direction too
- // Recursion is our friend
- segments = segments.concat(clearPathTo(segments[segments.length - 1].end, toPoint, !dirIsX));
- return segments;
- }
- // eslint-disable-next-line valid-jsdoc
- /**
- * Extract point to outside of obstacle in whichever direction is
- * closest. Returns new point outside obstacle.
- * @private
- */
- function extractFromObstacle(obstacle, point, goalPoint) {
- var dirIsX = min(obstacle.xMax - point.x,
- point.x - obstacle.xMin) <
- min(obstacle.yMax - point.y,
- point.y - obstacle.yMin),
- bounds = {
- soft: options.hardBounds,
- hard: options.hardBounds
- },
- useMax = getDodgeDirection(obstacle,
- point,
- goalPoint,
- dirIsX,
- bounds);
- return dirIsX ? {
- y: point.y,
- x: obstacle[useMax ? 'xMax' : 'xMin'] + (useMax ? 1 : -1)
- } : {
- x: point.x,
- y: obstacle[useMax ? 'yMax' : 'yMin'] + (useMax ? 1 : -1)
- };
- }
- // Cut the obstacle array to soft bounds for optimization in large
- // datasets.
- chartObstacles =
- chartObstacles.slice(startObstacleIx, endObstacleIx + 1);
- // If an obstacle envelops the end point, move it out of there and add
- // a little segment to where it was.
- if ((endObstacleIx = findObstacleFromPoint(chartObstacles, end)) > -1) {
- extractedEndPoint = extractFromObstacle(chartObstacles[endObstacleIx], end, start);
- endSegments.push({
- end: end,
- start: extractedEndPoint
- });
- end = extractedEndPoint;
- }
- // If it's still inside one or more obstacles, get out of there by
- // force-moving towards the start point.
- while ((endObstacleIx = findObstacleFromPoint(chartObstacles, end)) > -1) {
- useMax = end[dir] - start[dir] < 0;
- extractedEndPoint = {
- x: end.x,
- y: end.y
- };
- extractedEndPoint[dir] = chartObstacles[endObstacleIx][useMax ? dir + 'Max' : dir + 'Min'] + (useMax ? 1 : -1);
- endSegments.push({
- end: end,
- start: extractedEndPoint
- });
- end = extractedEndPoint;
- }
- // Find the path
- segments = clearPathTo(start, end, dirIsX);
- // Add the end-point segments
- segments = segments.concat(endSegments.reverse());
- return {
- path: pathFromSegments(segments),
- obstacles: segments
- };
- }, {
- requiresObstacles: true
- });
- // Define the available pathfinding algorithms.
- // Algorithms take up to 3 arguments: starting point, ending point, and an
- // options object.
- var algorithms = {
- fastAvoid: fastAvoid,
- straight: straight,
- simpleConnect: simpleConnect
- };
- return algorithms;
- });
- _registerModule(_modules, 'Gantt/Pathfinder.js', [_modules['Gantt/Connection.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js'], _modules['Gantt/PathfinderAlgorithms.js']], function (Connection, Chart, H, O, Point, U, pathfinderAlgorithms) {
- /* *
- *
- * (c) 2016 Highsoft AS
- * Authors: Øystein Moseng, Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- /**
- * The default pathfinder algorithm to use for a chart. It is possible to define
- * your own algorithms by adding them to the
- * `Highcharts.Pathfinder.prototype.algorithms`
- * object before the chart has been created.
- *
- * The default algorithms are as follows:
- *
- * `straight`: Draws a straight line between the connecting
- * points. Does not avoid other points when drawing.
- *
- * `simpleConnect`: Finds a path between the points using right angles
- * only. Takes only starting/ending points into
- * account, and will not avoid other points.
- *
- * `fastAvoid`: Finds a path between the points using right angles
- * only. Will attempt to avoid other points, but its
- * focus is performance over accuracy. Works well with
- * less dense datasets.
- *
- * @typedef {"fastAvoid"|"simpleConnect"|"straight"|string} Highcharts.PathfinderTypeValue
- */
- ''; // detach doclets above
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- defined = U.defined,
- error = U.error,
- extend = U.extend,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- splat = U.splat;
- var deg2rad = H.deg2rad,
- max = Math.max,
- min = Math.min;
- /*
- @todo:
- - Document how to write your own algorithms
- - Consider adding a Point.pathTo method that wraps creating a connection
- and rendering it
- */
- // Set default Pathfinder options
- extend(defaultOptions, {
- /**
- * The Pathfinder module allows you to define connections between any two
- * points, represented as lines - optionally with markers for the start
- * and/or end points. Multiple algorithms are available for calculating how
- * the connecting lines are drawn.
- *
- * Connector functionality requires Highcharts Gantt to be loaded. In Gantt
- * charts, the connectors are used to draw dependencies between tasks.
- *
- * @see [dependency](series.gantt.data.dependency)
- *
- * @sample gantt/pathfinder/demo
- * Pathfinder connections
- *
- * @declare Highcharts.ConnectorsOptions
- * @product gantt
- * @optionparent connectors
- */
- connectors: {
- /**
- * Enable connectors for this chart. Requires Highcharts Gantt.
- *
- * @type {boolean}
- * @default true
- * @since 6.2.0
- * @apioption connectors.enabled
- */
- /**
- * Set the default dash style for this chart's connecting lines.
- *
- * @type {string}
- * @default solid
- * @since 6.2.0
- * @apioption connectors.dashStyle
- */
- /**
- * Set the default color for this chart's Pathfinder connecting lines.
- * Defaults to the color of the point being connected.
- *
- * @type {Highcharts.ColorString}
- * @since 6.2.0
- * @apioption connectors.lineColor
- */
- /**
- * Set the default pathfinder margin to use, in pixels. Some Pathfinder
- * algorithms attempt to avoid obstacles, such as other points in the
- * chart. These algorithms use this margin to determine how close lines
- * can be to an obstacle. The default is to compute this automatically
- * from the size of the obstacles in the chart.
- *
- * To draw connecting lines close to existing points, set this to a low
- * number. For more space around existing points, set this number
- * higher.
- *
- * @sample gantt/pathfinder/algorithm-margin
- * Small algorithmMargin
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.algorithmMargin
- */
- /**
- * Set the default pathfinder algorithm to use for this chart. It is
- * possible to define your own algorithms by adding them to the
- * Highcharts.Pathfinder.prototype.algorithms object before the chart
- * has been created.
- *
- * The default algorithms are as follows:
- *
- * `straight`: Draws a straight line between the connecting
- * points. Does not avoid other points when drawing.
- *
- * `simpleConnect`: Finds a path between the points using right angles
- * only. Takes only starting/ending points into
- * account, and will not avoid other points.
- *
- * `fastAvoid`: Finds a path between the points using right angles
- * only. Will attempt to avoid other points, but its
- * focus is performance over accuracy. Works well with
- * less dense datasets.
- *
- * Default value: `straight` is used as default for most series types,
- * while `simpleConnect` is used as default for Gantt series, to show
- * dependencies between points.
- *
- * @sample gantt/pathfinder/demo
- * Different types used
- *
- * @type {Highcharts.PathfinderTypeValue}
- * @default undefined
- * @since 6.2.0
- */
- type: 'straight',
- /**
- * Set the default pixel width for this chart's Pathfinder connecting
- * lines.
- *
- * @since 6.2.0
- */
- lineWidth: 1,
- /**
- * Marker options for this chart's Pathfinder connectors. Note that
- * this option is overridden by the `startMarker` and `endMarker`
- * options.
- *
- * @declare Highcharts.ConnectorsMarkerOptions
- * @since 6.2.0
- */
- marker: {
- /**
- * Set the radius of the connector markers. The default is
- * automatically computed based on the algorithmMargin setting.
- *
- * Setting marker.width and marker.height will override this
- * setting.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.radius
- */
- /**
- * Set the width of the connector markers. If not supplied, this
- * is inferred from the marker radius.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.width
- */
- /**
- * Set the height of the connector markers. If not supplied, this
- * is inferred from the marker radius.
- *
- * @type {number}
- * @since 6.2.0
- * @apioption connectors.marker.height
- */
- /**
- * Set the color of the connector markers. By default this is the
- * same as the connector color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @since 6.2.0
- * @apioption connectors.marker.color
- */
- /**
- * Set the line/border color of the connector markers. By default
- * this is the same as the marker color.
- *
- * @type {Highcharts.ColorString}
- * @since 6.2.0
- * @apioption connectors.marker.lineColor
- */
- /**
- * Enable markers for the connectors.
- */
- enabled: false,
- /**
- * Horizontal alignment of the markers relative to the points.
- *
- * @type {Highcharts.AlignValue}
- */
- align: 'center',
- /**
- * Vertical alignment of the markers relative to the points.
- *
- * @type {Highcharts.VerticalAlignValue}
- */
- verticalAlign: 'middle',
- /**
- * Whether or not to draw the markers inside the points.
- */
- inside: false,
- /**
- * Set the line/border width of the pathfinder markers.
- */
- lineWidth: 1
- },
- /**
- * Marker options specific to the start markers for this chart's
- * Pathfinder connectors. Overrides the generic marker options.
- *
- * @declare Highcharts.ConnectorsStartMarkerOptions
- * @extends connectors.marker
- * @since 6.2.0
- */
- startMarker: {
- /**
- * Set the symbol of the connector start markers.
- */
- symbol: 'diamond'
- },
- /**
- * Marker options specific to the end markers for this chart's
- * Pathfinder connectors. Overrides the generic marker options.
- *
- * @declare Highcharts.ConnectorsEndMarkerOptions
- * @extends connectors.marker
- * @since 6.2.0
- */
- endMarker: {
- /**
- * Set the symbol of the connector end markers.
- */
- symbol: 'arrow-filled'
- }
- }
- });
- /**
- * Override Pathfinder connector options for a series. Requires Highcharts Gantt
- * to be loaded.
- *
- * @declare Highcharts.SeriesConnectorsOptionsObject
- * @extends connectors
- * @since 6.2.0
- * @excluding enabled, algorithmMargin
- * @product gantt
- * @apioption plotOptions.series.connectors
- */
- /**
- * Connect to a point. This option can be either a string, referring to the ID
- * of another point, or an object, or an array of either. If the option is an
- * array, each element defines a connection.
- *
- * @sample gantt/pathfinder/demo
- * Different connection types
- *
- * @declare Highcharts.XrangePointConnectorsOptionsObject
- * @type {string|Array<string|*>|*}
- * @extends plotOptions.series.connectors
- * @since 6.2.0
- * @excluding enabled
- * @product gantt
- * @requires highcharts-gantt
- * @apioption series.xrange.data.connect
- */
- /**
- * The ID of the point to connect to.
- *
- * @type {string}
- * @since 6.2.0
- * @product gantt
- * @apioption series.xrange.data.connect.to
- */
- /**
- * Get point bounding box using plotX/plotY and shapeArgs. If using
- * graphic.getBBox() directly, the bbox will be affected by animation.
- *
- * @private
- * @function
- *
- * @param {Highcharts.Point} point
- * The point to get BB of.
- *
- * @return {Highcharts.Dictionary<number>|null}
- * Result xMax, xMin, yMax, yMin.
- */
- function getPointBB(point) {
- var shapeArgs = point.shapeArgs,
- bb;
- // Prefer using shapeArgs (columns)
- if (shapeArgs) {
- return {
- xMin: shapeArgs.x,
- xMax: shapeArgs.x + shapeArgs.width,
- yMin: shapeArgs.y,
- yMax: shapeArgs.y + shapeArgs.height
- };
- }
- // Otherwise use plotX/plotY and bb
- bb = point.graphic && point.graphic.getBBox();
- return bb ? {
- xMin: point.plotX - bb.width / 2,
- xMax: point.plotX + bb.width / 2,
- yMin: point.plotY - bb.height / 2,
- yMax: point.plotY + bb.height / 2
- } : null;
- }
- /**
- * Calculate margin to place around obstacles for the pathfinder in pixels.
- * Returns a minimum of 1 pixel margin.
- *
- * @private
- * @function
- *
- * @param {Array<object>} obstacles
- * Obstacles to calculate margin from.
- *
- * @return {number}
- * The calculated margin in pixels. At least 1.
- */
- function calculateObstacleMargin(obstacles) {
- var len = obstacles.length,
- i = 0,
- j,
- obstacleDistance,
- distances = [],
- // Compute smallest distance between two rectangles
- distance = function (a,
- b,
- bbMargin) {
- // Count the distance even if we are slightly off
- var margin = pick(bbMargin, 10),
- yOverlap = a.yMax + margin > b.yMin - margin &&
- a.yMin - margin < b.yMax + margin,
- xOverlap = a.xMax + margin > b.xMin - margin &&
- a.xMin - margin < b.xMax + margin,
- xDistance = yOverlap ? (a.xMin > b.xMax ? a.xMin - b.xMax : b.xMin - a.xMax) : Infinity,
- yDistance = xOverlap ? (a.yMin > b.yMax ? a.yMin - b.yMax : b.yMin - a.yMax) : Infinity;
- // If the rectangles collide, try recomputing with smaller margin.
- // If they collide anyway, discard the obstacle.
- if (xOverlap && yOverlap) {
- return (margin ?
- distance(a, b, Math.floor(margin / 2)) :
- Infinity);
- }
- return min(xDistance, yDistance);
- };
- // Go over all obstacles and compare them to the others.
- for (; i < len; ++i) {
- // Compare to all obstacles ahead. We will already have compared this
- // obstacle to the ones before.
- for (j = i + 1; j < len; ++j) {
- obstacleDistance = distance(obstacles[i], obstacles[j]);
- // TODO: Magic number 80
- if (obstacleDistance < 80) { // Ignore large distances
- distances.push(obstacleDistance);
- }
- }
- }
- // Ensure we always have at least one value, even in very spaceous charts
- distances.push(80);
- return max(Math.floor(distances.sort(function (a, b) {
- return (a - b);
- })[
- // Discard first 10% of the relevant distances, and then grab
- // the smallest one.
- Math.floor(distances.length / 10)] / 2 - 1 // Divide the distance by 2 and subtract 1.
- ), 1 // 1 is the minimum margin
- );
- }
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The Pathfinder class.
- *
- * @private
- * @class
- * @name Highcharts.Pathfinder
- *
- * @param {Highcharts.Chart} chart
- * The chart to operate on.
- */
- var Pathfinder = /** @class */ (function () {
- function Pathfinder(chart) {
- /* *
- *
- * Properties
- *
- * */
- this.chart = void 0;
- this.chartObstacles = void 0;
- this.chartObstacleMetrics = void 0;
- this.connections = void 0;
- this.group = void 0;
- this.lineObstacles = void 0;
- this.init(chart);
- }
- /**
- * @name Highcharts.Pathfinder#algorithms
- * @type {Highcharts.Dictionary<Function>}
- */
- /**
- * Initialize the Pathfinder object.
- *
- * @function Highcharts.Pathfinder#init
- *
- * @param {Highcharts.Chart} chart
- * The chart context.
- */
- Pathfinder.prototype.init = function (chart) {
- // Initialize pathfinder with chart context
- this.chart = chart;
- // Init connection reference list
- this.connections = [];
- // Recalculate paths/obstacles on chart redraw
- addEvent(chart, 'redraw', function () {
- this.pathfinder.update();
- });
- };
- /**
- * Update Pathfinder connections from scratch.
- *
- * @function Highcharts.Pathfinder#update
- *
- * @param {boolean} [deferRender]
- * Whether or not to defer rendering of connections until
- * series.afterAnimate event has fired. Used on first render.
- */
- Pathfinder.prototype.update = function (deferRender) {
- var chart = this.chart,
- pathfinder = this,
- oldConnections = pathfinder.connections;
- // Rebuild pathfinder connections from options
- pathfinder.connections = [];
- chart.series.forEach(function (series) {
- if (series.visible && !series.options.isInternal) {
- series.points.forEach(function (point) {
- var to,
- connects = (point.options &&
- point.options.connect &&
- splat(point.options.connect));
- if (point.visible && point.isInside !== false && connects) {
- connects.forEach(function (connect) {
- to = chart.get(typeof connect === 'string' ?
- connect : connect.to);
- if (to instanceof Point &&
- to.series.visible &&
- to.visible &&
- to.isInside !== false) {
- // Add new connection
- pathfinder.connections.push(new Connection(point, // from
- to, typeof connect === 'string' ?
- {} :
- connect));
- }
- });
- }
- });
- }
- });
- // Clear connections that should not be updated, and move old info over
- // to new connections.
- for (var j = 0, k, found, lenOld = oldConnections.length, lenNew = pathfinder.connections.length; j < lenOld; ++j) {
- found = false;
- for (k = 0; k < lenNew; ++k) {
- if (oldConnections[j].fromPoint ===
- pathfinder.connections[k].fromPoint &&
- oldConnections[j].toPoint ===
- pathfinder.connections[k].toPoint) {
- pathfinder.connections[k].graphics =
- oldConnections[j].graphics;
- found = true;
- break;
- }
- }
- if (!found) {
- oldConnections[j].destroy();
- }
- }
- // Clear obstacles to force recalculation. This must be done on every
- // redraw in case positions have changed. Recalculation is handled in
- // Connection.getPath on demand.
- delete this.chartObstacles;
- delete this.lineObstacles;
- // Draw the pending connections
- pathfinder.renderConnections(deferRender);
- };
- /**
- * Draw the chart's connecting paths.
- *
- * @function Highcharts.Pathfinder#renderConnections
- *
- * @param {boolean} [deferRender]
- * Whether or not to defer render until series animation is finished.
- * Used on first render.
- */
- Pathfinder.prototype.renderConnections = function (deferRender) {
- if (deferRender) {
- // Render after series are done animating
- this.chart.series.forEach(function (series) {
- var render = function () {
- // Find pathfinder connections belonging to this series
- // that haven't rendered, and render them now.
- var pathfinder = series.chart.pathfinder,
- conns = pathfinder && pathfinder.connections || [];
- conns.forEach(function (connection) {
- if (connection.fromPoint &&
- connection.fromPoint.series === series) {
- connection.render();
- }
- });
- if (series.pathfinderRemoveRenderEvent) {
- series.pathfinderRemoveRenderEvent();
- delete series.pathfinderRemoveRenderEvent;
- }
- };
- if (series.options.animation === false) {
- render();
- }
- else {
- series.pathfinderRemoveRenderEvent = addEvent(series, 'afterAnimate', render);
- }
- });
- }
- else {
- // Go through connections and render them
- this.connections.forEach(function (connection) {
- connection.render();
- });
- }
- };
- /**
- * Get obstacles for the points in the chart. Does not include connecting
- * lines from Pathfinder. Applies algorithmMargin to the obstacles.
- *
- * @function Highcharts.Pathfinder#getChartObstacles
- *
- * @param {object} options
- * Options for the calculation. Currenlty only
- * options.algorithmMargin.
- *
- * @return {Array<object>}
- * An array of calculated obstacles. Each obstacle is defined as an
- * object with xMin, xMax, yMin and yMax properties.
- */
- Pathfinder.prototype.getChartObstacles = function (options) {
- var obstacles = [],
- series = this.chart.series,
- margin = pick(options.algorithmMargin, 0),
- calculatedMargin;
- for (var i = 0, sLen = series.length; i < sLen; ++i) {
- if (series[i].visible && !series[i].options.isInternal) {
- for (var j = 0, pLen = series[i].points.length, bb, point; j < pLen; ++j) {
- point = series[i].points[j];
- if (point.visible) {
- bb = getPointBB(point);
- if (bb) {
- obstacles.push({
- xMin: bb.xMin - margin,
- xMax: bb.xMax + margin,
- yMin: bb.yMin - margin,
- yMax: bb.yMax + margin
- });
- }
- }
- }
- }
- }
- // Sort obstacles by xMin for optimization
- obstacles = obstacles.sort(function (a, b) {
- return a.xMin - b.xMin;
- });
- // Add auto-calculated margin if the option is not defined
- if (!defined(options.algorithmMargin)) {
- calculatedMargin =
- options.algorithmMargin =
- calculateObstacleMargin(obstacles);
- obstacles.forEach(function (obstacle) {
- obstacle.xMin -= calculatedMargin;
- obstacle.xMax += calculatedMargin;
- obstacle.yMin -= calculatedMargin;
- obstacle.yMax += calculatedMargin;
- });
- }
- return obstacles;
- };
- /**
- * Utility function to get metrics for obstacles:
- * - Widest obstacle width
- * - Tallest obstacle height
- *
- * @function Highcharts.Pathfinder#getObstacleMetrics
- *
- * @param {Array<object>} obstacles
- * An array of obstacles to inspect.
- *
- * @return {object}
- * The calculated metrics, as an object with maxHeight and maxWidth
- * properties.
- */
- Pathfinder.prototype.getObstacleMetrics = function (obstacles) {
- var maxWidth = 0,
- maxHeight = 0,
- width,
- height,
- i = obstacles.length;
- while (i--) {
- width = obstacles[i].xMax - obstacles[i].xMin;
- height = obstacles[i].yMax - obstacles[i].yMin;
- if (maxWidth < width) {
- maxWidth = width;
- }
- if (maxHeight < height) {
- maxHeight = height;
- }
- }
- return {
- maxHeight: maxHeight,
- maxWidth: maxWidth
- };
- };
- /**
- * Utility to get which direction to start the pathfinding algorithm
- * (X vs Y), calculated from a set of marker options.
- *
- * @function Highcharts.Pathfinder#getAlgorithmStartDirection
- *
- * @param {Highcharts.ConnectorsMarkerOptions} markerOptions
- * Marker options to calculate from.
- *
- * @return {boolean}
- * Returns true for X, false for Y, and undefined for autocalculate.
- */
- Pathfinder.prototype.getAlgorithmStartDirection = function (markerOptions) {
- var xCenter = markerOptions.align !== 'left' &&
- markerOptions.align !== 'right', yCenter = markerOptions.verticalAlign !== 'top' &&
- markerOptions.verticalAlign !== 'bottom', undef;
- return xCenter ?
- (yCenter ? undef : false) : // x is centered
- (yCenter ? true : undef); // x is off-center
- };
- return Pathfinder;
- }());
- Pathfinder.prototype.algorithms = pathfinderAlgorithms;
- // Add to Highcharts namespace
- H.Pathfinder = Pathfinder;
- // Add pathfinding capabilities to Points
- extend(Point.prototype, /** @lends Point.prototype */ {
- /**
- * Get coordinates of anchor point for pathfinder connection.
- *
- * @private
- * @function Highcharts.Point#getPathfinderAnchorPoint
- *
- * @param {Highcharts.ConnectorsMarkerOptions} markerOptions
- * Connection options for position on point.
- *
- * @return {Highcharts.PositionObject}
- * An object with x/y properties for the position. Coordinates are
- * in plot values, not relative to point.
- */
- getPathfinderAnchorPoint: function (markerOptions) {
- var bb = getPointBB(this),
- x,
- y;
- switch (markerOptions.align) { // eslint-disable-line default-case
- case 'right':
- x = 'xMax';
- break;
- case 'left':
- x = 'xMin';
- }
- switch (markerOptions.verticalAlign) { // eslint-disable-line default-case
- case 'top':
- y = 'yMin';
- break;
- case 'bottom':
- y = 'yMax';
- }
- return {
- x: x ? bb[x] : (bb.xMin + bb.xMax) / 2,
- y: y ? bb[y] : (bb.yMin + bb.yMax) / 2
- };
- },
- /**
- * Utility to get the angle from one point to another.
- *
- * @private
- * @function Highcharts.Point#getRadiansToVector
- *
- * @param {Highcharts.PositionObject} v1
- * The first vector, as an object with x/y properties.
- *
- * @param {Highcharts.PositionObject} v2
- * The second vector, as an object with x/y properties.
- *
- * @return {number}
- * The angle in degrees
- */
- getRadiansToVector: function (v1, v2) {
- var box;
- if (!defined(v2)) {
- box = getPointBB(this);
- if (box) {
- v2 = {
- x: (box.xMin + box.xMax) / 2,
- y: (box.yMin + box.yMax) / 2
- };
- }
- }
- return Math.atan2(v2.y - v1.y, v1.x - v2.x);
- },
- /**
- * Utility to get the position of the marker, based on the path angle and
- * the marker's radius.
- *
- * @private
- * @function Highcharts.Point#getMarkerVector
- *
- * @param {number} radians
- * The angle in radians from the point center to another vector.
- *
- * @param {number} markerRadius
- * The radius of the marker, to calculate the additional distance to
- * the center of the marker.
- *
- * @param {object} anchor
- * The anchor point of the path and marker as an object with x/y
- * properties.
- *
- * @return {object}
- * The marker vector as an object with x/y properties.
- */
- getMarkerVector: function (radians, markerRadius, anchor) {
- var twoPI = Math.PI * 2.0,
- theta = radians,
- bb = getPointBB(this),
- rectWidth = bb.xMax - bb.xMin,
- rectHeight = bb.yMax - bb.yMin,
- rAtan = Math.atan2(rectHeight,
- rectWidth),
- tanTheta = 1,
- leftOrRightRegion = false,
- rectHalfWidth = rectWidth / 2.0,
- rectHalfHeight = rectHeight / 2.0,
- rectHorizontalCenter = bb.xMin + rectHalfWidth,
- rectVerticalCenter = bb.yMin + rectHalfHeight,
- edgePoint = {
- x: rectHorizontalCenter,
- y: rectVerticalCenter
- },
- markerPoint = {},
- xFactor = 1,
- yFactor = 1;
- while (theta < -Math.PI) {
- theta += twoPI;
- }
- while (theta > Math.PI) {
- theta -= twoPI;
- }
- tanTheta = Math.tan(theta);
- if ((theta > -rAtan) && (theta <= rAtan)) {
- // Right side
- yFactor = -1;
- leftOrRightRegion = true;
- }
- else if (theta > rAtan && theta <= (Math.PI - rAtan)) {
- // Top side
- yFactor = -1;
- }
- else if (theta > (Math.PI - rAtan) || theta <= -(Math.PI - rAtan)) {
- // Left side
- xFactor = -1;
- leftOrRightRegion = true;
- }
- else {
- // Bottom side
- xFactor = -1;
- }
- // Correct the edgePoint according to the placement of the marker
- if (leftOrRightRegion) {
- edgePoint.x += xFactor * (rectHalfWidth);
- edgePoint.y += yFactor * (rectHalfWidth) * tanTheta;
- }
- else {
- edgePoint.x += xFactor * (rectHeight / (2.0 * tanTheta));
- edgePoint.y += yFactor * (rectHalfHeight);
- }
- if (anchor.x !== rectHorizontalCenter) {
- edgePoint.x = anchor.x;
- }
- if (anchor.y !== rectVerticalCenter) {
- edgePoint.y = anchor.y;
- }
- markerPoint.x = edgePoint.x + (markerRadius * Math.cos(theta));
- markerPoint.y = edgePoint.y - (markerRadius * Math.sin(theta));
- return markerPoint;
- }
- });
- /**
- * Warn if using legacy options. Copy the options over. Note that this will
- * still break if using the legacy options in chart.update, addSeries etc.
- * @private
- */
- function warnLegacy(chart) {
- if (chart.options.pathfinder ||
- chart.series.reduce(function (acc, series) {
- if (series.options) {
- merge(true, (series.options.connectors = series.options.connectors ||
- {}), series.options.pathfinder);
- }
- return acc || series.options && series.options.pathfinder;
- }, false)) {
- merge(true, (chart.options.connectors = chart.options.connectors || {}), chart.options.pathfinder);
- error('WARNING: Pathfinder options have been renamed. ' +
- 'Use "chart.connectors" or "series.connectors" instead.');
- }
- }
- // Initialize Pathfinder for charts
- Chart.prototype.callbacks.push(function (chart) {
- var options = chart.options;
- if (options.connectors.enabled !== false) {
- warnLegacy(chart);
- this.pathfinder = new Pathfinder(this);
- this.pathfinder.update(true); // First draw, defer render
- }
- });
- return Pathfinder;
- });
- _registerModule(_modules, 'Series/XRangeSeries.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Color.js'], _modules['Core/Series/Point.js'], _modules['Core/Utilities.js']], function (Axis, H, Color, Point, U) {
- /* *
- *
- * X-range series module
- *
- * (c) 2010-2020 Torstein Honsi, Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var color = Color.parse;
- var addEvent = U.addEvent,
- clamp = U.clamp,
- correctFloat = U.correctFloat,
- defined = U.defined,
- find = U.find,
- isNumber = U.isNumber,
- isObject = U.isObject,
- merge = U.merge,
- pick = U.pick,
- seriesType = U.seriesType;
- /* *
- * @interface Highcharts.PointOptionsObject in parts/Point.ts
- */ /**
- * The ending X value of the range point.
- * @name Highcharts.PointOptionsObject#x2
- * @type {number|undefined}
- * @requires modules/xrange
- */
- var columnType = H.seriesTypes.column,
- seriesTypes = H.seriesTypes,
- Series = H.Series;
- /**
- * Return color of a point based on its category.
- *
- * @private
- * @function getColorByCategory
- *
- * @param {object} series
- * The series which the point belongs to.
- *
- * @param {object} point
- * The point to calculate its color for.
- *
- * @return {object}
- * Returns an object containing the properties color and colorIndex.
- */
- function getColorByCategory(series, point) {
- var colors = series.options.colors || series.chart.options.colors,
- colorCount = colors ?
- colors.length :
- series.chart.options.chart.colorCount,
- colorIndex = point.y % colorCount,
- color = colors && colors[colorIndex];
- return {
- colorIndex: colorIndex,
- color: color
- };
- }
- /**
- * @private
- * @class
- * @name Highcharts.seriesTypes.xrange
- *
- * @augments Highcharts.Series
- */
- seriesType('xrange', 'column'
- /**
- * The X-range series displays ranges on the X axis, typically time
- * intervals with a start and end date.
- *
- * @sample {highcharts} highcharts/demo/x-range/
- * X-range
- * @sample {highcharts} highcharts/css/x-range/
- * Styled mode X-range
- * @sample {highcharts} highcharts/chart/inverted-xrange/
- * Inverted X-range
- *
- * @extends plotOptions.column
- * @since 6.0.0
- * @product highcharts highstock gantt
- * @excluding boostThreshold, crisp, cropThreshold, depth, edgeColor,
- * edgeWidth, findNearestPointBy, getExtremesFromAll,
- * negativeColor, pointInterval, pointIntervalUnit,
- * pointPlacement, pointRange, pointStart, softThreshold,
- * stacking, threshold, data, dataSorting, boostBlending
- * @requires modules/xrange
- * @optionparent plotOptions.xrange
- */
- , {
- /**
- * A partial fill for each point, typically used to visualize how much
- * of a task is performed. The partial fill object can be set either on
- * series or point level.
- *
- * @sample {highcharts} highcharts/demo/x-range
- * X-range with partial fill
- *
- * @product highcharts highstock gantt
- * @apioption plotOptions.xrange.partialFill
- */
- /**
- * The fill color to be used for partial fills. Defaults to a darker
- * shade of the point color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock gantt
- * @apioption plotOptions.xrange.partialFill.fill
- */
- /**
- * A partial fill for each point, typically used to visualize how much
- * of a task is performed. See [completed](series.gantt.data.completed).
- *
- * @sample gantt/demo/progress-indicator
- * Gantt with progress indicator
- *
- * @product gantt
- * @apioption plotOptions.gantt.partialFill
- */
- /**
- * In an X-range series, this option makes all points of the same Y-axis
- * category the same color.
- */
- colorByPoint: true,
- dataLabels: {
- formatter: function () {
- var point = this.point,
- amount = point.partialFill;
- if (isObject(amount)) {
- amount = amount.amount;
- }
- if (isNumber(amount) && amount > 0) {
- return correctFloat(amount * 100) + '%';
- }
- },
- inside: true,
- verticalAlign: 'middle'
- },
- tooltip: {
- headerFormat: '<span style="font-size: 10px">{point.x} - {point.x2}</span><br/>',
- pointFormat: '<span style="color:{point.color}">\u25CF</span> {series.name}: <b>{point.yCategory}</b><br/>'
- },
- borderRadius: 3,
- pointRange: 0
- }, {
- type: 'xrange',
- parallelArrays: ['x', 'x2', 'y'],
- requireSorting: false,
- animate: seriesTypes.line.prototype.animate,
- cropShoulder: 1,
- getExtremesFromAll: true,
- autoIncrement: H.noop,
- buildKDTree: H.noop,
- /* eslint-disable valid-jsdoc */
- /**
- * @private
- * @function Highcarts.seriesTypes.xrange#init
- * @return {void}
- */
- init: function () {
- seriesTypes.column.prototype.init.apply(this, arguments);
- this.options.stacking = void 0; // #13161
- },
- /**
- * Borrow the column series metrics, but with swapped axes. This gives
- * free access to features like groupPadding, grouping, pointWidth etc.
- *
- * @private
- * @function Highcharts.Series#getColumnMetrics
- *
- * @return {Highcharts.ColumnMetricsObject}
- */
- getColumnMetrics: function () {
- var metrics,
- chart = this.chart;
- /**
- * @private
- */
- function swapAxes() {
- chart.series.forEach(function (s) {
- var xAxis = s.xAxis;
- s.xAxis = s.yAxis;
- s.yAxis = xAxis;
- });
- }
- swapAxes();
- metrics = columnType.prototype.getColumnMetrics.call(this);
- swapAxes();
- return metrics;
- },
- /**
- * Override cropData to show a point where x or x2 is outside visible
- * range, but one of them is inside.
- *
- * @private
- * @function Highcharts.Series#cropData
- *
- * @param {Array<number>} xData
- *
- * @param {Array<number>} yData
- *
- * @param {number} min
- *
- * @param {number} max
- *
- * @param {number} [cropShoulder]
- *
- * @return {*}
- */
- cropData: function (xData, yData, min, max) {
- // Replace xData with x2Data to find the appropriate cropStart
- var cropData = Series.prototype.cropData,
- crop = cropData.call(this,
- this.x2Data,
- yData,
- min,
- max);
- // Re-insert the cropped xData
- crop.xData = xData.slice(crop.start, crop.end);
- return crop;
- },
- /**
- * Finds the index of an existing point that matches the given point
- * options.
- *
- * @private
- * @function Highcharts.Series#findPointIndex
- * @param {object} options The options of the point.
- * @returns {number|undefined} Returns index of a matching point,
- * returns undefined if no match is found.
- */
- findPointIndex: function (options) {
- var _a = this,
- cropped = _a.cropped,
- cropStart = _a.cropStart,
- points = _a.points;
- var id = options.id;
- var pointIndex;
- if (id) {
- var point = find(points,
- function (point) {
- return point.id === id;
- });
- pointIndex = point ? point.index : void 0;
- }
- if (typeof pointIndex === 'undefined') {
- var point = find(points,
- function (point) {
- return (point.x === options.x &&
- point.x2 === options.x2 &&
- !point.touched);
- });
- pointIndex = point ? point.index : void 0;
- }
- // Reduce pointIndex if data is cropped
- if (cropped &&
- isNumber(pointIndex) &&
- isNumber(cropStart) &&
- pointIndex >= cropStart) {
- pointIndex -= cropStart;
- }
- return pointIndex;
- },
- /**
- * @private
- * @function Highcharts.Series#translatePoint
- *
- * @param {Highcharts.Point} point
- */
- translatePoint: function (point) {
- var series = this,
- xAxis = series.xAxis,
- yAxis = series.yAxis,
- metrics = series.columnMetrics,
- options = series.options,
- minPointLength = options.minPointLength || 0,
- plotX = point.plotX,
- posX = pick(point.x2,
- point.x + (point.len || 0)),
- plotX2 = xAxis.translate(posX, 0, 0, 0, 1),
- length = Math.abs(plotX2 - plotX),
- widthDifference,
- shapeArgs,
- partialFill,
- inverted = this.chart.inverted,
- borderWidth = pick(options.borderWidth, 1),
- crisper = borderWidth % 2 / 2,
- yOffset = metrics.offset,
- pointHeight = Math.round(metrics.width),
- dlLeft,
- dlRight,
- dlWidth,
- clipRectWidth,
- tooltipYOffset;
- if (minPointLength) {
- widthDifference = minPointLength - length;
- if (widthDifference < 0) {
- widthDifference = 0;
- }
- plotX -= widthDifference / 2;
- plotX2 += widthDifference / 2;
- }
- plotX = Math.max(plotX, -10);
- plotX2 = clamp(plotX2, -10, xAxis.len + 10);
- // Handle individual pointWidth
- if (defined(point.options.pointWidth)) {
- yOffset -= ((Math.ceil(point.options.pointWidth) - pointHeight) / 2);
- pointHeight = Math.ceil(point.options.pointWidth);
- }
- // Apply pointPlacement to the Y axis
- if (options.pointPlacement &&
- isNumber(point.plotY) &&
- yAxis.categories) {
- point.plotY = yAxis.translate(point.y, 0, 1, 0, 1, options.pointPlacement);
- }
- point.shapeArgs = {
- x: Math.floor(Math.min(plotX, plotX2)) + crisper,
- y: Math.floor(point.plotY + yOffset) + crisper,
- width: Math.round(Math.abs(plotX2 - plotX)),
- height: pointHeight,
- r: series.options.borderRadius
- };
- // Align data labels inside the shape and inside the plot area
- dlLeft = point.shapeArgs.x;
- dlRight = dlLeft + point.shapeArgs.width;
- if (dlLeft < 0 || dlRight > xAxis.len) {
- dlLeft = clamp(dlLeft, 0, xAxis.len);
- dlRight = clamp(dlRight, 0, xAxis.len);
- dlWidth = dlRight - dlLeft;
- point.dlBox = merge(point.shapeArgs, {
- x: dlLeft,
- width: dlRight - dlLeft,
- centerX: dlWidth ? dlWidth / 2 : null
- });
- }
- else {
- point.dlBox = null;
- }
- // Tooltip position
- var tooltipPos = point.tooltipPos;
- var xIndex = !inverted ? 0 : 1;
- var yIndex = !inverted ? 1 : 0;
- tooltipYOffset = series.columnMetrics ?
- series.columnMetrics.offset : -metrics.width / 2;
- // Limit position by the correct axis size (#9727)
- tooltipPos[xIndex] = clamp(tooltipPos[xIndex] + ((!inverted ? 1 : -1) * (xAxis.reversed ? -1 : 1) *
- (length / 2)), 0, xAxis.len - 1);
- tooltipPos[yIndex] = clamp(tooltipPos[yIndex] + ((inverted ? -1 : 1) * tooltipYOffset), 0, yAxis.len - 1);
- // Add a partShapeArgs to the point, based on the shapeArgs property
- partialFill = point.partialFill;
- if (partialFill) {
- // Get the partial fill amount
- if (isObject(partialFill)) {
- partialFill = partialFill.amount;
- }
- // If it was not a number, assume 0
- if (!isNumber(partialFill)) {
- partialFill = 0;
- }
- shapeArgs = point.shapeArgs;
- point.partShapeArgs = {
- x: shapeArgs.x,
- y: shapeArgs.y,
- width: shapeArgs.width,
- height: shapeArgs.height,
- r: series.options.borderRadius
- };
- clipRectWidth = Math.max(Math.round(length * partialFill + point.plotX -
- plotX), 0);
- point.clipRectArgs = {
- x: xAxis.reversed ? // #10717
- shapeArgs.x + length - clipRectWidth :
- shapeArgs.x,
- y: shapeArgs.y,
- width: clipRectWidth,
- height: shapeArgs.height
- };
- }
- },
- /**
- * @private
- * @function Highcharts.Series#translate
- */
- translate: function () {
- columnType.prototype.translate.apply(this, arguments);
- this.points.forEach(function (point) {
- this.translatePoint(point);
- }, this);
- },
- /**
- * Draws a single point in the series. Needed for partial fill.
- *
- * This override turns point.graphic into a group containing the
- * original graphic and an overlay displaying the partial fill.
- *
- * @private
- * @function Highcharts.Series#drawPoint
- *
- * @param {Highcharts.Point} point
- * An instance of Point in the series.
- *
- * @param {"animate"|"attr"} verb
- * 'animate' (animates changes) or 'attr' (sets options)
- */
- drawPoint: function (point, verb) {
- var series = this,
- seriesOpts = series.options,
- renderer = series.chart.renderer,
- graphic = point.graphic,
- type = point.shapeType,
- shapeArgs = point.shapeArgs,
- partShapeArgs = point.partShapeArgs,
- clipRectArgs = point.clipRectArgs,
- pfOptions = point.partialFill,
- cutOff = seriesOpts.stacking && !seriesOpts.borderRadius,
- pointState = point.state,
- stateOpts = (seriesOpts.states[pointState || 'normal'] ||
- {}),
- pointStateVerb = typeof pointState === 'undefined' ?
- 'attr' : verb,
- pointAttr = series.pointAttribs(point,
- pointState),
- animation = pick(series.chart.options.chart.animation,
- stateOpts.animation),
- fill;
- if (!point.isNull && point.visible !== false) {
- // Original graphic
- if (graphic) { // update
- graphic.rect[verb](shapeArgs);
- }
- else {
- point.graphic = graphic = renderer.g('point')
- .addClass(point.getClassName())
- .add(point.group || series.group);
- graphic.rect = renderer[type](merge(shapeArgs))
- .addClass(point.getClassName())
- .addClass('highcharts-partfill-original')
- .add(graphic);
- }
- // Partial fill graphic
- if (partShapeArgs) {
- if (graphic.partRect) {
- graphic.partRect[verb](merge(partShapeArgs));
- graphic.partialClipRect[verb](merge(clipRectArgs));
- }
- else {
- graphic.partialClipRect = renderer.clipRect(clipRectArgs.x, clipRectArgs.y, clipRectArgs.width, clipRectArgs.height);
- graphic.partRect =
- renderer[type](partShapeArgs)
- .addClass('highcharts-partfill-overlay')
- .add(graphic)
- .clip(graphic.partialClipRect);
- }
- }
- // Presentational
- if (!series.chart.styledMode) {
- graphic
- .rect[verb](pointAttr, animation)
- .shadow(seriesOpts.shadow, null, cutOff);
- if (partShapeArgs) {
- // Ensure pfOptions is an object
- if (!isObject(pfOptions)) {
- pfOptions = {};
- }
- if (isObject(seriesOpts.partialFill)) {
- pfOptions = merge(pfOptions, seriesOpts.partialFill);
- }
- fill = (pfOptions.fill ||
- color(pointAttr.fill).brighten(-0.3).get() ||
- color(point.color || series.color)
- .brighten(-0.3).get());
- pointAttr.fill = fill;
- graphic
- .partRect[pointStateVerb](pointAttr, animation)
- .shadow(seriesOpts.shadow, null, cutOff);
- }
- }
- }
- else if (graphic) {
- point.graphic = graphic.destroy(); // #1269
- }
- },
- /**
- * @private
- * @function Highcharts.Series#drawPoints
- */
- drawPoints: function () {
- var series = this,
- verb = series.getAnimationVerb();
- // Draw the columns
- series.points.forEach(function (point) {
- series.drawPoint(point, verb);
- });
- },
- /**
- * Returns "animate", or "attr" if the number of points is above the
- * animation limit.
- *
- * @private
- * @function Highcharts.Series#getAnimationVerb
- *
- * @return {string}
- */
- getAnimationVerb: function () {
- return (this.chart.pointCount < (this.options.animationLimit || 250) ?
- 'animate' :
- 'attr');
- }
- /*
- // Override to remove stroke from points. For partial fill.
- pointAttribs: function () {
- var series = this,
- retVal = columnType.prototype.pointAttribs
- .apply(series,
- arguments);
- //retVal['stroke-width'] = 0;
- return retVal;
- }
- //*/
- /* eslint-enable valid-jsdoc */
- }, {
- /**
- * The ending X value of the range point.
- * @name Highcharts.Point#x2
- * @type {number|undefined}
- * @requires modules/xrange
- */
- /**
- * Extend applyOptions so that `colorByPoint` for x-range means that one
- * color is applied per Y axis category.
- *
- * @private
- * @function Highcharts.Point#applyOptions
- *
- * @return {Highcharts.Series}
- */
- /* eslint-disable valid-jsdoc */
- /**
- * @private
- */
- resolveColor: function () {
- var series = this.series,
- colorByPoint;
- if (series.options.colorByPoint && !this.options.color) {
- colorByPoint = getColorByCategory(series, this);
- if (!series.chart.styledMode) {
- this.color = colorByPoint.color;
- }
- if (!this.options.colorIndex) {
- this.colorIndex = colorByPoint.colorIndex;
- }
- }
- else if (!this.color) {
- this.color = series.color;
- }
- },
- /**
- * Extend init to have y default to 0.
- *
- * @private
- * @function Highcharts.Point#init
- *
- * @return {Highcharts.Point}
- */
- init: function () {
- Point.prototype.init.apply(this, arguments);
- if (!this.y) {
- this.y = 0;
- }
- return this;
- },
- /**
- * @private
- * @function Highcharts.Point#setState
- */
- setState: function () {
- Point.prototype.setState.apply(this, arguments);
- this.series.drawPoint(this, this.series.getAnimationVerb());
- },
- /**
- * @private
- * @function Highcharts.Point#getLabelConfig
- *
- * @return {Highcharts.PointLabelObject}
- */
- // Add x2 and yCategory to the available properties for tooltip formats
- getLabelConfig: function () {
- var point = this,
- cfg = Point.prototype.getLabelConfig.call(point),
- yCats = point.series.yAxis.categories;
- cfg.x2 = point.x2;
- cfg.yCategory = point.yCategory = yCats && yCats[point.y];
- return cfg;
- },
- tooltipDateKeys: ['x', 'x2'],
- /**
- * @private
- * @function Highcharts.Point#isValid
- *
- * @return {boolean}
- */
- isValid: function () {
- return typeof this.x === 'number' &&
- typeof this.x2 === 'number';
- }
- /* eslint-enable valid-jsdoc */
- });
- /**
- * Max x2 should be considered in xAxis extremes
- */
- addEvent(Axis, 'afterGetSeriesExtremes', function () {
- var axis = this, // eslint-disable-line no-invalid-this
- axisSeries = axis.series,
- dataMax,
- modMax;
- if (axis.isXAxis) {
- dataMax = pick(axis.dataMax, -Number.MAX_VALUE);
- axisSeries.forEach(function (series) {
- if (series.x2Data) {
- series.x2Data
- .forEach(function (val) {
- if (val > dataMax) {
- dataMax = val;
- modMax = true;
- }
- });
- }
- });
- if (modMax) {
- axis.dataMax = dataMax;
- }
- }
- });
- /**
- * An `xrange` series. If the [type](#series.xrange.type) option is not
- * specified, it is inherited from [chart.type](#chart.type).
- *
- * @extends series,plotOptions.xrange
- * @excluding boostThreshold, crisp, cropThreshold, depth, edgeColor, edgeWidth,
- * findNearestPointBy, getExtremesFromAll, negativeColor,
- * pointInterval, pointIntervalUnit, pointPlacement, pointRange,
- * pointStart, softThreshold, stacking, threshold, dataSorting,
- * boostBlending
- * @product highcharts highstock gantt
- * @requires modules/xrange
- * @apioption series.xrange
- */
- /**
- * An array of data points for the series. For the `xrange` series type,
- * points can be given in the following ways:
- *
- * 1. An array of objects with named values. The objects are point configuration
- * objects as seen below.
- * ```js
- * data: [{
- * x: Date.UTC(2017, 0, 1),
- * x2: Date.UTC(2017, 0, 3),
- * name: "Test",
- * y: 0,
- * color: "#00FF00"
- * }, {
- * x: Date.UTC(2017, 0, 4),
- * x2: Date.UTC(2017, 0, 5),
- * name: "Deploy",
- * y: 1,
- * color: "#FF0000"
- * }]
- * ```
- *
- * @sample {highcharts} highcharts/series/data-array-of-objects/
- * Config objects
- *
- * @declare Highcharts.XrangePointOptionsObject
- * @type {Array<*>}
- * @extends series.line.data
- * @product highcharts highstock gantt
- * @apioption series.xrange.data
- */
- /**
- * The starting X value of the range point.
- *
- * @sample {highcharts} highcharts/demo/x-range
- * X-range
- *
- * @type {number}
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.x
- */
- /**
- * The ending X value of the range point.
- *
- * @sample {highcharts} highcharts/demo/x-range
- * X-range
- *
- * @type {number}
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.x2
- */
- /**
- * The Y value of the range point.
- *
- * @sample {highcharts} highcharts/demo/x-range
- * X-range
- *
- * @type {number}
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.y
- */
- /**
- * A partial fill for each point, typically used to visualize how much of
- * a task is performed. The partial fill object can be set either on series
- * or point level.
- *
- * @sample {highcharts} highcharts/demo/x-range
- * X-range with partial fill
- *
- * @declare Highcharts.XrangePointPartialFillOptionsObject
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.partialFill
- */
- /**
- * The amount of the X-range point to be filled. Values can be 0-1 and are
- * converted to percentages in the default data label formatter.
- *
- * @type {number}
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.partialFill.amount
- */
- /**
- * The fill color to be used for partial fills. Defaults to a darker shade
- * of the point color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @product highcharts highstock gantt
- * @apioption series.xrange.data.partialFill.fill
- */
- ''; // adds doclets above to transpiled file
- });
- _registerModule(_modules, 'Series/GanttSeries.js', [_modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Utilities.js']], function (H, O, U) {
- /* *
- *
- * (c) 2016-2020 Highsoft AS
- *
- * Author: Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var dateFormat = O.dateFormat;
- var isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- seriesType = U.seriesType,
- splat = U.splat;
- var seriesTypes = H.seriesTypes,
- Series = H.Series,
- parent = seriesTypes.xrange;
- /**
- * @private
- * @class
- * @name Highcharts.seriesTypes.gantt
- *
- * @augments Highcharts.Series
- */
- seriesType('gantt', 'xrange'
- /**
- * A `gantt` series. If the [type](#series.gantt.type) option is not specified,
- * it is inherited from [chart.type](#chart.type).
- *
- * @extends plotOptions.xrange
- * @product gantt
- * @requires highcharts-gantt
- * @optionparent plotOptions.gantt
- */
- , {
- // options - default options merged with parent
- grouping: false,
- dataLabels: {
- enabled: true
- },
- tooltip: {
- headerFormat: '<span style="font-size: 10px">{series.name}</span><br/>',
- pointFormat: null,
- pointFormatter: function () {
- var point = this,
- series = point.series,
- tooltip = series.chart.tooltip,
- xAxis = series.xAxis,
- formats = series.tooltipOptions.dateTimeLabelFormats,
- startOfWeek = xAxis.options.startOfWeek,
- ttOptions = series.tooltipOptions,
- format = ttOptions.xDateFormat,
- start,
- end,
- milestone = point.options.milestone,
- retVal = '<b>' + (point.name || point.yCategory) + '</b>';
- if (ttOptions.pointFormat) {
- return point.tooltipFormatter(ttOptions.pointFormat);
- }
- if (!format) {
- format = splat(tooltip.getDateFormat(xAxis.closestPointRange, point.start, startOfWeek, formats))[0];
- }
- start = dateFormat(format, point.start);
- end = dateFormat(format, point.end);
- retVal += '<br/>';
- if (!milestone) {
- retVal += 'Start: ' + start + '<br/>';
- retVal += 'End: ' + end + '<br/>';
- }
- else {
- retVal += start + '<br/>';
- }
- return retVal;
- }
- },
- connectors: {
- type: 'simpleConnect',
- /**
- * @declare Highcharts.ConnectorsAnimationOptionsObject
- */
- animation: {
- reversed: true // Dependencies go from child to parent
- },
- startMarker: {
- enabled: true,
- symbol: 'arrow-filled',
- radius: 4,
- fill: '#fa0',
- align: 'left'
- },
- endMarker: {
- enabled: false,
- align: 'right'
- }
- }
- }, {
- pointArrayMap: ['start', 'end', 'y'],
- // Keyboard navigation, don't use nearest vertical mode
- keyboardMoveVertical: false,
- /* eslint-disable valid-jsdoc */
- /**
- * Handle milestones, as they have no x2.
- * @private
- */
- translatePoint: function (point) {
- var series = this,
- shapeArgs,
- size;
- parent.prototype.translatePoint.call(series, point);
- if (point.options.milestone) {
- shapeArgs = point.shapeArgs;
- size = shapeArgs.height;
- point.shapeArgs = {
- x: shapeArgs.x - (size / 2),
- y: shapeArgs.y,
- width: size,
- height: size
- };
- }
- },
- /**
- * Draws a single point in the series.
- *
- * This override draws the point as a diamond if point.options.milestone
- * is true, and uses the original drawPoint() if it is false or not set.
- *
- * @requires highcharts-gantt
- *
- * @private
- * @function Highcharts.seriesTypes.gantt#drawPoint
- *
- * @param {Highcharts.Point} point
- * An instance of Point in the series
- *
- * @param {"animate"|"attr"} verb
- * 'animate' (animates changes) or 'attr' (sets options)
- *
- * @return {void}
- */
- drawPoint: function (point, verb) {
- var series = this,
- seriesOpts = series.options,
- renderer = series.chart.renderer,
- shapeArgs = point.shapeArgs,
- plotY = point.plotY,
- graphic = point.graphic,
- state = point.selected && 'select',
- cutOff = seriesOpts.stacking && !seriesOpts.borderRadius,
- diamondShape;
- if (point.options.milestone) {
- if (isNumber(plotY) && point.y !== null && point.visible !== false) {
- diamondShape = renderer.symbols.diamond(shapeArgs.x, shapeArgs.y, shapeArgs.width, shapeArgs.height);
- if (graphic) {
- graphic[verb]({
- d: diamondShape
- });
- }
- else {
- point.graphic = graphic = renderer.path(diamondShape)
- .addClass(point.getClassName(), true)
- .add(point.group || series.group);
- }
- // Presentational
- if (!series.chart.styledMode) {
- point.graphic
- .attr(series.pointAttribs(point, state))
- .shadow(seriesOpts.shadow, null, cutOff);
- }
- }
- else if (graphic) {
- point.graphic = graphic.destroy(); // #1269
- }
- }
- else {
- parent.prototype.drawPoint.call(series, point, verb);
- }
- },
- setData: Series.prototype.setData,
- /**
- * @private
- */
- setGanttPointAliases: function (options) {
- /**
- * Add a value to options if the value exists.
- * @private
- */
- function addIfExists(prop, val) {
- if (typeof val !== 'undefined') {
- options[prop] = val;
- }
- }
- addIfExists('x', pick(options.start, options.x));
- addIfExists('x2', pick(options.end, options.x2));
- addIfExists('partialFill', pick(options.completed, options.partialFill));
- addIfExists('connect', pick(options.dependency, options.connect));
- }
- /* eslint-enable valid-jsdoc */
- }, merge(parent.prototype.pointClass.prototype, {
- // pointProps - point member overrides. We inherit from parent as well.
- /* eslint-disable valid-jsdoc */
- /**
- * Applies the options containing the x and y data and possible some
- * extra properties. This is called on point init or from point.update.
- *
- * @private
- * @function Highcharts.Point#applyOptions
- *
- * @param {object} options
- * The point options
- *
- * @param {number} x
- * The x value
- *
- * @return {Highcharts.Point}
- * The Point instance
- */
- applyOptions: function (options, x) {
- var point = this,
- retVal = merge(options);
- H.seriesTypes.gantt.prototype.setGanttPointAliases(retVal);
- retVal = parent.prototype.pointClass.prototype.applyOptions
- .call(point, retVal, x);
- return retVal;
- },
- isValid: function () {
- return ((typeof this.start === 'number' ||
- typeof this.x === 'number') &&
- (typeof this.end === 'number' ||
- typeof this.x2 === 'number' ||
- this.milestone));
- }
- /* eslint-enable valid-jsdoc */
- }));
- /**
- * A `gantt` series.
- *
- * @extends series,plotOptions.gantt
- * @excluding boostThreshold, connectors, dashStyle, findNearestPointBy,
- * getExtremesFromAll, marker, negativeColor, pointInterval,
- * pointIntervalUnit, pointPlacement, pointStart
- * @product gantt
- * @requires highcharts-gantt
- * @apioption series.gantt
- */
- /**
- * Data for a Gantt series.
- *
- * @declare Highcharts.GanttPointOptionsObject
- * @type {Array<*>}
- * @extends series.xrange.data
- * @excluding className, color, colorIndex, connect, dataLabels, events,
- * partialFill, selected, x, x2
- * @product gantt
- * @apioption series.gantt.data
- */
- /**
- * Whether the grid node belonging to this point should start as collapsed. Used
- * in axes of type treegrid.
- *
- * @sample {gantt} gantt/treegrid-axis/collapsed/
- * Start as collapsed
- *
- * @type {boolean}
- * @default false
- * @product gantt
- * @apioption series.gantt.data.collapsed
- */
- /**
- * The start time of a task.
- *
- * @type {number}
- * @product gantt
- * @apioption series.gantt.data.start
- */
- /**
- * The end time of a task.
- *
- * @type {number}
- * @product gantt
- * @apioption series.gantt.data.end
- */
- /**
- * The Y value of a task.
- *
- * @type {number}
- * @product gantt
- * @apioption series.gantt.data.y
- */
- /**
- * The name of a task. If a `treegrid` y-axis is used (default in Gantt charts),
- * this will be picked up automatically, and used to calculate the y-value.
- *
- * @type {string}
- * @product gantt
- * @apioption series.gantt.data.name
- */
- /**
- * Progress indicator, how much of the task completed. If it is a number, the
- * `fill` will be applied automatically.
- *
- * @sample {gantt} gantt/demo/progress-indicator
- * Progress indicator
- *
- * @type {number|*}
- * @extends series.xrange.data.partialFill
- * @product gantt
- * @apioption series.gantt.data.completed
- */
- /**
- * The amount of the progress indicator, ranging from 0 (not started) to 1
- * (finished).
- *
- * @type {number}
- * @default 0
- * @apioption series.gantt.data.completed.amount
- */
- /**
- * The fill of the progress indicator. Defaults to a darkened variety of the
- * main color.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption series.gantt.data.completed.fill
- */
- /**
- * The ID of the point (task) that this point depends on in Gantt charts.
- * Aliases [connect](series.xrange.data.connect). Can also be an object,
- * specifying further connecting [options](series.gantt.connectors) between the
- * points. Multiple connections can be specified by providing an array.
- *
- * @sample gantt/demo/project-management
- * Dependencies
- * @sample gantt/pathfinder/demo
- * Different connection types
- *
- * @type {string|Array<string|*>|*}
- * @extends series.xrange.data.connect
- * @since 6.2.0
- * @product gantt
- * @apioption series.gantt.data.dependency
- */
- /**
- * Whether this point is a milestone. If so, only the `start` option is handled,
- * while `end` is ignored.
- *
- * @sample gantt/gantt/milestones
- * Milestones
- *
- * @type {boolean}
- * @since 6.2.0
- * @product gantt
- * @apioption series.gantt.data.milestone
- */
- /**
- * The ID of the parent point (task) of this point in Gantt charts.
- *
- * @sample gantt/demo/subtasks
- * Gantt chart with subtasks
- *
- * @type {string}
- * @since 6.2.0
- * @product gantt
- * @apioption series.gantt.data.parent
- */
- /**
- * @excluding afterAnimate
- * @apioption series.gantt.events
- */
- ''; // adds doclets above to the transpiled file
- });
- _registerModule(_modules, 'Core/Chart/GanttChart.js', [_modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (Chart, H, U) {
- /* *
- *
- * (c) 2016-2020 Highsoft AS
- *
- * Author: Lars A. V. Cabrera
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var getOptions = U.getOptions,
- isArray = U.isArray,
- merge = U.merge,
- splat = U.splat;
- /**
- * Factory function for Gantt charts.
- *
- * @example
- * // Render a chart in to div#container
- * var chart = Highcharts.ganttChart('container', {
- * title: {
- * text: 'My chart'
- * },
- * series: [{
- * data: ...
- * }]
- * });
- *
- * @function Highcharts.ganttChart
- *
- * @param {string|Highcharts.HTMLDOMElement} renderTo
- * The DOM element to render to, or its id.
- *
- * @param {Highcharts.Options} options
- * The chart options structure.
- *
- * @param {Highcharts.ChartCallbackFunction} [callback]
- * Function to run when the chart has loaded and and all external images
- * are loaded. Defining a
- * [chart.events.load](https://api.highcharts.com/highcharts/chart.events.load)
- * handler is equivalent.
- *
- * @return {Highcharts.Chart}
- * Returns the Chart object.
- */
- H.ganttChart = function (renderTo, options, callback) {
- var hasRenderToArg = typeof renderTo === 'string' || renderTo.nodeName,
- seriesOptions = options.series,
- defaultOptions = getOptions(),
- defaultLinkedTo,
- userOptions = options;
- options = arguments[hasRenderToArg ? 1 : 0];
- // If user hasn't defined axes as array, make it into an array and add a
- // second axis by default.
- if (!isArray(options.xAxis)) {
- options.xAxis = [options.xAxis || {}, {}];
- }
- // apply X axis options to both single and multi x axes
- options.xAxis = options.xAxis.map(function (xAxisOptions, i) {
- if (i === 1) { // Second xAxis
- defaultLinkedTo = 0;
- }
- return merge(defaultOptions.xAxis, {
- grid: {
- enabled: true
- },
- opposite: true,
- linkedTo: defaultLinkedTo
- }, xAxisOptions, // user options
- {
- type: 'datetime'
- });
- });
- // apply Y axis options to both single and multi y axes
- options.yAxis = (splat(options.yAxis || {})).map(function (yAxisOptions) {
- return merge(defaultOptions.yAxis, // #3802
- {
- grid: {
- enabled: true
- },
- staticScale: 50,
- reversed: true,
- // Set default type treegrid, but only if 'categories' is
- // undefined
- type: yAxisOptions.categories ? yAxisOptions.type : 'treegrid'
- }, yAxisOptions // user options
- );
- });
- options.series = null;
- options = merge(true, {
- chart: {
- type: 'gantt'
- },
- title: {
- text: null
- },
- legend: {
- enabled: false
- },
- navigator: {
- series: { type: 'gantt' }
- }
- }, options, // user's options
- // forced options
- {
- isGantt: true
- });
- options.series = userOptions.series = seriesOptions;
- (options.series || []).forEach(function (series) {
- if (series.data) {
- series.data.forEach(function (point) {
- H.seriesTypes.gantt.prototype.setGanttPointAliases(point);
- });
- }
- });
- return hasRenderToArg ?
- new Chart(renderTo, options, callback) :
- new Chart(options, options); // @todo does not look correct
- };
- });
- _registerModule(_modules, 'Core/Axis/ScrollbarAxis.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- defined = U.defined,
- pick = U.pick;
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Creates scrollbars if enabled.
- *
- * @private
- */
- var ScrollbarAxis = /** @class */ (function () {
- function ScrollbarAxis() {
- }
- /**
- * Attaches to axis events to create scrollbars if enabled.
- *
- * @private
- *
- * @param AxisClass
- * Axis class to extend.
- *
- * @param ScrollbarClass
- * Scrollbar class to use.
- */
- ScrollbarAxis.compose = function (AxisClass, ScrollbarClass) {
- // Wrap axis initialization and create scrollbar if enabled:
- addEvent(AxisClass, 'afterInit', function () {
- var axis = this;
- if (axis.options &&
- axis.options.scrollbar &&
- axis.options.scrollbar.enabled) {
- // Predefined options:
- axis.options.scrollbar.vertical = !axis.horiz;
- axis.options.startOnTick = axis.options.endOnTick = false;
- axis.scrollbar = new ScrollbarClass(axis.chart.renderer, axis.options.scrollbar, axis.chart);
- addEvent(axis.scrollbar, 'changed', function (e) {
- var axisMin = pick(axis.options && axis.options.min,
- axis.min),
- axisMax = pick(axis.options && axis.options.max,
- axis.max),
- unitedMin = defined(axis.dataMin) ?
- Math.min(axisMin,
- axis.min,
- axis.dataMin) : axisMin,
- unitedMax = defined(axis.dataMax) ?
- Math.max(axisMax,
- axis.max,
- axis.dataMax) : axisMax,
- range = unitedMax - unitedMin,
- to,
- from;
- // #12834, scroll when show/hide series, wrong extremes
- if (!defined(axisMin) || !defined(axisMax)) {
- return;
- }
- if ((axis.horiz && !axis.reversed) ||
- (!axis.horiz && axis.reversed)) {
- to = unitedMin + range * this.to;
- from = unitedMin + range * this.from;
- }
- else {
- // y-values in browser are reversed, but this also
- // applies for reversed horizontal axis:
- to = unitedMin + range * (1 - this.from);
- from = unitedMin + range * (1 - this.to);
- }
- if (pick(this.options.liveRedraw, H.svg && !H.isTouchDevice && !this.chart.isBoosting) ||
- // Mouseup always should change extremes
- e.DOMType === 'mouseup' ||
- // Internal events
- !defined(e.DOMType)) {
- axis.setExtremes(from, to, true, e.DOMType !== 'mousemove', e);
- }
- else {
- // When live redraw is disabled, don't change extremes
- // Only change the position of the scollbar thumb
- this.setRange(this.from, this.to);
- }
- });
- }
- });
- // Wrap rendering axis, and update scrollbar if one is created:
- addEvent(AxisClass, 'afterRender', function () {
- var axis = this,
- scrollMin = Math.min(pick(axis.options.min,
- axis.min),
- axis.min,
- pick(axis.dataMin,
- axis.min) // #6930
- ),
- scrollMax = Math.max(pick(axis.options.max,
- axis.max),
- axis.max,
- pick(axis.dataMax,
- axis.max) // #6930
- ),
- scrollbar = axis.scrollbar,
- offset = axis.axisTitleMargin + (axis.titleOffset || 0),
- scrollbarsOffsets = axis.chart.scrollbarsOffsets,
- axisMargin = axis.options.margin || 0,
- offsetsIndex,
- from,
- to;
- if (scrollbar) {
- if (axis.horiz) {
- // Reserve space for labels/title
- if (!axis.opposite) {
- scrollbarsOffsets[1] += offset;
- }
- scrollbar.position(axis.left, axis.top + axis.height + 2 + scrollbarsOffsets[1] -
- (axis.opposite ? axisMargin : 0), axis.width, axis.height);
- // Next scrollbar should reserve space for margin (if set)
- if (!axis.opposite) {
- scrollbarsOffsets[1] += axisMargin;
- }
- offsetsIndex = 1;
- }
- else {
- // Reserve space for labels/title
- if (axis.opposite) {
- scrollbarsOffsets[0] += offset;
- }
- scrollbar.position(axis.left + axis.width + 2 + scrollbarsOffsets[0] -
- (axis.opposite ? 0 : axisMargin), axis.top, axis.width, axis.height);
- // Next scrollbar should reserve space for margin (if set)
- if (axis.opposite) {
- scrollbarsOffsets[0] += axisMargin;
- }
- offsetsIndex = 0;
- }
- scrollbarsOffsets[offsetsIndex] += scrollbar.size +
- scrollbar.options.margin;
- if (isNaN(scrollMin) ||
- isNaN(scrollMax) ||
- !defined(axis.min) ||
- !defined(axis.max) ||
- axis.min === axis.max // #10733
- ) {
- // default action: when extremes are the same or there is
- // not extremes on the axis, but scrollbar exists, make it
- // full size
- scrollbar.setRange(0, 1);
- }
- else {
- from =
- (axis.min - scrollMin) / (scrollMax - scrollMin);
- to =
- (axis.max - scrollMin) / (scrollMax - scrollMin);
- if ((axis.horiz && !axis.reversed) ||
- (!axis.horiz && axis.reversed)) {
- scrollbar.setRange(from, to);
- }
- else {
- // inverse vertical axis
- scrollbar.setRange(1 - to, 1 - from);
- }
- }
- }
- });
- // Make space for a scrollbar:
- addEvent(AxisClass, 'afterGetOffset', function () {
- var axis = this,
- index = axis.horiz ? 2 : 1,
- scrollbar = axis.scrollbar;
- if (scrollbar) {
- axis.chart.scrollbarsOffsets = [0, 0]; // reset scrollbars offsets
- axis.chart.axisOffset[index] +=
- scrollbar.size + scrollbar.options.margin;
- }
- });
- };
- return ScrollbarAxis;
- }());
- return ScrollbarAxis;
- });
- _registerModule(_modules, 'Core/Scrollbar.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Globals.js'], _modules['Core/Axis/ScrollbarAxis.js'], _modules['Core/Utilities.js'], _modules['Core/Options.js']], function (Axis, H, ScrollbarAxis, U, O) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var addEvent = U.addEvent,
- correctFloat = U.correctFloat,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- fireEvent = U.fireEvent,
- merge = U.merge,
- pick = U.pick,
- removeEvent = U.removeEvent;
- var defaultOptions = O.defaultOptions;
- var hasTouch = H.hasTouch,
- isTouchDevice = H.isTouchDevice;
- /**
- * When we have vertical scrollbar, rifles and arrow in buttons should be
- * rotated. The same method is used in Navigator's handles, to rotate them.
- *
- * @function Highcharts.swapXY
- *
- * @param {Highcharts.SVGPathArray} path
- * Path to be rotated.
- *
- * @param {boolean} [vertical]
- * If vertical scrollbar, swap x-y values.
- *
- * @return {Highcharts.SVGPathArray}
- * Rotated path.
- *
- * @requires modules/stock
- */
- var swapXY = H.swapXY = function (path,
- vertical) {
- if (vertical) {
- path.forEach(function (seg) {
- var len = seg.length;
- var temp;
- for (var i = 0; i < len; i += 2) {
- temp = seg[i + 1];
- if (typeof temp === 'number') {
- seg[i + 1] = seg[i + 2];
- seg[i + 2] = temp;
- }
- }
- });
- }
- return path;
- };
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * A reusable scrollbar, internally used in Highstock's navigator and optionally
- * on individual axes.
- *
- * @private
- * @class
- * @name Highcharts.Scrollbar
- * @param {Highcharts.SVGRenderer} renderer
- * @param {Highcharts.ScrollbarOptions} options
- * @param {Highcharts.Chart} chart
- */
- var Scrollbar = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function Scrollbar(renderer, options, chart) {
- /* *
- *
- * Properties
- *
- * */
- this._events = [];
- this.chartX = 0;
- this.chartY = 0;
- this.from = 0;
- this.group = void 0;
- this.scrollbar = void 0;
- this.scrollbarButtons = [];
- this.scrollbarGroup = void 0;
- this.scrollbarLeft = 0;
- this.scrollbarRifles = void 0;
- this.scrollbarStrokeWidth = 1;
- this.scrollbarTop = 0;
- this.size = 0;
- this.to = 0;
- this.track = void 0;
- this.trackBorderWidth = 1;
- this.userOptions = {};
- this.x = 0;
- this.y = 0;
- this.chart = chart;
- this.options = options;
- this.renderer = chart.renderer;
- this.init(renderer, options, chart);
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * Set up the mouse and touch events for the Scrollbar
- *
- * @private
- * @function Highcharts.Scrollbar#addEvents
- * @return {void}
- */
- Scrollbar.prototype.addEvents = function () {
- var buttonsOrder = this.options.inverted ? [1, 0] : [0, 1],
- buttons = this.scrollbarButtons,
- bar = this.scrollbarGroup.element,
- track = this.track.element,
- mouseDownHandler = this.mouseDownHandler.bind(this),
- mouseMoveHandler = this.mouseMoveHandler.bind(this),
- mouseUpHandler = this.mouseUpHandler.bind(this),
- _events;
- // Mouse events
- _events = [
- [buttons[buttonsOrder[0]].element, 'click', this.buttonToMinClick.bind(this)],
- [buttons[buttonsOrder[1]].element, 'click', this.buttonToMaxClick.bind(this)],
- [track, 'click', this.trackClick.bind(this)],
- [bar, 'mousedown', mouseDownHandler],
- [bar.ownerDocument, 'mousemove', mouseMoveHandler],
- [bar.ownerDocument, 'mouseup', mouseUpHandler]
- ];
- // Touch events
- if (hasTouch) {
- _events.push([bar, 'touchstart', mouseDownHandler], [bar.ownerDocument, 'touchmove', mouseMoveHandler], [bar.ownerDocument, 'touchend', mouseUpHandler]);
- }
- // Add them all
- _events.forEach(function (args) {
- addEvent.apply(null, args);
- });
- this._events = _events;
- };
- Scrollbar.prototype.buttonToMaxClick = function (e) {
- var scroller = this;
- var range = (scroller.to - scroller.from) * pick(scroller.options.step, 0.2);
- scroller.updatePosition(scroller.from + range, scroller.to + range);
- fireEvent(scroller, 'changed', {
- from: scroller.from,
- to: scroller.to,
- trigger: 'scrollbar',
- DOMEvent: e
- });
- };
- Scrollbar.prototype.buttonToMinClick = function (e) {
- var scroller = this;
- var range = correctFloat(scroller.to - scroller.from) *
- pick(scroller.options.step, 0.2);
- scroller.updatePosition(correctFloat(scroller.from - range), correctFloat(scroller.to - range));
- fireEvent(scroller, 'changed', {
- from: scroller.from,
- to: scroller.to,
- trigger: 'scrollbar',
- DOMEvent: e
- });
- };
- /**
- * Get normalized (0-1) cursor position over the scrollbar
- *
- * @private
- * @function Highcharts.Scrollbar#cursorToScrollbarPosition
- *
- * @param {*} normalizedEvent
- * normalized event, with chartX and chartY values
- *
- * @return {Highcharts.Dictionary<number>}
- * Local position {chartX, chartY}
- */
- Scrollbar.prototype.cursorToScrollbarPosition = function (normalizedEvent) {
- var scroller = this,
- options = scroller.options,
- minWidthDifference = options.minWidth > scroller.calculatedWidth ?
- options.minWidth :
- 0; // minWidth distorts translation
- return {
- chartX: (normalizedEvent.chartX - scroller.x -
- scroller.xOffset) /
- (scroller.barWidth - minWidthDifference),
- chartY: (normalizedEvent.chartY - scroller.y -
- scroller.yOffset) /
- (scroller.barWidth - minWidthDifference)
- };
- };
- /**
- * Destroys allocated elements.
- *
- * @private
- * @function Highcharts.Scrollbar#destroy
- * @return {void}
- */
- Scrollbar.prototype.destroy = function () {
- var scroller = this.chart.scroller;
- // Disconnect events added in addEvents
- this.removeEvents();
- // Destroy properties
- [
- 'track',
- 'scrollbarRifles',
- 'scrollbar',
- 'scrollbarGroup',
- 'group'
- ].forEach(function (prop) {
- if (this[prop] && this[prop].destroy) {
- this[prop] = this[prop].destroy();
- }
- }, this);
- // #6421, chart may have more scrollbars
- if (scroller && this === scroller.scrollbar) {
- scroller.scrollbar = null;
- // Destroy elements in collection
- destroyObjectProperties(scroller.scrollbarButtons);
- }
- };
- /**
- * Draw the scrollbar buttons with arrows
- *
- * @private
- * @function Highcharts.Scrollbar#drawScrollbarButton
- * @param {number} index
- * 0 is left, 1 is right
- * @return {void}
- */
- Scrollbar.prototype.drawScrollbarButton = function (index) {
- var scroller = this,
- renderer = scroller.renderer,
- scrollbarButtons = scroller.scrollbarButtons,
- options = scroller.options,
- size = scroller.size,
- group,
- tempElem;
- group = renderer.g().add(scroller.group);
- scrollbarButtons.push(group);
- // Create a rectangle for the scrollbar button
- tempElem = renderer.rect()
- .addClass('highcharts-scrollbar-button')
- .add(group);
- // Presentational attributes
- if (!this.chart.styledMode) {
- tempElem.attr({
- stroke: options.buttonBorderColor,
- 'stroke-width': options.buttonBorderWidth,
- fill: options.buttonBackgroundColor
- });
- }
- // Place the rectangle based on the rendered stroke width
- tempElem.attr(tempElem.crisp({
- x: -0.5,
- y: -0.5,
- width: size + 1,
- height: size + 1,
- r: options.buttonBorderRadius
- }, tempElem.strokeWidth()));
- // Button arrow
- tempElem = renderer
- .path(swapXY([[
- 'M',
- size / 2 + (index ? -1 : 1),
- size / 2 - 3
- ], [
- 'L',
- size / 2 + (index ? -1 : 1),
- size / 2 + 3
- ], [
- 'L',
- size / 2 + (index ? 2 : -2),
- size / 2
- ]], options.vertical))
- .addClass('highcharts-scrollbar-arrow')
- .add(scrollbarButtons[index]);
- if (!this.chart.styledMode) {
- tempElem.attr({
- fill: options.buttonArrowColor
- });
- }
- };
- /**
- * @private
- * @function Highcharts.Scrollbar#init
- * @param {Highcharts.SVGRenderer} renderer
- * @param {Highcharts.ScrollbarOptions} options
- * @param {Highcharts.Chart} chart
- */
- Scrollbar.prototype.init = function (renderer, options, chart) {
- this.scrollbarButtons = [];
- this.renderer = renderer;
- this.userOptions = options;
- this.options = merge(Scrollbar.defaultOptions, options);
- this.chart = chart;
- // backward compatibility
- this.size = pick(this.options.size, this.options.height);
- // Init
- if (options.enabled) {
- this.render();
- this.addEvents();
- }
- };
- Scrollbar.prototype.mouseDownHandler = function (e) {
- var scroller = this;
- var normalizedEvent = scroller.chart.pointer.normalize(e),
- mousePosition = scroller.cursorToScrollbarPosition(normalizedEvent);
- scroller.chartX = mousePosition.chartX;
- scroller.chartY = mousePosition.chartY;
- scroller.initPositions = [scroller.from, scroller.to];
- scroller.grabbedCenter = true;
- };
- /**
- * Event handler for the mouse move event.
- * @private
- */
- Scrollbar.prototype.mouseMoveHandler = function (e) {
- var scroller = this;
- var normalizedEvent = scroller.chart.pointer.normalize(e),
- options = scroller.options,
- direction = options.vertical ? 'chartY' : 'chartX',
- initPositions = scroller.initPositions || [],
- scrollPosition,
- chartPosition,
- change;
- // In iOS, a mousemove event with e.pageX === 0 is fired when
- // holding the finger down in the center of the scrollbar. This
- // should be ignored.
- if (scroller.grabbedCenter &&
- // #4696, scrollbar failed on Android
- (!e.touches || e.touches[0][direction] !== 0)) {
- chartPosition = scroller.cursorToScrollbarPosition(normalizedEvent)[direction];
- scrollPosition = scroller[direction];
- change = chartPosition - scrollPosition;
- scroller.hasDragged = true;
- scroller.updatePosition(initPositions[0] + change, initPositions[1] + change);
- if (scroller.hasDragged) {
- fireEvent(scroller, 'changed', {
- from: scroller.from,
- to: scroller.to,
- trigger: 'scrollbar',
- DOMType: e.type,
- DOMEvent: e
- });
- }
- }
- };
- /**
- * Event handler for the mouse up event.
- * @private
- */
- Scrollbar.prototype.mouseUpHandler = function (e) {
- var scroller = this;
- if (scroller.hasDragged) {
- fireEvent(scroller, 'changed', {
- from: scroller.from,
- to: scroller.to,
- trigger: 'scrollbar',
- DOMType: e.type,
- DOMEvent: e
- });
- }
- scroller.grabbedCenter =
- scroller.hasDragged =
- scroller.chartX =
- scroller.chartY = null;
- };
- /**
- * Position the scrollbar, method called from a parent with defined
- * dimensions.
- *
- * @private
- * @function Highcharts.Scrollbar#position
- * @param {number} x
- * x-position on the chart
- * @param {number} y
- * y-position on the chart
- * @param {number} width
- * width of the scrollbar
- * @param {number} height
- * height of the scorllbar
- * @return {void}
- */
- Scrollbar.prototype.position = function (x, y, width, height) {
- var scroller = this,
- options = scroller.options,
- vertical = options.vertical,
- xOffset = height,
- yOffset = 0,
- method = scroller.rendered ? 'animate' : 'attr';
- scroller.x = x;
- scroller.y = y + this.trackBorderWidth;
- scroller.width = width; // width with buttons
- scroller.height = height;
- scroller.xOffset = xOffset;
- scroller.yOffset = yOffset;
- // If Scrollbar is a vertical type, swap options:
- if (vertical) {
- scroller.width = scroller.yOffset = width = yOffset = scroller.size;
- scroller.xOffset = xOffset = 0;
- scroller.barWidth = height - width * 2; // width without buttons
- scroller.x = x = x + scroller.options.margin;
- }
- else {
- scroller.height = scroller.xOffset = height = xOffset =
- scroller.size;
- scroller.barWidth = width - height * 2; // width without buttons
- scroller.y = scroller.y + scroller.options.margin;
- }
- // Set general position for a group:
- scroller.group[method]({
- translateX: x,
- translateY: scroller.y
- });
- // Resize background/track:
- scroller.track[method]({
- width: width,
- height: height
- });
- // Move right/bottom button ot it's place:
- scroller.scrollbarButtons[1][method]({
- translateX: vertical ? 0 : width - xOffset,
- translateY: vertical ? height - yOffset : 0
- });
- };
- /**
- * Removes the event handlers attached previously with addEvents.
- *
- * @private
- * @function Highcharts.Scrollbar#removeEvents
- * @return {void}
- */
- Scrollbar.prototype.removeEvents = function () {
- this._events.forEach(function (args) {
- removeEvent.apply(null, args);
- });
- this._events.length = 0;
- };
- /**
- * Render scrollbar with all required items.
- *
- * @private
- * @function Highcharts.Scrollbar#render
- */
- Scrollbar.prototype.render = function () {
- var scroller = this,
- renderer = scroller.renderer,
- options = scroller.options,
- size = scroller.size,
- styledMode = this.chart.styledMode,
- group;
- // Draw the scrollbar group
- scroller.group = group = renderer.g('scrollbar').attr({
- zIndex: options.zIndex,
- translateY: -99999
- }).add();
- // Draw the scrollbar track:
- scroller.track = renderer.rect()
- .addClass('highcharts-scrollbar-track')
- .attr({
- x: 0,
- r: options.trackBorderRadius || 0,
- height: size,
- width: size
- }).add(group);
- if (!styledMode) {
- scroller.track.attr({
- fill: options.trackBackgroundColor,
- stroke: options.trackBorderColor,
- 'stroke-width': options.trackBorderWidth
- });
- }
- this.trackBorderWidth = scroller.track.strokeWidth();
- scroller.track.attr({
- y: -this.trackBorderWidth % 2 / 2
- });
- // Draw the scrollbar itself
- scroller.scrollbarGroup = renderer.g().add(group);
- scroller.scrollbar = renderer.rect()
- .addClass('highcharts-scrollbar-thumb')
- .attr({
- height: size,
- width: size,
- r: options.barBorderRadius || 0
- }).add(scroller.scrollbarGroup);
- scroller.scrollbarRifles = renderer
- .path(swapXY([
- ['M', -3, size / 4],
- ['L', -3, 2 * size / 3],
- ['M', 0, size / 4],
- ['L', 0, 2 * size / 3],
- ['M', 3, size / 4],
- ['L', 3, 2 * size / 3]
- ], options.vertical))
- .addClass('highcharts-scrollbar-rifles')
- .add(scroller.scrollbarGroup);
- if (!styledMode) {
- scroller.scrollbar.attr({
- fill: options.barBackgroundColor,
- stroke: options.barBorderColor,
- 'stroke-width': options.barBorderWidth
- });
- scroller.scrollbarRifles.attr({
- stroke: options.rifleColor,
- 'stroke-width': 1
- });
- }
- scroller.scrollbarStrokeWidth = scroller.scrollbar.strokeWidth();
- scroller.scrollbarGroup.translate(-scroller.scrollbarStrokeWidth % 2 / 2, -scroller.scrollbarStrokeWidth % 2 / 2);
- // Draw the buttons:
- scroller.drawScrollbarButton(0);
- scroller.drawScrollbarButton(1);
- };
- /**
- * Set scrollbar size, with a given scale.
- *
- * @private
- * @function Highcharts.Scrollbar#setRange
- * @param {number} from
- * scale (0-1) where bar should start
- * @param {number} to
- * scale (0-1) where bar should end
- * @return {void}
- */
- Scrollbar.prototype.setRange = function (from, to) {
- var scroller = this,
- options = scroller.options,
- vertical = options.vertical,
- minWidth = options.minWidth,
- fullWidth = scroller.barWidth,
- fromPX,
- toPX,
- newPos,
- newSize,
- newRiflesPos,
- method = (this.rendered &&
- !this.hasDragged &&
- !(this.chart.navigator && this.chart.navigator.hasDragged)) ? 'animate' : 'attr';
- if (!defined(fullWidth)) {
- return;
- }
- from = Math.max(from, 0);
- fromPX = Math.ceil(fullWidth * from);
- toPX = fullWidth * Math.min(to, 1);
- scroller.calculatedWidth = newSize = correctFloat(toPX - fromPX);
- // We need to recalculate position, if minWidth is used
- if (newSize < minWidth) {
- fromPX = (fullWidth - minWidth + newSize) * from;
- newSize = minWidth;
- }
- newPos = Math.floor(fromPX + scroller.xOffset + scroller.yOffset);
- newRiflesPos = newSize / 2 - 0.5; // -0.5 -> rifle line width / 2
- // Store current position:
- scroller.from = from;
- scroller.to = to;
- if (!vertical) {
- scroller.scrollbarGroup[method]({
- translateX: newPos
- });
- scroller.scrollbar[method]({
- width: newSize
- });
- scroller.scrollbarRifles[method]({
- translateX: newRiflesPos
- });
- scroller.scrollbarLeft = newPos;
- scroller.scrollbarTop = 0;
- }
- else {
- scroller.scrollbarGroup[method]({
- translateY: newPos
- });
- scroller.scrollbar[method]({
- height: newSize
- });
- scroller.scrollbarRifles[method]({
- translateY: newRiflesPos
- });
- scroller.scrollbarTop = newPos;
- scroller.scrollbarLeft = 0;
- }
- if (newSize <= 12) {
- scroller.scrollbarRifles.hide();
- }
- else {
- scroller.scrollbarRifles.show(true);
- }
- // Show or hide the scrollbar based on the showFull setting
- if (options.showFull === false) {
- if (from <= 0 && to >= 1) {
- scroller.group.hide();
- }
- else {
- scroller.group.show();
- }
- }
- scroller.rendered = true;
- };
- Scrollbar.prototype.trackClick = function (e) {
- var scroller = this;
- var normalizedEvent = scroller.chart.pointer.normalize(e),
- range = scroller.to - scroller.from,
- top = scroller.y + scroller.scrollbarTop,
- left = scroller.x + scroller.scrollbarLeft;
- if ((scroller.options.vertical && normalizedEvent.chartY > top) ||
- (!scroller.options.vertical && normalizedEvent.chartX > left)) {
- // On the top or on the left side of the track:
- scroller.updatePosition(scroller.from + range, scroller.to + range);
- }
- else {
- // On the bottom or the right side of the track:
- scroller.updatePosition(scroller.from - range, scroller.to - range);
- }
- fireEvent(scroller, 'changed', {
- from: scroller.from,
- to: scroller.to,
- trigger: 'scrollbar',
- DOMEvent: e
- });
- };
- /**
- * Update the scrollbar with new options
- *
- * @private
- * @function Highcharts.Scrollbar#update
- * @param {Highcharts.ScrollbarOptions} options
- * @return {void}
- */
- Scrollbar.prototype.update = function (options) {
- this.destroy();
- this.init(this.chart.renderer, merge(true, this.options, options), this.chart);
- };
- /**
- * Update position option in the Scrollbar, with normalized 0-1 scale
- *
- * @private
- * @function Highcharts.Scrollbar#updatePosition
- * @param {number} from
- * @param {number} to
- * @return {void}
- */
- Scrollbar.prototype.updatePosition = function (from, to) {
- if (to > 1) {
- from = correctFloat(1 - correctFloat(to - from));
- to = 1;
- }
- if (from < 0) {
- to = correctFloat(to - from);
- from = 0;
- }
- this.from = from;
- this.to = to;
- };
- /* *
- *
- * Static Properties
- *
- * */
- /**
- *
- * The scrollbar is a means of panning over the X axis of a stock chart.
- * Scrollbars can also be applied to other types of axes.
- *
- * Another approach to scrollable charts is the [chart.scrollablePlotArea](
- * https://api.highcharts.com/highcharts/chart.scrollablePlotArea) option that
- * is especially suitable for simpler cartesian charts on mobile.
- *
- * In styled mode, all the presentational options for the
- * scrollbar are replaced by the classes `.highcharts-scrollbar-thumb`,
- * `.highcharts-scrollbar-arrow`, `.highcharts-scrollbar-button`,
- * `.highcharts-scrollbar-rifles` and `.highcharts-scrollbar-track`.
- *
- * @sample stock/yaxis/inverted-bar-scrollbar/
- * A scrollbar on a simple bar chart
- *
- * @product highstock gantt
- * @optionparent scrollbar
- *
- * @private
- */
- Scrollbar.defaultOptions = {
- /**
- * The height of the scrollbar. The height also applies to the width
- * of the scroll arrows so that they are always squares. Defaults to
- * 20 for touch devices and 14 for mouse devices.
- *
- * @sample stock/scrollbar/height/
- * A 30px scrollbar
- *
- * @type {number}
- * @default 20/14
- */
- height: isTouchDevice ? 20 : 14,
- /**
- * The border rounding radius of the bar.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- */
- barBorderRadius: 0,
- /**
- * The corner radius of the scrollbar buttons.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- */
- buttonBorderRadius: 0,
- /**
- * Enable or disable the scrollbar.
- *
- * @sample stock/scrollbar/enabled/
- * Disable the scrollbar, only use navigator
- *
- * @type {boolean}
- * @default true
- * @apioption scrollbar.enabled
- */
- /**
- * Whether to redraw the main chart as the scrollbar or the navigator
- * zoomed window is moved. Defaults to `true` for modern browsers and
- * `false` for legacy IE browsers as well as mobile devices.
- *
- * @sample stock/scrollbar/liveredraw
- * Setting live redraw to false
- *
- * @type {boolean}
- * @since 1.3
- */
- liveRedraw: void 0,
- /**
- * The margin between the scrollbar and its axis when the scrollbar is
- * applied directly to an axis.
- */
- margin: 10,
- /**
- * The minimum width of the scrollbar.
- *
- * @since 1.2.5
- */
- minWidth: 6,
- /**
- * Whether to show or hide the scrollbar when the scrolled content is
- * zoomed out to it full extent.
- *
- * @type {boolean}
- * @default true
- * @apioption scrollbar.showFull
- */
- step: 0.2,
- /**
- * The z index of the scrollbar group.
- */
- zIndex: 3,
- /**
- * The background color of the scrollbar itself.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- barBackgroundColor: '#cccccc',
- /**
- * The width of the bar's border.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- */
- barBorderWidth: 1,
- /**
- * The color of the scrollbar's border.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- barBorderColor: '#cccccc',
- /**
- * The color of the small arrow inside the scrollbar buttons.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- buttonArrowColor: '#333333',
- /**
- * The color of scrollbar buttons.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- buttonBackgroundColor: '#e6e6e6',
- /**
- * The color of the border of the scrollbar buttons.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- buttonBorderColor: '#cccccc',
- /**
- * The border width of the scrollbar buttons.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- */
- buttonBorderWidth: 1,
- /**
- * The color of the small rifles in the middle of the scrollbar.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- rifleColor: '#333333',
- /**
- * The color of the track background.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- trackBackgroundColor: '#f2f2f2',
- /**
- * The color of the border of the scrollbar track.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- trackBorderColor: '#f2f2f2',
- /**
- * The corner radius of the border of the scrollbar track.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- *
- * @type {number}
- * @default 0
- * @apioption scrollbar.trackBorderRadius
- */
- /**
- * The width of the border of the scrollbar track.
- *
- * @sample stock/scrollbar/style/
- * Scrollbar styling
- */
- trackBorderWidth: 1
- };
- return Scrollbar;
- }());
- if (!H.Scrollbar) {
- defaultOptions.scrollbar = merge(true, Scrollbar.defaultOptions, defaultOptions.scrollbar);
- H.Scrollbar = Scrollbar;
- ScrollbarAxis.compose(Axis, Scrollbar);
- }
- return H.Scrollbar;
- });
- _registerModule(_modules, 'Extensions/RangeSelector.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Globals.js'], _modules['Core/Options.js'], _modules['Core/Renderer/SVG/SVGElement.js'], _modules['Core/Utilities.js']], function (Axis, Chart, H, O, SVGElement, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- createElement = U.createElement,
- css = U.css,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- discardElement = U.discardElement,
- extend = U.extend,
- fireEvent = U.fireEvent,
- isNumber = U.isNumber,
- merge = U.merge,
- objectEach = U.objectEach,
- pick = U.pick,
- pInt = U.pInt,
- splat = U.splat;
- /**
- * Define the time span for the button
- *
- * @typedef {"all"|"day"|"hour"|"millisecond"|"minute"|"month"|"second"|"week"|"year"|"ytd"} Highcharts.RangeSelectorButtonTypeValue
- */
- /**
- * Callback function to react on button clicks.
- *
- * @callback Highcharts.RangeSelectorClickCallbackFunction
- *
- * @param {global.Event} e
- * Event arguments.
- *
- * @param {boolean|undefined}
- * Return false to cancel the default button event.
- */
- /**
- * Callback function to parse values entered in the input boxes and return a
- * valid JavaScript time as milliseconds since 1970.
- *
- * @callback Highcharts.RangeSelectorParseCallbackFunction
- *
- * @param {string} value
- * Input value to parse.
- *
- * @return {number}
- * Parsed JavaScript time value.
- */
- /* ************************************************************************** *
- * Start Range Selector code *
- * ************************************************************************** */
- extend(defaultOptions, {
- /**
- * The range selector is a tool for selecting ranges to display within
- * the chart. It provides buttons to select preconfigured ranges in
- * the chart, like 1 day, 1 week, 1 month etc. It also provides input
- * boxes where min and max dates can be manually input.
- *
- * @product highstock gantt
- * @optionparent rangeSelector
- */
- rangeSelector: {
- /**
- * Whether to enable all buttons from the start. By default buttons are
- * only enabled if the corresponding time range exists on the X axis,
- * but enabling all buttons allows for dynamically loading different
- * time ranges.
- *
- * @sample {highstock} stock/rangeselector/allbuttonsenabled-true/
- * All buttons enabled
- *
- * @type {boolean}
- * @default false
- * @since 2.0.3
- * @apioption rangeSelector.allButtonsEnabled
- */
- /**
- * An array of configuration objects for the buttons.
- *
- * Defaults to:
- * ```js
- * buttons: [{
- * type: 'month',
- * count: 1,
- * text: '1m'
- * }, {
- * type: 'month',
- * count: 3,
- * text: '3m'
- * }, {
- * type: 'month',
- * count: 6,
- * text: '6m'
- * }, {
- * type: 'ytd',
- * text: 'YTD'
- * }, {
- * type: 'year',
- * count: 1,
- * text: '1y'
- * }, {
- * type: 'all',
- * text: 'All'
- * }]
- * ```
- *
- * @sample {highstock} stock/rangeselector/datagrouping/
- * Data grouping by buttons
- *
- * @type {Array<*>}
- * @apioption rangeSelector.buttons
- */
- /**
- * How many units of the defined type the button should span. If `type`
- * is "month" and `count` is 3, the button spans three months.
- *
- * @type {number}
- * @default 1
- * @apioption rangeSelector.buttons.count
- */
- /**
- * Fires when clicking on the rangeSelector button. One parameter,
- * event, is passed to the function, containing common event
- * information.
- *
- * ```js
- * click: function(e) {
- * console.log(this);
- * }
- * ```
- *
- * Return false to stop default button's click action.
- *
- * @sample {highstock} stock/rangeselector/button-click/
- * Click event on the button
- *
- * @type {Highcharts.RangeSelectorClickCallbackFunction}
- * @apioption rangeSelector.buttons.events.click
- */
- /**
- * Additional range (in milliseconds) added to the end of the calculated
- * time span.
- *
- * @sample {highstock} stock/rangeselector/min-max-offsets/
- * Button offsets
- *
- * @type {number}
- * @default 0
- * @since 6.0.0
- * @apioption rangeSelector.buttons.offsetMax
- */
- /**
- * Additional range (in milliseconds) added to the start of the
- * calculated time span.
- *
- * @sample {highstock} stock/rangeselector/min-max-offsets/
- * Button offsets
- *
- * @type {number}
- * @default 0
- * @since 6.0.0
- * @apioption rangeSelector.buttons.offsetMin
- */
- /**
- * When buttons apply dataGrouping on a series, by default zooming
- * in/out will deselect buttons and unset dataGrouping. Enable this
- * option to keep buttons selected when extremes change.
- *
- * @sample {highstock} stock/rangeselector/preserve-datagrouping/
- * Different preserveDataGrouping settings
- *
- * @type {boolean}
- * @default false
- * @since 6.1.2
- * @apioption rangeSelector.buttons.preserveDataGrouping
- */
- /**
- * A custom data grouping object for each button.
- *
- * @see [series.dataGrouping](#plotOptions.series.dataGrouping)
- *
- * @sample {highstock} stock/rangeselector/datagrouping/
- * Data grouping by range selector buttons
- *
- * @type {*}
- * @extends plotOptions.series.dataGrouping
- * @apioption rangeSelector.buttons.dataGrouping
- */
- /**
- * The text for the button itself.
- *
- * @type {string}
- * @apioption rangeSelector.buttons.text
- */
- /**
- * Defined the time span for the button. Can be one of `millisecond`,
- * `second`, `minute`, `hour`, `day`, `week`, `month`, `year`, `ytd`,
- * and `all`.
- *
- * @type {Highcharts.RangeSelectorButtonTypeValue}
- * @apioption rangeSelector.buttons.type
- */
- /**
- * The space in pixels between the buttons in the range selector.
- *
- * @type {number}
- * @default 0
- * @apioption rangeSelector.buttonSpacing
- */
- /**
- * Enable or disable the range selector.
- *
- * @sample {highstock} stock/rangeselector/enabled/
- * Disable the range selector
- *
- * @type {boolean}
- * @default true
- * @apioption rangeSelector.enabled
- */
- /**
- * The vertical alignment of the rangeselector box. Allowed properties
- * are `top`, `middle`, `bottom`.
- *
- * @sample {highstock} stock/rangeselector/vertical-align-middle/
- * Middle
- * @sample {highstock} stock/rangeselector/vertical-align-bottom/
- * Bottom
- *
- * @type {Highcharts.VerticalAlignValue}
- * @since 6.0.0
- */
- verticalAlign: 'top',
- /**
- * A collection of attributes for the buttons. The object takes SVG
- * attributes like `fill`, `stroke`, `stroke-width`, as well as `style`,
- * a collection of CSS properties for the text.
- *
- * The object can also be extended with states, so you can set
- * presentational options for `hover`, `select` or `disabled` button
- * states.
- *
- * CSS styles for the text label.
- *
- * In styled mode, the buttons are styled by the
- * `.highcharts-range-selector-buttons .highcharts-button` rule with its
- * different states.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {Highcharts.SVGAttributes}
- */
- buttonTheme: {
- /** @ignore */
- width: 28,
- /** @ignore */
- height: 18,
- /** @ignore */
- padding: 2,
- /** @ignore */
- zIndex: 7 // #484, #852
- },
- /**
- * When the rangeselector is floating, the plot area does not reserve
- * space for it. This opens for positioning anywhere on the chart.
- *
- * @sample {highstock} stock/rangeselector/floating/
- * Placing the range selector between the plot area and the
- * navigator
- *
- * @since 6.0.0
- */
- floating: false,
- /**
- * The x offset of the range selector relative to its horizontal
- * alignment within `chart.spacingLeft` and `chart.spacingRight`.
- *
- * @since 6.0.0
- */
- x: 0,
- /**
- * The y offset of the range selector relative to its horizontal
- * alignment within `chart.spacingLeft` and `chart.spacingRight`.
- *
- * @since 6.0.0
- */
- y: 0,
- /**
- * Deprecated. The height of the range selector. Currently it is
- * calculated dynamically.
- *
- * @deprecated
- * @type {number|undefined}
- * @since 2.1.9
- */
- height: void 0,
- /**
- * The border color of the date input boxes.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {Highcharts.ColorString}
- * @default #cccccc
- * @since 1.3.7
- * @apioption rangeSelector.inputBoxBorderColor
- */
- /**
- * The pixel height of the date input boxes.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {number}
- * @default 17
- * @since 1.3.7
- * @apioption rangeSelector.inputBoxHeight
- */
- /**
- * CSS for the container DIV holding the input boxes. Deprecated as
- * of 1.2.5\. Use [inputPosition](#rangeSelector.inputPosition) instead.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @deprecated
- * @type {Highcharts.CSSObject}
- * @apioption rangeSelector.inputBoxStyle
- */
- /**
- * The pixel width of the date input boxes.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {number}
- * @default 90
- * @since 1.3.7
- * @apioption rangeSelector.inputBoxWidth
- */
- /**
- * The date format in the input boxes when not selected for editing.
- * Defaults to `%b %e, %Y`.
- *
- * @sample {highstock} stock/rangeselector/input-format/
- * Milliseconds in the range selector
- *
- * @type {string}
- * @default %b %e, %Y
- * @apioption rangeSelector.inputDateFormat
- */
- /**
- * A custom callback function to parse values entered in the input boxes
- * and return a valid JavaScript time as milliseconds since 1970.
- * The first argument passed is a value to parse,
- * second is a boolean indicating use of the UTC time.
- *
- * @sample {highstock} stock/rangeselector/input-format/
- * Milliseconds in the range selector
- *
- * @type {Highcharts.RangeSelectorParseCallbackFunction}
- * @since 1.3.3
- * @apioption rangeSelector.inputDateParser
- */
- /**
- * The date format in the input boxes when they are selected for
- * editing. This must be a format that is recognized by JavaScript
- * Date.parse.
- *
- * @sample {highstock} stock/rangeselector/input-format/
- * Milliseconds in the range selector
- *
- * @type {string}
- * @default %Y-%m-%d
- * @apioption rangeSelector.inputEditDateFormat
- */
- /**
- * Enable or disable the date input boxes. Defaults to enabled when
- * there is enough space, disabled if not (typically mobile).
- *
- * @sample {highstock} stock/rangeselector/input-datepicker/
- * Extending the input with a jQuery UI datepicker
- *
- * @type {boolean}
- * @default true
- * @apioption rangeSelector.inputEnabled
- */
- /**
- * Positioning for the input boxes. Allowed properties are `align`,
- * `x` and `y`.
- *
- * @since 1.2.4
- */
- inputPosition: {
- /**
- * The alignment of the input box. Allowed properties are `left`,
- * `center`, `right`.
- *
- * @sample {highstock} stock/rangeselector/input-button-position/
- * Alignment
- *
- * @type {Highcharts.AlignValue}
- * @since 6.0.0
- */
- align: 'right',
- /**
- * X offset of the input row.
- */
- x: 0,
- /**
- * Y offset of the input row.
- */
- y: 0
- },
- /**
- * The index of the button to appear pre-selected.
- *
- * @type {number}
- * @apioption rangeSelector.selected
- */
- /**
- * Positioning for the button row.
- *
- * @since 1.2.4
- */
- buttonPosition: {
- /**
- * The alignment of the input box. Allowed properties are `left`,
- * `center`, `right`.
- *
- * @sample {highstock} stock/rangeselector/input-button-position/
- * Alignment
- *
- * @type {Highcharts.AlignValue}
- * @since 6.0.0
- */
- align: 'left',
- /**
- * X offset of the button row.
- */
- x: 0,
- /**
- * Y offset of the button row.
- */
- y: 0
- },
- /**
- * CSS for the HTML inputs in the range selector.
- *
- * In styled mode, the inputs are styled by the
- * `.highcharts-range-input text` rule in SVG mode, and
- * `input.highcharts-range-selector` when active.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {Highcharts.CSSObject}
- * @apioption rangeSelector.inputStyle
- */
- /**
- * CSS styles for the labels - the Zoom, From and To texts.
- *
- * In styled mode, the labels are styled by the
- * `.highcharts-range-label` class.
- *
- * @sample {highstock} stock/rangeselector/styling/
- * Styling the buttons and inputs
- *
- * @type {Highcharts.CSSObject}
- */
- labelStyle: {
- /** @ignore */
- color: '#666666'
- }
- }
- });
- defaultOptions.lang = merge(defaultOptions.lang,
- /**
- * Language object. The language object is global and it can't be set
- * on each chart initialization. Instead, use `Highcharts.setOptions` to
- * set it before any chart is initialized.
- *
- * ```js
- * Highcharts.setOptions({
- * lang: {
- * months: [
- * 'Janvier', 'Février', 'Mars', 'Avril',
- * 'Mai', 'Juin', 'Juillet', 'Août',
- * 'Septembre', 'Octobre', 'Novembre', 'Décembre'
- * ],
- * weekdays: [
- * 'Dimanche', 'Lundi', 'Mardi', 'Mercredi',
- * 'Jeudi', 'Vendredi', 'Samedi'
- * ]
- * }
- * });
- * ```
- *
- * @optionparent lang
- */
- {
- /**
- * The text for the label for the range selector buttons.
- *
- * @product highstock gantt
- */
- rangeSelectorZoom: 'Zoom',
- /**
- * The text for the label for the "from" input box in the range
- * selector.
- *
- * @product highstock gantt
- */
- rangeSelectorFrom: 'From',
- /**
- * The text for the label for the "to" input box in the range selector.
- *
- * @product highstock gantt
- */
- rangeSelectorTo: 'To'
- });
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * The range selector.
- *
- * @private
- * @class
- * @name Highcharts.RangeSelector
- * @param {Highcharts.Chart} chart
- */
- var RangeSelector = /** @class */ (function () {
- function RangeSelector(chart) {
- /* *
- *
- * Properties
- *
- * */
- this.buttons = void 0;
- this.buttonOptions = RangeSelector.prototype.defaultButtons;
- this.options = void 0;
- this.chart = chart;
- // Run RangeSelector
- this.init(chart);
- }
- /**
- * The method to run when one of the buttons in the range selectors is
- * clicked
- *
- * @private
- * @function Highcharts.RangeSelector#clickButton
- * @param {number} i
- * The index of the button
- * @param {boolean} [redraw]
- * @return {void}
- */
- RangeSelector.prototype.clickButton = function (i, redraw) {
- var rangeSelector = this,
- chart = rangeSelector.chart,
- rangeOptions = rangeSelector.buttonOptions[i],
- baseAxis = chart.xAxis[0],
- unionExtremes = (chart.scroller && chart.scroller.getUnionExtremes()) || baseAxis || {},
- dataMin = unionExtremes.dataMin,
- dataMax = unionExtremes.dataMax,
- newMin,
- newMax = baseAxis && Math.round(Math.min(baseAxis.max,
- pick(dataMax,
- baseAxis.max))), // #1568
- type = rangeOptions.type,
- baseXAxisOptions,
- range = rangeOptions._range,
- rangeMin,
- minSetting,
- rangeSetting,
- ctx,
- ytdExtremes,
- dataGrouping = rangeOptions.dataGrouping;
- // chart has no data, base series is removed
- if (dataMin === null || dataMax === null) {
- return;
- }
- // Set the fixed range before range is altered
- chart.fixedRange = range;
- // Apply dataGrouping associated to button
- if (dataGrouping) {
- this.forcedDataGrouping = true;
- Axis.prototype.setDataGrouping.call(baseAxis || { chart: this.chart }, dataGrouping, false);
- this.frozenStates = rangeOptions.preserveDataGrouping;
- }
- // Apply range
- if (type === 'month' || type === 'year') {
- if (!baseAxis) {
- // This is set to the user options and picked up later when the
- // axis is instantiated so that we know the min and max.
- range = rangeOptions;
- }
- else {
- ctx = {
- range: rangeOptions,
- max: newMax,
- chart: chart,
- dataMin: dataMin,
- dataMax: dataMax
- };
- newMin = baseAxis.minFromRange.call(ctx);
- if (isNumber(ctx.newMax)) {
- newMax = ctx.newMax;
- }
- }
- // Fixed times like minutes, hours, days
- }
- else if (range) {
- newMin = Math.max(newMax - range, dataMin);
- newMax = Math.min(newMin + range, dataMax);
- }
- else if (type === 'ytd') {
- // On user clicks on the buttons, or a delayed action running from
- // the beforeRender event (below), the baseAxis is defined.
- if (baseAxis) {
- // When "ytd" is the pre-selected button for the initial view,
- // its calculation is delayed and rerun in the beforeRender
- // event (below). When the series are initialized, but before
- // the chart is rendered, we have access to the xData array
- // (#942).
- if (typeof dataMax === 'undefined') {
- dataMin = Number.MAX_VALUE;
- dataMax = Number.MIN_VALUE;
- chart.series.forEach(function (series) {
- // reassign it to the last item
- var xData = series.xData;
- dataMin = Math.min(xData[0], dataMin);
- dataMax = Math.max(xData[xData.length - 1], dataMax);
- });
- redraw = false;
- }
- ytdExtremes = rangeSelector.getYTDExtremes(dataMax, dataMin, chart.time.useUTC);
- newMin = rangeMin = ytdExtremes.min;
- newMax = ytdExtremes.max;
- // "ytd" is pre-selected. We don't yet have access to processed
- // point and extremes data (things like pointStart and pointInterval
- // are missing), so we delay the process (#942)
- }
- else {
- rangeSelector.deferredYTDClick = i;
- return;
- }
- }
- else if (type === 'all' && baseAxis) {
- newMin = dataMin;
- newMax = dataMax;
- }
- if (defined(newMin)) {
- newMin += rangeOptions._offsetMin;
- }
- if (defined(newMax)) {
- newMax += rangeOptions._offsetMax;
- }
- rangeSelector.setSelected(i);
- // Update the chart
- if (!baseAxis) {
- // Axis not yet instanciated. Temporarily set min and range
- // options and remove them on chart load (#4317).
- baseXAxisOptions = splat(chart.options.xAxis)[0];
- rangeSetting = baseXAxisOptions.range;
- baseXAxisOptions.range = range;
- minSetting = baseXAxisOptions.min;
- baseXAxisOptions.min = rangeMin;
- addEvent(chart, 'load', function resetMinAndRange() {
- baseXAxisOptions.range = rangeSetting;
- baseXAxisOptions.min = minSetting;
- });
- }
- else {
- // Existing axis object. Set extremes after render time.
- baseAxis.setExtremes(newMin, newMax, pick(redraw, 1), null, // auto animation
- {
- trigger: 'rangeSelectorButton',
- rangeSelectorButton: rangeOptions
- });
- }
- };
- /**
- * Set the selected option. This method only sets the internal flag, it
- * doesn't update the buttons or the actual zoomed range.
- *
- * @private
- * @function Highcharts.RangeSelector#setSelected
- * @param {number} [selected]
- * @return {void}
- */
- RangeSelector.prototype.setSelected = function (selected) {
- this.selected = this.options.selected = selected;
- };
- /**
- * Initialize the range selector
- *
- * @private
- * @function Highcharts.RangeSelector#init
- * @param {Highcharts.Chart} chart
- * @return {void}
- */
- RangeSelector.prototype.init = function (chart) {
- var rangeSelector = this,
- options = chart.options.rangeSelector,
- buttonOptions = options.buttons || rangeSelector.defaultButtons.slice(),
- selectedOption = options.selected,
- blurInputs = function () {
- var minInput = rangeSelector.minInput,
- maxInput = rangeSelector.maxInput;
- // #3274 in some case blur is not defined
- if (minInput && minInput.blur) {
- fireEvent(minInput, 'blur');
- }
- if (maxInput && maxInput.blur) {
- fireEvent(maxInput, 'blur');
- }
- };
- rangeSelector.chart = chart;
- rangeSelector.options = options;
- rangeSelector.buttons = [];
- rangeSelector.buttonOptions = buttonOptions;
- this.unMouseDown = addEvent(chart.container, 'mousedown', blurInputs);
- this.unResize = addEvent(chart, 'resize', blurInputs);
- // Extend the buttonOptions with actual range
- buttonOptions.forEach(rangeSelector.computeButtonRange);
- // zoomed range based on a pre-selected button index
- if (typeof selectedOption !== 'undefined' &&
- buttonOptions[selectedOption]) {
- this.clickButton(selectedOption, false);
- }
- addEvent(chart, 'load', function () {
- // If a data grouping is applied to the current button, release it
- // when extremes change
- if (chart.xAxis && chart.xAxis[0]) {
- addEvent(chart.xAxis[0], 'setExtremes', function (e) {
- if (this.max - this.min !==
- chart.fixedRange &&
- e.trigger !== 'rangeSelectorButton' &&
- e.trigger !== 'updatedData' &&
- rangeSelector.forcedDataGrouping &&
- !rangeSelector.frozenStates) {
- this.setDataGrouping(false, false);
- }
- });
- }
- });
- };
- /**
- * Dynamically update the range selector buttons after a new range has been
- * set
- *
- * @private
- * @function Highcharts.RangeSelector#updateButtonStates
- * @return {void}
- */
- RangeSelector.prototype.updateButtonStates = function () {
- var rangeSelector = this,
- chart = this.chart,
- baseAxis = chart.xAxis[0],
- actualRange = Math.round(baseAxis.max - baseAxis.min),
- hasNoData = !baseAxis.hasVisibleSeries,
- day = 24 * 36e5, // A single day in milliseconds
- unionExtremes = (chart.scroller &&
- chart.scroller.getUnionExtremes()) || baseAxis,
- dataMin = unionExtremes.dataMin,
- dataMax = unionExtremes.dataMax,
- ytdExtremes = rangeSelector.getYTDExtremes(dataMax,
- dataMin,
- chart.time.useUTC),
- ytdMin = ytdExtremes.min,
- ytdMax = ytdExtremes.max,
- selected = rangeSelector.selected,
- selectedExists = isNumber(selected),
- allButtonsEnabled = rangeSelector.options.allButtonsEnabled,
- buttons = rangeSelector.buttons;
- rangeSelector.buttonOptions.forEach(function (rangeOptions, i) {
- var range = rangeOptions._range,
- type = rangeOptions.type,
- count = rangeOptions.count || 1,
- button = buttons[i],
- state = 0,
- disable,
- select,
- offsetRange = rangeOptions._offsetMax -
- rangeOptions._offsetMin,
- isSelected = i === selected,
- // Disable buttons where the range exceeds what is allowed in
- // the current view
- isTooGreatRange = range >
- dataMax - dataMin,
- // Disable buttons where the range is smaller than the minimum
- // range
- isTooSmallRange = range < baseAxis.minRange,
- // Do not select the YTD button if not explicitly told so
- isYTDButNotSelected = false,
- // Disable the All button if we're already showing all
- isAllButAlreadyShowingAll = false,
- isSameRange = range === actualRange;
- // Months and years have a variable range so we check the extremes
- if ((type === 'month' || type === 'year') &&
- (actualRange + 36e5 >=
- { month: 28, year: 365 }[type] * day * count - offsetRange) &&
- (actualRange - 36e5 <=
- { month: 31, year: 366 }[type] * day * count + offsetRange)) {
- isSameRange = true;
- }
- else if (type === 'ytd') {
- isSameRange = (ytdMax - ytdMin + offsetRange) === actualRange;
- isYTDButNotSelected = !isSelected;
- }
- else if (type === 'all') {
- isSameRange = (baseAxis.max - baseAxis.min >=
- dataMax - dataMin);
- isAllButAlreadyShowingAll = (!isSelected &&
- selectedExists &&
- isSameRange);
- }
- // The new zoom area happens to match the range for a button - mark
- // it selected. This happens when scrolling across an ordinal gap.
- // It can be seen in the intraday demos when selecting 1h and scroll
- // across the night gap.
- disable = (!allButtonsEnabled &&
- (isTooGreatRange ||
- isTooSmallRange ||
- isAllButAlreadyShowingAll ||
- hasNoData));
- select = ((isSelected && isSameRange) ||
- (isSameRange && !selectedExists && !isYTDButNotSelected) ||
- (isSelected && rangeSelector.frozenStates));
- if (disable) {
- state = 3;
- }
- else if (select) {
- selectedExists = true; // Only one button can be selected
- state = 2;
- }
- // If state has changed, update the button
- if (button.state !== state) {
- button.setState(state);
- // Reset (#9209)
- if (state === 0 && selected === i) {
- rangeSelector.setSelected(null);
- }
- }
- });
- };
- /**
- * Compute and cache the range for an individual button
- *
- * @private
- * @function Highcharts.RangeSelector#computeButtonRange
- * @param {Highcharts.RangeSelectorButtonsOptions} rangeOptions
- * @return {void}
- */
- RangeSelector.prototype.computeButtonRange = function (rangeOptions) {
- var type = rangeOptions.type,
- count = rangeOptions.count || 1,
- // these time intervals have a fixed number of milliseconds, as
- // opposed to month, ytd and year
- fixedTimes = {
- millisecond: 1,
- second: 1000,
- minute: 60 * 1000,
- hour: 3600 * 1000,
- day: 24 * 3600 * 1000,
- week: 7 * 24 * 3600 * 1000
- };
- // Store the range on the button object
- if (fixedTimes[type]) {
- rangeOptions._range = fixedTimes[type] * count;
- }
- else if (type === 'month' || type === 'year') {
- rangeOptions._range = {
- month: 30,
- year: 365
- }[type] * 24 * 36e5 * count;
- }
- rangeOptions._offsetMin = pick(rangeOptions.offsetMin, 0);
- rangeOptions._offsetMax = pick(rangeOptions.offsetMax, 0);
- rangeOptions._range +=
- rangeOptions._offsetMax - rangeOptions._offsetMin;
- };
- /**
- * Set the internal and displayed value of a HTML input for the dates
- *
- * @private
- * @function Highcharts.RangeSelector#setInputValue
- * @param {string} name
- * @param {number} [inputTime]
- * @return {void}
- */
- RangeSelector.prototype.setInputValue = function (name, inputTime) {
- var options = this.chart.options.rangeSelector,
- time = this.chart.time,
- input = this[name + 'Input'];
- if (defined(inputTime)) {
- input.previousValue = input.HCTime;
- input.HCTime = inputTime;
- }
- input.value = time.dateFormat(options.inputEditDateFormat || '%Y-%m-%d', input.HCTime);
- this[name + 'DateBox'].attr({
- text: time.dateFormat(options.inputDateFormat || '%b %e, %Y', input.HCTime)
- });
- };
- /**
- * @private
- * @function Highcharts.RangeSelector#showInput
- * @param {string} name
- * @return {void}
- */
- RangeSelector.prototype.showInput = function (name) {
- var inputGroup = this.inputGroup,
- dateBox = this[name + 'DateBox'];
- css(this[name + 'Input'], {
- left: (inputGroup.translateX + dateBox.x) + 'px',
- top: inputGroup.translateY + 'px',
- width: (dateBox.width - 2) + 'px',
- height: (dateBox.height - 2) + 'px',
- border: '2px solid silver'
- });
- };
- /**
- * @private
- * @function Highcharts.RangeSelector#hideInput
- * @param {string} name
- * @return {void}
- */
- RangeSelector.prototype.hideInput = function (name) {
- css(this[name + 'Input'], {
- border: 0,
- width: '1px',
- height: '1px'
- });
- this.setInputValue(name);
- };
- /**
- * @private
- * @function Highcharts.RangeSelector#defaultInputDateParser
- */
- RangeSelector.prototype.defaultInputDateParser = function (inputDate, useUTC) {
- var date = new Date();
- if (H.isSafari) {
- return Date.parse(inputDate.split(' ').join('T'));
- }
- if (useUTC) {
- return Date.parse(inputDate + 'Z');
- }
- return Date.parse(inputDate) - date.getTimezoneOffset() * 60 * 1000;
- };
- /**
- * Draw either the 'from' or the 'to' HTML input box of the range selector
- *
- * @private
- * @function Highcharts.RangeSelector#drawInput
- * @param {string} name
- * @return {void}
- */
- RangeSelector.prototype.drawInput = function (name) {
- var rangeSelector = this,
- chart = rangeSelector.chart,
- chartStyle = chart.renderer.style || {},
- renderer = chart.renderer,
- options = chart.options.rangeSelector,
- lang = defaultOptions.lang,
- div = rangeSelector.div,
- isMin = name === 'min',
- input,
- label,
- dateBox,
- inputGroup = this.inputGroup,
- defaultInputDateParser = this.defaultInputDateParser;
- /**
- * @private
- */
- function updateExtremes() {
- var inputValue = input.value,
- value,
- chartAxis = chart.xAxis[0],
- dataAxis = chart.scroller && chart.scroller.xAxis ?
- chart.scroller.xAxis :
- chartAxis,
- dataMin = dataAxis.dataMin,
- dataMax = dataAxis.dataMax;
- value = (options.inputDateParser || defaultInputDateParser)(inputValue, chart.time.useUTC);
- if (value !== input.previousValue) {
- input.previousValue = value;
- // If the value isn't parsed directly to a value by the
- // browser's Date.parse method, like YYYY-MM-DD in IE, try
- // parsing it a different way
- if (!isNumber(value)) {
- value = inputValue.split('-');
- value = Date.UTC(pInt(value[0]), pInt(value[1]) - 1, pInt(value[2]));
- }
- if (isNumber(value)) {
- // Correct for timezone offset (#433)
- if (!chart.time.useUTC) {
- value =
- value + new Date().getTimezoneOffset() * 60 * 1000;
- }
- // Validate the extremes. If it goes beyound the data min or
- // max, use the actual data extreme (#2438).
- if (isMin) {
- if (value > rangeSelector.maxInput.HCTime) {
- value = void 0;
- }
- else if (value < dataMin) {
- value = dataMin;
- }
- }
- else {
- if (value < rangeSelector.minInput.HCTime) {
- value = void 0;
- }
- else if (value > dataMax) {
- value = dataMax;
- }
- }
- // Set the extremes
- if (typeof value !== 'undefined') { // @todo typof undefined
- chartAxis.setExtremes(isMin ? value : chartAxis.min, isMin ? chartAxis.max : value, void 0, void 0, { trigger: 'rangeSelectorInput' });
- }
- }
- }
- }
- // Create the text label
- this[name + 'Label'] = label = renderer
- .label(lang[isMin ? 'rangeSelectorFrom' : 'rangeSelectorTo'], this.inputGroup.offset)
- .addClass('highcharts-range-label')
- .attr({
- padding: 2
- })
- .add(inputGroup);
- inputGroup.offset += label.width + 5;
- // Create an SVG label that shows updated date ranges and and records
- // click events that bring in the HTML input.
- this[name + 'DateBox'] = dateBox = renderer
- .label('', inputGroup.offset)
- .addClass('highcharts-range-input')
- .attr({
- padding: 2,
- width: options.inputBoxWidth || 90,
- height: options.inputBoxHeight || 17,
- 'text-align': 'center'
- })
- .on('click', function () {
- // If it is already focused, the onfocus event doesn't fire
- // (#3713)
- rangeSelector.showInput(name);
- rangeSelector[name + 'Input'].focus();
- });
- if (!chart.styledMode) {
- dateBox.attr({
- stroke: options.inputBoxBorderColor || '#cccccc',
- 'stroke-width': 1
- });
- }
- dateBox.add(inputGroup);
- inputGroup.offset += dateBox.width + (isMin ? 10 : 0);
- // Create the HTML input element. This is rendered as 1x1 pixel then set
- // to the right size when focused.
- this[name + 'Input'] = input = createElement('input', {
- name: name,
- className: 'highcharts-range-selector',
- type: 'text'
- }, {
- top: chart.plotTop + 'px' // prevent jump on focus in Firefox
- }, div);
- if (!chart.styledMode) {
- // Styles
- label.css(merge(chartStyle, options.labelStyle));
- dateBox.css(merge({
- color: '#333333'
- }, chartStyle, options.inputStyle));
- css(input, extend({
- position: 'absolute',
- border: 0,
- width: '1px',
- height: '1px',
- padding: 0,
- textAlign: 'center',
- fontSize: chartStyle.fontSize,
- fontFamily: chartStyle.fontFamily,
- top: '-9999em' // #4798
- }, options.inputStyle));
- }
- // Blow up the input box
- input.onfocus = function () {
- rangeSelector.showInput(name);
- };
- // Hide away the input box
- input.onblur = function () {
- // update extermes only when inputs are active
- if (input === H.doc.activeElement) { // Only when focused
- // Update also when no `change` event is triggered, like when
- // clicking inside the SVG (#4710)
- updateExtremes();
- }
- // #10404 - move hide and blur outside focus
- rangeSelector.hideInput(name);
- input.blur(); // #4606
- };
- // handle changes in the input boxes
- input.onchange = updateExtremes;
- input.onkeypress = function (event) {
- // IE does not fire onchange on enter
- if (event.keyCode === 13) {
- updateExtremes();
- }
- };
- };
- /**
- * Get the position of the range selector buttons and inputs. This can be
- * overridden from outside for custom positioning.
- *
- * @private
- * @function Highcharts.RangeSelector#getPosition
- *
- * @return {Highcharts.Dictionary<number>}
- */
- RangeSelector.prototype.getPosition = function () {
- var chart = this.chart,
- options = chart.options.rangeSelector,
- top = options.verticalAlign === 'top' ?
- chart.plotTop - chart.axisOffset[0] :
- 0; // set offset only for varticalAlign top
- return {
- buttonTop: top + options.buttonPosition.y,
- inputTop: top + options.inputPosition.y - 10
- };
- };
- /**
- * Get the extremes of YTD. Will choose dataMax if its value is lower than
- * the current timestamp. Will choose dataMin if its value is higher than
- * the timestamp for the start of current year.
- *
- * @private
- * @function Highcharts.RangeSelector#getYTDExtremes
- *
- * @param {number} dataMax
- *
- * @param {number} dataMin
- *
- * @return {*}
- * Returns min and max for the YTD
- */
- RangeSelector.prototype.getYTDExtremes = function (dataMax, dataMin, useUTC) {
- var time = this.chart.time,
- min,
- now = new time.Date(dataMax),
- year = time.get('FullYear',
- now),
- startOfYear = useUTC ?
- time.Date.UTC(year, 0, 1) : // eslint-disable-line new-cap
- +new time.Date(year, 0, 1);
- min = Math.max(dataMin || 0, startOfYear);
- now = now.getTime();
- return {
- max: Math.min(dataMax || now, now),
- min: min
- };
- };
- /**
- * Render the range selector including the buttons and the inputs. The first
- * time render is called, the elements are created and positioned. On
- * subsequent calls, they are moved and updated.
- *
- * @private
- * @function Highcharts.RangeSelector#render
- * @param {number} [min]
- * X axis minimum
- * @param {number} [max]
- * X axis maximum
- * @return {void}
- */
- RangeSelector.prototype.render = function (min, max) {
- var rangeSelector = this,
- chart = rangeSelector.chart,
- renderer = chart.renderer,
- container = chart.container,
- chartOptions = chart.options,
- navButtonOptions = (chartOptions.exporting &&
- chartOptions.exporting.enabled !== false &&
- chartOptions.navigation &&
- chartOptions.navigation.buttonOptions),
- lang = defaultOptions.lang,
- div = rangeSelector.div,
- options = chartOptions.rangeSelector,
- // Place inputs above the container
- inputsZIndex = pick(chartOptions.chart.style &&
- chartOptions.chart.style.zIndex, 0) + 1,
- floating = options.floating,
- buttons = rangeSelector.buttons,
- inputGroup = rangeSelector.inputGroup,
- buttonTheme = options.buttonTheme,
- buttonPosition = options.buttonPosition,
- inputPosition = options.inputPosition,
- inputEnabled = options.inputEnabled,
- states = buttonTheme && buttonTheme.states,
- plotLeft = chart.plotLeft,
- buttonLeft,
- buttonGroup = rangeSelector.buttonGroup,
- group,
- groupHeight,
- rendered = rangeSelector.rendered,
- verticalAlign = rangeSelector.options.verticalAlign,
- legend = chart.legend,
- legendOptions = legend && legend.options,
- buttonPositionY = buttonPosition.y,
- inputPositionY = inputPosition.y,
- animate = chart.hasLoaded,
- verb = animate ? 'animate' : 'attr',
- exportingX = 0,
- alignTranslateY,
- legendHeight,
- minPosition,
- translateY = 0,
- translateX;
- if (options.enabled === false) {
- return;
- }
- // create the elements
- if (!rendered) {
- rangeSelector.group = group = renderer.g('range-selector-group')
- .attr({
- zIndex: 7
- })
- .add();
- rangeSelector.buttonGroup = buttonGroup =
- renderer.g('range-selector-buttons').add(group);
- rangeSelector.zoomText = renderer
- .text(lang.rangeSelectorZoom, 0, 15)
- .add(buttonGroup);
- if (!chart.styledMode) {
- rangeSelector.zoomText.css(options.labelStyle);
- buttonTheme['stroke-width'] =
- pick(buttonTheme['stroke-width'], 0);
- }
- rangeSelector.buttonOptions.forEach(function (rangeOptions, i) {
- buttons[i] = renderer
- .button(rangeOptions.text, 0, 0, function (e) {
- // extract events from button object and call
- var buttonEvents = (rangeOptions.events &&
- rangeOptions.events.click),
- callDefaultEvent;
- if (buttonEvents) {
- callDefaultEvent =
- buttonEvents.call(rangeOptions, e);
- }
- if (callDefaultEvent !== false) {
- rangeSelector.clickButton(i);
- }
- rangeSelector.isActive = true;
- }, buttonTheme, states && states.hover, states && states.select, states && states.disabled)
- .attr({
- 'text-align': 'center'
- })
- .add(buttonGroup);
- });
- // first create a wrapper outside the container in order to make
- // the inputs work and make export correct
- if (inputEnabled !== false) {
- rangeSelector.div = div = createElement('div', null, {
- position: 'relative',
- height: 0,
- zIndex: inputsZIndex
- });
- container.parentNode.insertBefore(div, container);
- // Create the group to keep the inputs
- rangeSelector.inputGroup = inputGroup =
- renderer.g('input-group').add(group);
- inputGroup.offset = 0;
- rangeSelector.drawInput('min');
- rangeSelector.drawInput('max');
- }
- }
- // #8769, allow dynamically updating margins
- rangeSelector.zoomText[verb]({
- x: pick(plotLeft + buttonPosition.x, plotLeft)
- });
- // button start position
- buttonLeft = pick(plotLeft + buttonPosition.x, plotLeft) +
- rangeSelector.zoomText.getBBox().width + 5;
- rangeSelector.buttonOptions.forEach(function (rangeOptions, i) {
- buttons[i][verb]({ x: buttonLeft });
- // increase button position for the next button
- buttonLeft += buttons[i].width + pick(options.buttonSpacing, 5);
- });
- plotLeft = chart.plotLeft - chart.spacing[3];
- rangeSelector.updateButtonStates();
- // detect collisiton with exporting
- if (navButtonOptions &&
- this.titleCollision(chart) &&
- verticalAlign === 'top' &&
- buttonPosition.align === 'right' && ((buttonPosition.y +
- buttonGroup.getBBox().height - 12) <
- ((navButtonOptions.y || 0) +
- navButtonOptions.height))) {
- exportingX = -40;
- }
- translateX = buttonPosition.x - chart.spacing[3];
- if (buttonPosition.align === 'right') {
- translateX += exportingX - plotLeft; // (#13014)
- }
- else if (buttonPosition.align === 'center') {
- translateX -= plotLeft / 2;
- }
- // align button group
- buttonGroup.align({
- y: buttonPosition.y,
- width: buttonGroup.getBBox().width,
- align: buttonPosition.align,
- x: translateX
- }, true, chart.spacingBox);
- // skip animation
- rangeSelector.group.placed = animate;
- rangeSelector.buttonGroup.placed = animate;
- if (inputEnabled !== false) {
- var inputGroupX,
- inputGroupWidth,
- buttonGroupX,
- buttonGroupWidth;
- // detect collision with exporting
- if (navButtonOptions &&
- this.titleCollision(chart) &&
- verticalAlign === 'top' &&
- inputPosition.align === 'right' && ((inputPosition.y -
- inputGroup.getBBox().height - 12) <
- ((navButtonOptions.y || 0) +
- navButtonOptions.height +
- chart.spacing[0]))) {
- exportingX = -40;
- }
- else {
- exportingX = 0;
- }
- if (inputPosition.align === 'left') {
- translateX = plotLeft;
- }
- else if (inputPosition.align === 'right') {
- translateX = -Math.max(chart.axisOffset[1], -exportingX);
- }
- // Update the alignment to the updated spacing box
- inputGroup.align({
- y: inputPosition.y,
- width: inputGroup.getBBox().width,
- align: inputPosition.align,
- // fix wrong getBBox() value on right align
- x: inputPosition.x + translateX - 2
- }, true, chart.spacingBox);
- // detect collision
- inputGroupX = (inputGroup.alignAttr.translateX +
- inputGroup.alignOptions.x -
- exportingX +
- // getBBox for detecing left margin
- inputGroup.getBBox().x +
- // 2px padding to not overlap input and label
- 2);
- inputGroupWidth = inputGroup.alignOptions.width;
- buttonGroupX = buttonGroup.alignAttr.translateX +
- buttonGroup.getBBox().x;
- // 20 is minimal spacing between elements
- buttonGroupWidth = buttonGroup.getBBox().width + 20;
- if ((inputPosition.align ===
- buttonPosition.align) || ((buttonGroupX + buttonGroupWidth > inputGroupX) &&
- (inputGroupX + inputGroupWidth > buttonGroupX) &&
- (buttonPositionY <
- (inputPositionY +
- inputGroup.getBBox().height)))) {
- inputGroup.attr({
- translateX: inputGroup.alignAttr.translateX +
- (chart.axisOffset[1] >= -exportingX ? 0 : -exportingX),
- translateY: inputGroup.alignAttr.translateY +
- buttonGroup.getBBox().height + 10
- });
- }
- // Set or reset the input values
- rangeSelector.setInputValue('min', min);
- rangeSelector.setInputValue('max', max);
- // skip animation
- rangeSelector.inputGroup.placed = animate;
- }
- // vertical align
- rangeSelector.group.align({
- verticalAlign: verticalAlign
- }, true, chart.spacingBox);
- // set position
- groupHeight =
- rangeSelector.group.getBBox().height + 20; // # 20 padding
- alignTranslateY =
- rangeSelector.group.alignAttr.translateY;
- // calculate bottom position
- if (verticalAlign === 'bottom') {
- legendHeight = (legendOptions &&
- legendOptions.verticalAlign === 'bottom' &&
- legendOptions.enabled &&
- !legendOptions.floating ?
- legend.legendHeight + pick(legendOptions.margin, 10) :
- 0);
- groupHeight = groupHeight + legendHeight - 20;
- translateY = (alignTranslateY -
- groupHeight -
- (floating ? 0 : options.y) -
- (chart.titleOffset ? chart.titleOffset[2] : 0) -
- 10 // 10 spacing
- );
- }
- if (verticalAlign === 'top') {
- if (floating) {
- translateY = 0;
- }
- if (chart.titleOffset && chart.titleOffset[0]) {
- translateY = chart.titleOffset[0];
- }
- translateY += ((chart.margin[0] - chart.spacing[0]) || 0);
- }
- else if (verticalAlign === 'middle') {
- if (inputPositionY === buttonPositionY) {
- if (inputPositionY < 0) {
- translateY = alignTranslateY + minPosition;
- }
- else {
- translateY = alignTranslateY;
- }
- }
- else if (inputPositionY || buttonPositionY) {
- if (inputPositionY < 0 ||
- buttonPositionY < 0) {
- translateY -= Math.min(inputPositionY, buttonPositionY);
- }
- else {
- translateY =
- alignTranslateY - groupHeight + minPosition;
- }
- }
- }
- rangeSelector.group.translate(options.x, options.y + Math.floor(translateY));
- // translate HTML inputs
- if (inputEnabled !== false) {
- rangeSelector.minInput.style.marginTop =
- rangeSelector.group.translateY + 'px';
- rangeSelector.maxInput.style.marginTop =
- rangeSelector.group.translateY + 'px';
- }
- rangeSelector.rendered = true;
- };
- /**
- * Extracts height of range selector
- *
- * @private
- * @function Highcharts.RangeSelector#getHeight
- * @return {number}
- * Returns rangeSelector height
- */
- RangeSelector.prototype.getHeight = function () {
- var rangeSelector = this,
- options = rangeSelector.options,
- rangeSelectorGroup = rangeSelector.group,
- inputPosition = options.inputPosition,
- buttonPosition = options.buttonPosition,
- yPosition = options.y,
- buttonPositionY = buttonPosition.y,
- inputPositionY = inputPosition.y,
- rangeSelectorHeight = 0,
- minPosition;
- if (options.height) {
- return options.height;
- }
- rangeSelectorHeight = rangeSelectorGroup ?
- // 13px to keep back compatibility
- (rangeSelectorGroup.getBBox(true).height) + 13 +
- yPosition :
- 0;
- minPosition = Math.min(inputPositionY, buttonPositionY);
- if ((inputPositionY < 0 && buttonPositionY < 0) ||
- (inputPositionY > 0 && buttonPositionY > 0)) {
- rangeSelectorHeight += Math.abs(minPosition);
- }
- return rangeSelectorHeight;
- };
- /**
- * Detect collision with title or subtitle
- *
- * @private
- * @function Highcharts.RangeSelector#titleCollision
- *
- * @param {Highcharts.Chart} chart
- *
- * @return {boolean}
- * Returns collision status
- */
- RangeSelector.prototype.titleCollision = function (chart) {
- return !(chart.options.title.text ||
- chart.options.subtitle.text);
- };
- /**
- * Update the range selector with new options
- *
- * @private
- * @function Highcharts.RangeSelector#update
- * @param {Highcharts.RangeSelectorOptions} options
- * @return {void}
- */
- RangeSelector.prototype.update = function (options) {
- var chart = this.chart;
- merge(true, chart.options.rangeSelector, options);
- this.destroy();
- this.init(chart);
- chart.rangeSelector.render();
- };
- /**
- * Destroys allocated elements.
- *
- * @private
- * @function Highcharts.RangeSelector#destroy
- */
- RangeSelector.prototype.destroy = function () {
- var rSelector = this,
- minInput = rSelector.minInput,
- maxInput = rSelector.maxInput;
- rSelector.unMouseDown();
- rSelector.unResize();
- // Destroy elements in collections
- destroyObjectProperties(rSelector.buttons);
- // Clear input element events
- if (minInput) {
- minInput.onfocus = minInput.onblur = minInput.onchange = null;
- }
- if (maxInput) {
- maxInput.onfocus = maxInput.onblur = maxInput.onchange = null;
- }
- // Destroy HTML and SVG elements
- objectEach(rSelector, function (val, key) {
- if (val && key !== 'chart') {
- if (val instanceof SVGElement) {
- // SVGElement
- val.destroy();
- }
- else if (val instanceof window.HTMLElement) {
- // HTML element
- discardElement(val);
- }
- }
- if (val !== RangeSelector.prototype[key]) {
- rSelector[key] = null;
- }
- }, this);
- };
- return RangeSelector;
- }());
- /**
- * The default buttons for pre-selecting time frames
- */
- RangeSelector.prototype.defaultButtons = [{
- type: 'month',
- count: 1,
- text: '1m'
- }, {
- type: 'month',
- count: 3,
- text: '3m'
- }, {
- type: 'month',
- count: 6,
- text: '6m'
- }, {
- type: 'ytd',
- text: 'YTD'
- }, {
- type: 'year',
- count: 1,
- text: '1y'
- }, {
- type: 'all',
- text: 'All'
- }];
- /**
- * Get the axis min value based on the range option and the current max. For
- * stock charts this is extended via the {@link RangeSelector} so that if the
- * selected range is a multiple of months or years, it is compensated for
- * various month lengths.
- *
- * @private
- * @function Highcharts.Axis#minFromRange
- * @return {number|undefined}
- * The new minimum value.
- */
- Axis.prototype.minFromRange = function () {
- var rangeOptions = this.range,
- type = rangeOptions.type,
- min,
- max = this.max,
- dataMin,
- range,
- time = this.chart.time,
- // Get the true range from a start date
- getTrueRange = function (base,
- count) {
- var timeName = type === 'year' ? 'FullYear' : 'Month';
- var date = new time.Date(base);
- var basePeriod = time.get(timeName,
- date);
- time.set(timeName, date, basePeriod + count);
- if (basePeriod === time.get(timeName, date)) {
- time.set('Date', date, 0); // #6537
- }
- return date.getTime() - base;
- };
- if (isNumber(rangeOptions)) {
- min = max - rangeOptions;
- range = rangeOptions;
- }
- else {
- min = max + getTrueRange(max, -rangeOptions.count);
- // Let the fixedRange reflect initial settings (#5930)
- if (this.chart) {
- this.chart.fixedRange = max - min;
- }
- }
- dataMin = pick(this.dataMin, Number.MIN_VALUE);
- if (!isNumber(min)) {
- min = dataMin;
- }
- if (min <= dataMin) {
- min = dataMin;
- if (typeof range === 'undefined') { // #4501
- range = getTrueRange(min, rangeOptions.count);
- }
- this.newMax = Math.min(min + range, this.dataMax);
- }
- if (!isNumber(max)) {
- min = void 0;
- }
- return min;
- };
- if (!H.RangeSelector) {
- // Initialize rangeselector for stock charts
- addEvent(Chart, 'afterGetContainer', function () {
- if (this.options.rangeSelector.enabled) {
- this.rangeSelector = new RangeSelector(this);
- }
- });
- addEvent(Chart, 'beforeRender', function () {
- var chart = this,
- axes = chart.axes,
- rangeSelector = chart.rangeSelector,
- verticalAlign;
- if (rangeSelector) {
- if (isNumber(rangeSelector.deferredYTDClick)) {
- rangeSelector.clickButton(rangeSelector.deferredYTDClick);
- delete rangeSelector.deferredYTDClick;
- }
- axes.forEach(function (axis) {
- axis.updateNames();
- axis.setScale();
- });
- chart.getAxisMargins();
- rangeSelector.render();
- verticalAlign = rangeSelector.options.verticalAlign;
- if (!rangeSelector.options.floating) {
- if (verticalAlign === 'bottom') {
- this.extraBottomMargin = true;
- }
- else if (verticalAlign !== 'middle') {
- this.extraTopMargin = true;
- }
- }
- }
- });
- addEvent(Chart, 'update', function (e) {
- var chart = this,
- options = e.options,
- optionsRangeSelector = options.rangeSelector,
- rangeSelector = chart.rangeSelector,
- verticalAlign,
- extraBottomMarginWas = this.extraBottomMargin,
- extraTopMarginWas = this.extraTopMargin;
- if (optionsRangeSelector &&
- optionsRangeSelector.enabled &&
- !defined(rangeSelector)) {
- this.options.rangeSelector.enabled = true;
- this.rangeSelector = new RangeSelector(this);
- }
- this.extraBottomMargin = false;
- this.extraTopMargin = false;
- if (rangeSelector) {
- rangeSelector.render();
- verticalAlign = (optionsRangeSelector &&
- optionsRangeSelector.verticalAlign) || (rangeSelector.options && rangeSelector.options.verticalAlign);
- if (!rangeSelector.options.floating) {
- if (verticalAlign === 'bottom') {
- this.extraBottomMargin = true;
- }
- else if (verticalAlign !== 'middle') {
- this.extraTopMargin = true;
- }
- }
- if (this.extraBottomMargin !== extraBottomMarginWas ||
- this.extraTopMargin !== extraTopMarginWas) {
- this.isDirtyBox = true;
- }
- }
- });
- addEvent(Chart, 'render', function () {
- var chart = this,
- rangeSelector = chart.rangeSelector,
- verticalAlign;
- if (rangeSelector && !rangeSelector.options.floating) {
- rangeSelector.render();
- verticalAlign = rangeSelector.options.verticalAlign;
- if (verticalAlign === 'bottom') {
- this.extraBottomMargin = true;
- }
- else if (verticalAlign !== 'middle') {
- this.extraTopMargin = true;
- }
- }
- });
- addEvent(Chart, 'getMargins', function () {
- var rangeSelector = this.rangeSelector,
- rangeSelectorHeight;
- if (rangeSelector) {
- rangeSelectorHeight = rangeSelector.getHeight();
- if (this.extraTopMargin) {
- this.plotTop += rangeSelectorHeight;
- }
- if (this.extraBottomMargin) {
- this.marginBottom += rangeSelectorHeight;
- }
- }
- });
- Chart.prototype.callbacks.push(function (chart) {
- var extremes,
- rangeSelector = chart.rangeSelector,
- unbindRender,
- unbindSetExtremes,
- legend,
- alignTo,
- verticalAlign;
- /**
- * @private
- */
- function renderRangeSelector() {
- extremes = chart.xAxis[0].getExtremes();
- legend = chart.legend;
- verticalAlign = rangeSelector === null || rangeSelector === void 0 ? void 0 : rangeSelector.options.verticalAlign;
- if (isNumber(extremes.min)) {
- rangeSelector.render(extremes.min, extremes.max);
- }
- // Re-align the legend so that it's below the rangeselector
- if (rangeSelector && legend.display &&
- verticalAlign === 'top' &&
- verticalAlign === legend.options.verticalAlign) {
- // Create a new alignment box for the legend.
- alignTo = merge(chart.spacingBox);
- if (legend.options.layout === 'vertical') {
- alignTo.y = chart.plotTop;
- }
- else {
- alignTo.y += rangeSelector.getHeight();
- }
- legend.group.placed = false; // Don't animate the alignment.
- legend.align(alignTo);
- }
- }
- if (rangeSelector) {
- // redraw the scroller on setExtremes
- unbindSetExtremes = addEvent(chart.xAxis[0], 'afterSetExtremes', function (e) {
- rangeSelector.render(e.min, e.max);
- });
- // redraw the scroller chart resize
- unbindRender = addEvent(chart, 'redraw', renderRangeSelector);
- // do it now
- renderRangeSelector();
- }
- // Remove resize/afterSetExtremes at chart destroy
- addEvent(chart, 'destroy', function destroyEvents() {
- if (rangeSelector) {
- unbindRender();
- unbindSetExtremes();
- }
- });
- });
- H.RangeSelector = RangeSelector;
- }
- return H.RangeSelector;
- });
- _registerModule(_modules, 'Core/Axis/NavigatorAxis.js', [_modules['Core/Globals.js'], _modules['Core/Utilities.js']], function (H, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var isTouchDevice = H.isTouchDevice;
- var addEvent = U.addEvent,
- correctFloat = U.correctFloat,
- defined = U.defined,
- isNumber = U.isNumber,
- pick = U.pick;
- /* eslint-disable valid-jsdoc */
- /**
- * @private
- * @class
- */
- var NavigatorAxisAdditions = /** @class */ (function () {
- /* *
- *
- * Constructors
- *
- * */
- function NavigatorAxisAdditions(axis) {
- this.axis = axis;
- }
- /* *
- *
- * Functions
- *
- * */
- /**
- * @private
- */
- NavigatorAxisAdditions.prototype.destroy = function () {
- this.axis = void 0;
- };
- /**
- * Add logic to normalize the zoomed range in order to preserve the pressed
- * state of range selector buttons
- *
- * @private
- * @function Highcharts.Axis#toFixedRange
- * @param {number} [pxMin]
- * @param {number} [pxMax]
- * @param {number} [fixedMin]
- * @param {number} [fixedMax]
- * @return {*}
- */
- NavigatorAxisAdditions.prototype.toFixedRange = function (pxMin, pxMax, fixedMin, fixedMax) {
- var navigator = this;
- var axis = navigator.axis;
- var chart = axis.chart;
- var fixedRange = chart && chart.fixedRange,
- halfPointRange = (axis.pointRange || 0) / 2,
- newMin = pick(fixedMin,
- axis.translate(pxMin,
- true, !axis.horiz)),
- newMax = pick(fixedMax,
- axis.translate(pxMax,
- true, !axis.horiz)),
- changeRatio = fixedRange && (newMax - newMin) / fixedRange;
- // Add/remove half point range to/from the extremes (#1172)
- if (!defined(fixedMin)) {
- newMin = correctFloat(newMin + halfPointRange);
- }
- if (!defined(fixedMax)) {
- newMax = correctFloat(newMax - halfPointRange);
- }
- // If the difference between the fixed range and the actual requested
- // range is too great, the user is dragging across an ordinal gap, and
- // we need to release the range selector button.
- if (changeRatio > 0.7 && changeRatio < 1.3) {
- if (fixedMax) {
- newMin = newMax - fixedRange;
- }
- else {
- newMax = newMin + fixedRange;
- }
- }
- if (!isNumber(newMin) || !isNumber(newMax)) { // #1195, #7411
- newMin = newMax = void 0;
- }
- return {
- min: newMin,
- max: newMax
- };
- };
- return NavigatorAxisAdditions;
- }());
- /**
- * @private
- * @class
- */
- var NavigatorAxis = /** @class */ (function () {
- function NavigatorAxis() {
- }
- /* *
- *
- * Static Functions
- *
- * */
- /**
- * @private
- */
- NavigatorAxis.compose = function (AxisClass) {
- AxisClass.keepProps.push('navigatorAxis');
- /* eslint-disable no-invalid-this */
- addEvent(AxisClass, 'init', function () {
- var axis = this;
- if (!axis.navigatorAxis) {
- axis.navigatorAxis = new NavigatorAxisAdditions(axis);
- }
- });
- // For Stock charts, override selection zooming with some special
- // features because X axis zooming is already allowed by the Navigator
- // and Range selector.
- addEvent(AxisClass, 'zoom', function (e) {
- var axis = this;
- var chart = axis.chart;
- var chartOptions = chart.options;
- var navigator = chartOptions.navigator;
- var navigatorAxis = axis.navigatorAxis;
- var pinchType = chartOptions.chart.pinchType;
- var rangeSelector = chartOptions.rangeSelector;
- var zoomType = chartOptions.chart.zoomType;
- var previousZoom;
- if (axis.isXAxis && ((navigator && navigator.enabled) ||
- (rangeSelector && rangeSelector.enabled))) {
- // For y only zooming, ignore the X axis completely
- if (zoomType === 'y') {
- e.zoomed = false;
- // For xy zooming, record the state of the zoom before zoom
- // selection, then when the reset button is pressed, revert to
- // this state. This should apply only if the chart is
- // initialized with a range (#6612), otherwise zoom all the way
- // out.
- }
- else if (((!isTouchDevice && zoomType === 'xy') ||
- (isTouchDevice && pinchType === 'xy')) &&
- axis.options.range) {
- previousZoom = navigatorAxis.previousZoom;
- if (defined(e.newMin)) {
- navigatorAxis.previousZoom = [axis.min, axis.max];
- }
- else if (previousZoom) {
- e.newMin = previousZoom[0];
- e.newMax = previousZoom[1];
- navigatorAxis.previousZoom = void 0;
- }
- }
- }
- if (typeof e.zoomed !== 'undefined') {
- e.preventDefault();
- }
- });
- /* eslint-enable no-invalid-this */
- };
- /* *
- *
- * Static Properties
- *
- * */
- /**
- * @private
- */
- NavigatorAxis.AdditionsClass = NavigatorAxisAdditions;
- return NavigatorAxis;
- }());
- return NavigatorAxis;
- });
- _registerModule(_modules, 'Core/Navigator.js', [_modules['Core/Axis/Axis.js'], _modules['Core/Chart/Chart.js'], _modules['Core/Color.js'], _modules['Core/Globals.js'], _modules['Core/Axis/NavigatorAxis.js'], _modules['Core/Options.js'], _modules['Core/Scrollbar.js'], _modules['Core/Utilities.js']], function (Axis, Chart, Color, H, NavigatorAxis, O, Scrollbar, U) {
- /* *
- *
- * (c) 2010-2020 Torstein Honsi
- *
- * License: www.highcharts.com/license
- *
- * !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
- *
- * */
- var color = Color.parse;
- var defaultOptions = O.defaultOptions;
- var addEvent = U.addEvent,
- clamp = U.clamp,
- correctFloat = U.correctFloat,
- defined = U.defined,
- destroyObjectProperties = U.destroyObjectProperties,
- erase = U.erase,
- extend = U.extend,
- find = U.find,
- isArray = U.isArray,
- isNumber = U.isNumber,
- merge = U.merge,
- pick = U.pick,
- removeEvent = U.removeEvent,
- splat = U.splat;
- var hasTouch = H.hasTouch,
- isTouchDevice = H.isTouchDevice,
- Series = H.Series,
- seriesTypes = H.seriesTypes,
- defaultSeriesType,
- // Finding the min or max of a set of variables where we don't know if they
- // are defined, is a pattern that is repeated several places in Highcharts.
- // Consider making this a global utility method.
- numExt = function (extreme) {
- var args = [];
- for (var _i = 1; _i < arguments.length; _i++) {
- args[_i - 1] = arguments[_i];
- }
- var numbers = [].filter.call(args,
- isNumber);
- if (numbers.length) {
- return Math[extreme].apply(0, numbers);
- }
- };
- defaultSeriesType = typeof seriesTypes.areaspline === 'undefined' ?
- 'line' :
- 'areaspline';
- extend(defaultOptions, {
- /**
- * Maximum range which can be set using the navigator's handles.
- * Opposite of [xAxis.minRange](#xAxis.minRange).
- *
- * @sample {highstock} stock/navigator/maxrange/
- * Defined max and min range
- *
- * @type {number}
- * @since 6.0.0
- * @product highstock gantt
- * @apioption xAxis.maxRange
- */
- /**
- * The navigator is a small series below the main series, displaying
- * a view of the entire data set. It provides tools to zoom in and
- * out on parts of the data as well as panning across the dataset.
- *
- * @product highstock gantt
- * @optionparent navigator
- */
- navigator: {
- /**
- * Whether the navigator and scrollbar should adapt to updated data
- * in the base X axis. When loading data async, as in the demo below,
- * this should be `false`. Otherwise new data will trigger navigator
- * redraw, which will cause unwanted looping. In the demo below, the
- * data in the navigator is set only once. On navigating, only the main
- * chart content is updated.
- *
- * @sample {highstock} stock/demo/lazy-loading/
- * Set to false with async data loading
- *
- * @type {boolean}
- * @default true
- * @apioption navigator.adaptToUpdatedData
- */
- /**
- * An integer identifying the index to use for the base series, or a
- * string representing the id of the series.
- *
- * **Note**: As of Highcharts 5.0, this is now a deprecated option.
- * Prefer [series.showInNavigator](#plotOptions.series.showInNavigator).
- *
- * @see [series.showInNavigator](#plotOptions.series.showInNavigator)
- *
- * @deprecated
- * @type {number|string}
- * @default 0
- * @apioption navigator.baseSeries
- */
- /**
- * Enable or disable the navigator.
- *
- * @sample {highstock} stock/navigator/enabled/
- * Disable the navigator
- *
- * @type {boolean}
- * @default true
- * @apioption navigator.enabled
- */
- /**
- * When the chart is inverted, whether to draw the navigator on the
- * opposite side.
- *
- * @type {boolean}
- * @default false
- * @since 5.0.8
- * @apioption navigator.opposite
- */
- /**
- * The height of the navigator.
- *
- * @sample {highstock} stock/navigator/height/
- * A higher navigator
- */
- height: 40,
- /**
- * The distance from the nearest element, the X axis or X axis labels.
- *
- * @sample {highstock} stock/navigator/margin/
- * A margin of 2 draws the navigator closer to the X axis labels
- */
- margin: 25,
- /**
- * Whether the mask should be inside the range marking the zoomed
- * range, or outside. In Highstock 1.x it was always `false`.
- *
- * @sample {highstock} stock/navigator/maskinside-false/
- * False, mask outside
- *
- * @since 2.0
- */
- maskInside: true,
- /**
- * Options for the handles for dragging the zoomed area.
- *
- * @sample {highstock} stock/navigator/handles/
- * Colored handles
- */
- handles: {
- /**
- * Width for handles.
- *
- * @sample {highstock} stock/navigator/styled-handles/
- * Styled handles
- *
- * @since 6.0.0
- */
- width: 7,
- /**
- * Height for handles.
- *
- * @sample {highstock} stock/navigator/styled-handles/
- * Styled handles
- *
- * @since 6.0.0
- */
- height: 15,
- /**
- * Array to define shapes of handles. 0-index for left, 1-index for
- * right.
- *
- * Additionally, the URL to a graphic can be given on this form:
- * `url(graphic.png)`. Note that for the image to be applied to
- * exported charts, its URL needs to be accessible by the export
- * server.
- *
- * Custom callbacks for symbol path generation can also be added to
- * `Highcharts.SVGRenderer.prototype.symbols`. The callback is then
- * used by its method name, as shown in the demo.
- *
- * @sample {highstock} stock/navigator/styled-handles/
- * Styled handles
- *
- * @type {Array<string>}
- * @default ["navigator-handle", "navigator-handle"]
- * @since 6.0.0
- */
- symbols: ['navigator-handle', 'navigator-handle'],
- /**
- * Allows to enable/disable handles.
- *
- * @since 6.0.0
- */
- enabled: true,
- /**
- * The width for the handle border and the stripes inside.
- *
- * @sample {highstock} stock/navigator/styled-handles/
- * Styled handles
- *
- * @since 6.0.0
- * @apioption navigator.handles.lineWidth
- */
- lineWidth: 1,
- /**
- * The fill for the handle.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- backgroundColor: '#f2f2f2',
- /**
- * The stroke for the handle border and the stripes inside.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- */
- borderColor: '#999999'
- },
- /**
- * The color of the mask covering the areas of the navigator series
- * that are currently not visible in the main series. The default
- * color is bluish with an opacity of 0.3 to see the series below.
- *
- * @see In styled mode, the mask is styled with the
- * `.highcharts-navigator-mask` and
- * `.highcharts-navigator-mask-inside` classes.
- *
- * @sample {highstock} stock/navigator/maskfill/
- * Blue, semi transparent mask
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default rgba(102,133,194,0.3)
- */
- maskFill: color('#6685c2').setOpacity(0.3).get(),
- /**
- * The color of the line marking the currently zoomed area in the
- * navigator.
- *
- * @sample {highstock} stock/navigator/outline/
- * 2px blue outline
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @default #cccccc
- */
- outlineColor: '#cccccc',
- /**
- * The width of the line marking the currently zoomed area in the
- * navigator.
- *
- * @see In styled mode, the outline stroke width is set with the
- * `.highcharts-navigator-outline` class.
- *
- * @sample {highstock} stock/navigator/outline/
- * 2px blue outline
- *
- * @type {number}
- */
- outlineWidth: 1,
- /**
- * Options for the navigator series. Available options are the same
- * as any series, documented at [plotOptions](#plotOptions.series)
- * and [series](#series).
- *
- * Unless data is explicitly defined on navigator.series, the data
- * is borrowed from the first series in the chart.
- *
- * Default series options for the navigator series are:
- * ```js
- * series: {
- * type: 'areaspline',
- * fillOpacity: 0.05,
- * dataGrouping: {
- * smoothed: true
- * },
- * lineWidth: 1,
- * marker: {
- * enabled: false
- * }
- * }
- * ```
- *
- * @see In styled mode, the navigator series is styled with the
- * `.highcharts-navigator-series` class.
- *
- * @sample {highstock} stock/navigator/series-data/
- * Using a separate data set for the navigator
- * @sample {highstock} stock/navigator/series/
- * A green navigator series
- *
- * @type {*|Array<*>|Highcharts.SeriesOptionsType|Array<Highcharts.SeriesOptionsType>}
- */
- series: {
- /**
- * The type of the navigator series.
- *
- * Heads up:
- * In column-type navigator, zooming is limited to at least one
- * point with its `pointRange`.
- *
- * @sample {highstock} stock/navigator/column/
- * Column type navigator
- *
- * @type {string}
- * @default {highstock} `areaspline` if defined, otherwise `line`
- * @default {gantt} gantt
- */
- type: defaultSeriesType,
- /**
- * The fill opacity of the navigator series.
- */
- fillOpacity: 0.05,
- /**
- * The pixel line width of the navigator series.
- */
- lineWidth: 1,
- /**
- * @ignore-option
- */
- compare: null,
- /**
- * Unless data is explicitly defined, the data is borrowed from the
- * first series in the chart.
- *
- * @type {Array<number|Array<number|string|null>|object|null>}
- * @product highstock
- * @apioption navigator.series.data
- */
- /**
- * Data grouping options for the navigator series.
- *
- * @extends plotOptions.series.dataGrouping
- */
- dataGrouping: {
- approximation: 'average',
- enabled: true,
- groupPixelWidth: 2,
- smoothed: true,
- // Day and week differs from plotOptions.series.dataGrouping
- units: [
- ['millisecond', [1, 2, 5, 10, 20, 25, 50, 100, 200, 500]],
- ['second', [1, 2, 5, 10, 15, 30]],
- ['minute', [1, 2, 5, 10, 15, 30]],
- ['hour', [1, 2, 3, 4, 6, 8, 12]],
- ['day', [1, 2, 3, 4]],
- ['week', [1, 2, 3]],
- ['month', [1, 3, 6]],
- ['year', null]
- ]
- },
- /**
- * Data label options for the navigator series. Data labels are
- * disabled by default on the navigator series.
- *
- * @extends plotOptions.series.dataLabels
- */
- dataLabels: {
- enabled: false,
- zIndex: 2 // #1839
- },
- id: 'highcharts-navigator-series',
- className: 'highcharts-navigator-series',
- /**
- * Sets the fill color of the navigator series.
- *
- * @type {Highcharts.ColorString|Highcharts.GradientColorObject|Highcharts.PatternObject}
- * @apioption navigator.series.color
- */
- /**
- * Line color for the navigator series. Allows setting the color
- * while disallowing the default candlestick setting.
- *
- * @type {Highcharts.ColorString|null}
- */
- lineColor: null,
- marker: {
- enabled: false
- },
- /**
- * Since Highstock v8, default value is the same as default
- * `pointRange` defined for a specific type (e.g. `null` for
- * column type).
- *
- * In Highstock version < 8, defaults to 0.
- *
- * @extends plotOptions.series.pointRange
- * @type {number|null}
- * @apioption navigator.series.pointRange
- */
- /**
- * The threshold option. Setting it to 0 will make the default
- * navigator area series draw its area from the 0 value and up.
- *
- * @type {number|null}
- */
- threshold: null
- },
- /**
- * Options for the navigator X axis. Default series options for the
- * navigator xAxis are:
- * ```js
- * xAxis: {
- * tickWidth: 0,
- * lineWidth: 0,
- * gridLineWidth: 1,
- * tickPixelInterval: 200,
- * labels: {
- * align: 'left',
- * style: {
- * color: '#888'
- * },
- * x: 3,
- * y: -4
- * }
- * }
- * ```
- *
- * @extends xAxis
- * @excluding linkedTo, maxZoom, minRange, opposite, range, scrollbar,
- * showEmpty, maxRange
- */
- xAxis: {
- /**
- * Additional range on the right side of the xAxis. Works similar to
- * xAxis.maxPadding, but value is set in milliseconds.
- * Can be set for both, main xAxis and navigator's xAxis.
- *
- * @since 6.0.0
- */
- overscroll: 0,
- className: 'highcharts-navigator-xaxis',
- tickLength: 0,
- lineWidth: 0,
- gridLineColor: '#e6e6e6',
- gridLineWidth: 1,
- tickPixelInterval: 200,
- labels: {
- align: 'left',
- /**
- * @type {Highcharts.CSSObject}
- */
- style: {
- /** @ignore */
- color: '#999999'
- },
- x: 3,
- y: -4
- },
- crosshair: false
- },
- /**
- * Options for the navigator Y axis. Default series options for the
- * navigator yAxis are:
- * ```js
- * yAxis: {
- * gridLineWidth: 0,
- * startOnTick: false,
- * endOnTick: false,
- * minPadding: 0.1,
- * maxPadding: 0.1,
- * labels: {
- * enabled: false
- * },
- * title: {
- * text: null
- * },
- * tickWidth: 0
- * }
- * ```
- *
- * @extends yAxis
- * @excluding height, linkedTo, maxZoom, minRange, ordinal, range,
- * showEmpty, scrollbar, top, units, maxRange, minLength,
- * maxLength, resize
- */
- yAxis: {
- className: 'highcharts-navigator-yaxis',
- gridLineWidth: 0,
- startOnTick: false,
- endOnTick: false,
- minPadding: 0.1,
- maxPadding: 0.1,
- labels: {
- enabled: false
- },
- crosshair: false,
- title: {
- text: null
- },
- tickLength: 0,
- tickWidth: 0
- }
- }
- });
- /* eslint-disable no-invalid-this, valid-jsdoc */
- /**
- * Draw one of the handles on the side of the zoomed range in the navigator
- *
- * @private
- * @function Highcharts.Renderer#symbols.navigator-handle
- * @param {number} x
- * @param {number} y
- * @param {number} w
- * @param {number} h
- * @param {Highcharts.NavigatorHandlesOptions} options
- * @return {Highcharts.SVGPathArray}
- * Path to be used in a handle
- */
- H.Renderer.prototype.symbols['navigator-handle'] = function (x, y, w, h, options) {
- var halfWidth = (options && options.width || 0) / 2,
- markerPosition = Math.round(halfWidth / 3) + 0.5,
- height = options && options.height || 0;
- return [
- ['M', -halfWidth - 1, 0.5],
- ['L', halfWidth, 0.5],
- ['L', halfWidth, height + 0.5],
- ['L', -halfWidth - 1, height + 0.5],
- ['L', -halfWidth - 1, 0.5],
- ['M', -markerPosition, 4],
- ['L', -markerPosition, height - 3],
- ['M', markerPosition - 1, 4],
- ['L', markerPosition - 1, height - 3]
- ];
- };
- /**
- * The Navigator class
- *
- * @private
- * @class
- * @name Highcharts.Navigator
- *
- * @param {Highcharts.Chart} chart
- * Chart object
- */
- var Navigator = /** @class */ (function () {
- function Navigator(chart) {
- this.baseSeries = void 0;
- this.chart = void 0;
- this.handles = void 0;
- this.height = void 0;
- this.left = void 0;
- this.navigatorEnabled = void 0;
- this.navigatorGroup = void 0;
- this.navigatorOptions = void 0;
- this.navigatorSeries = void 0;
- this.navigatorSize = void 0;
- this.opposite = void 0;
- this.outline = void 0;
- this.outlineHeight = void 0;
- this.range = void 0;
- this.rendered = void 0;
- this.shades = void 0;
- this.size = void 0;
- this.top = void 0;
- this.xAxis = void 0;
- this.yAxis = void 0;
- this.zoomedMax = void 0;
- this.zoomedMin = void 0;
- this.init(chart);
- }
- /**
- * Draw one of the handles on the side of the zoomed range in the navigator
- *
- * @private
- * @function Highcharts.Navigator#drawHandle
- *
- * @param {number} x
- * The x center for the handle
- *
- * @param {number} index
- * 0 for left and 1 for right
- *
- * @param {boolean|undefined} inverted
- * flag for chart.inverted
- *
- * @param {string} verb
- * use 'animate' or 'attr'
- */
- Navigator.prototype.drawHandle = function (x, index, inverted, verb) {
- var navigator = this,
- height = navigator.navigatorOptions.handles.height;
- // Place it
- navigator.handles[index][verb](inverted ? {
- translateX: Math.round(navigator.left + navigator.height / 2),
- translateY: Math.round(navigator.top + parseInt(x, 10) + 0.5 - height)
- } : {
- translateX: Math.round(navigator.left + parseInt(x, 10)),
- translateY: Math.round(navigator.top + navigator.height / 2 - height / 2 - 1)
- });
- };
- /**
- * Render outline around the zoomed range
- *
- * @private
- * @function Highcharts.Navigator#drawOutline
- *
- * @param {number} zoomedMin
- * in pixels position where zoomed range starts
- *
- * @param {number} zoomedMax
- * in pixels position where zoomed range ends
- *
- * @param {boolean|undefined} inverted
- * flag if chart is inverted
- *
- * @param {string} verb
- * use 'animate' or 'attr'
- */
- Navigator.prototype.drawOutline = function (zoomedMin, zoomedMax, inverted, verb) {
- var navigator = this,
- maskInside = navigator.navigatorOptions.maskInside,
- outlineWidth = navigator.outline.strokeWidth(),
- halfOutline = outlineWidth / 2,
- outlineCorrection = (outlineWidth % 2) / 2, // #5800
- outlineHeight = navigator.outlineHeight,
- scrollbarHeight = navigator.scrollbarHeight || 0,
- navigatorSize = navigator.size,
- left = navigator.left - scrollbarHeight,
- navigatorTop = navigator.top,
- verticalMin,
- path;
- if (inverted) {
- left -= halfOutline;
- verticalMin = navigatorTop + zoomedMax + outlineCorrection;
- zoomedMax = navigatorTop + zoomedMin + outlineCorrection;
- path = [
- ['M', left + outlineHeight, navigatorTop - scrollbarHeight - outlineCorrection],
- ['L', left + outlineHeight, verticalMin],
- ['L', left, verticalMin],
- ['L', left, zoomedMax],
- ['L', left + outlineHeight, zoomedMax],
- ['L', left + outlineHeight, navigatorTop + navigatorSize + scrollbarHeight]
- ];
- if (maskInside) {
- path.push(['M', left + outlineHeight, verticalMin - halfOutline], // upper left of zoomed range
- ['L', left + outlineHeight, zoomedMax + halfOutline] // upper right of z.r.
- );
- }
- }
- else {
- zoomedMin += left + scrollbarHeight - outlineCorrection;
- zoomedMax += left + scrollbarHeight - outlineCorrection;
- navigatorTop += halfOutline;
- path = [
- ['M', left, navigatorTop],
- ['L', zoomedMin, navigatorTop],
- ['L', zoomedMin, navigatorTop + outlineHeight],
- ['L', zoomedMax, navigatorTop + outlineHeight],
- ['L', zoomedMax, navigatorTop],
- ['L', left + navigatorSize + scrollbarHeight * 2, navigatorTop] // right
- ];
- if (maskInside) {
- path.push(['M', zoomedMin - halfOutline, navigatorTop], // upper left of zoomed range
- ['L', zoomedMax + halfOutline, navigatorTop] // upper right of z.r.
- );
- }
- }
- navigator.outline[verb]({
- d: path
- });
- };
- /**
- * Render outline around the zoomed range
- *
- * @private
- * @function Highcharts.Navigator#drawMasks
- *
- * @param {number} zoomedMin
- * in pixels position where zoomed range starts
- *
- * @param {number} zoomedMax
- * in pixels position where zoomed range ends
- *
- * @param {boolean|undefined} inverted
- * flag if chart is inverted
- *
- * @param {string} verb
- * use 'animate' or 'attr'
- */
- Navigator.prototype.drawMasks = function (zoomedMin, zoomedMax, inverted, verb) {
- var navigator = this,
- left = navigator.left,
- top = navigator.top,
- navigatorHeight = navigator.height,
- height,
- width,
- x,
- y;
- // Determine rectangle position & size
- // According to (non)inverted position:
- if (inverted) {
- x = [left, left, left];
- y = [top, top + zoomedMin, top + zoomedMax];
- width = [navigatorHeight, navigatorHeight, navigatorHeight];
- height = [
- zoomedMin,
- zoomedMax - zoomedMin,
- navigator.size - zoomedMax
- ];
- }
- else {
- x = [left, left + zoomedMin, left + zoomedMax];
- y = [top, top, top];
- width = [
- zoomedMin,
- zoomedMax - zoomedMin,
- navigator.size - zoomedMax
- ];
- height = [navigatorHeight, navigatorHeight, navigatorHeight];
- }
- navigator.shades.forEach(function (shade, i) {
- shade[verb]({
- x: x[i],
- y: y[i],
- width: width[i],
- height: height[i]
- });
- });
- };
- /**
- * Generate DOM elements for a navigator:
- *
- * - main navigator group
- *
- * - all shades
- *
- * - outline
- *
- * - handles
- *
- * @private
- * @function Highcharts.Navigator#renderElements
- */
- Navigator.prototype.renderElements = function () {
- var navigator = this,
- navigatorOptions = navigator.navigatorOptions,
- maskInside = navigatorOptions.maskInside,
- chart = navigator.chart,
- inverted = chart.inverted,
- renderer = chart.renderer,
- navigatorGroup,
- mouseCursor = {
- cursor: inverted ? 'ns-resize' : 'ew-resize'
- };
- // Create the main navigator group
- navigator.navigatorGroup = navigatorGroup = renderer.g('navigator')
- .attr({
- zIndex: 8,
- visibility: 'hidden'
- })
- .add();
- // Create masks, each mask will get events and fill:
- [
- !maskInside,
- maskInside,
- !maskInside
- ].forEach(function (hasMask, index) {
- navigator.shades[index] = renderer.rect()
- .addClass('highcharts-navigator-mask' +
- (index === 1 ? '-inside' : '-outside'))
- .add(navigatorGroup);
- if (!chart.styledMode) {
- navigator.shades[index]
- .attr({
- fill: hasMask ?
- navigatorOptions.maskFill :
- 'rgba(0,0,0,0)'
- })
- .css((index === 1) && mouseCursor);
- }
- });
- // Create the outline:
- navigator.outline = renderer.path()
- .addClass('highcharts-navigator-outline')
- .add(navigatorGroup);
- if (!chart.styledMode) {
- navigator.outline.attr({
- 'stroke-width': navigatorOptions.outlineWidth,
- stroke: navigatorOptions.outlineColor
- });
- }
- // Create the handlers:
- if (navigatorOptions.handles.enabled) {
- [0, 1].forEach(function (index) {
- navigatorOptions.handles.inverted = chart.inverted;
- navigator.handles[index] = renderer.symbol(navigatorOptions.handles.symbols[index], -navigatorOptions.handles.width / 2 - 1, 0, navigatorOptions.handles.width, navigatorOptions.handles.height, navigatorOptions.handles);
- // zIndex = 6 for right handle, 7 for left.
- // Can't be 10, because of the tooltip in inverted chart #2908
- navigator.handles[index].attr({ zIndex: 7 - index })
- .addClass('highcharts-navigator-handle ' +
- 'highcharts-navigator-handle-' +
- ['left', 'right'][index]).add(navigatorGroup);
- if (!chart.styledMode) {
- var handlesOptions = navigatorOptions.handles;
- navigator.handles[index]
- .attr({
- fill: handlesOptions.backgroundColor,
- stroke: handlesOptions.borderColor,
- 'stroke-width': handlesOptions.lineWidth
- })
- .css(mouseCursor);
- }
- });
- }
- };
- /**
- * Update navigator
- *
- * @private
- * @function Highcharts.Navigator#update
- *
- * @param {Highcharts.NavigatorOptions} options
- * Options to merge in when updating navigator
- */
- Navigator.prototype.update = function (options) {
- // Remove references to old navigator series in base series
- (this.series || []).forEach(function (series) {
- if (series.baseSeries) {
- delete series.baseSeries.navigatorSeries;
- }
- });
- // Destroy and rebuild navigator
- this.destroy();
- var chartOptions = this.chart.options;
- merge(true, chartOptions.navigator, this.options, options);
- this.init(this.chart);
- };
- /**
- * Render the navigator
- *
- * @private
- * @function Highcharts.Navigator#render
- * @param {number} min
- * X axis value minimum
- * @param {number} max
- * X axis value maximum
- * @param {number} [pxMin]
- * Pixel value minimum
- * @param {number} [pxMax]
- * Pixel value maximum
- * @return {void}
- */
- Navigator.prototype.render = function (min, max, pxMin, pxMax) {
- var navigator = this,
- chart = navigator.chart,
- navigatorWidth,
- scrollbarLeft,
- scrollbarTop,
- scrollbarHeight = navigator.scrollbarHeight,
- navigatorSize,
- xAxis = navigator.xAxis,
- pointRange = xAxis.pointRange || 0,
- scrollbarXAxis = xAxis.navigatorAxis.fake ? chart.xAxis[0] : xAxis,
- navigatorEnabled = navigator.navigatorEnabled,
- zoomedMin,
- zoomedMax,
- rendered = navigator.rendered,
- inverted = chart.inverted,
- verb,
- newMin,
- newMax,
- currentRange,
- minRange = chart.xAxis[0].minRange,
- maxRange = chart.xAxis[0].options.maxRange;
- // Don't redraw while moving the handles (#4703).
- if (this.hasDragged && !defined(pxMin)) {
- return;
- }
- min = correctFloat(min - pointRange / 2);
- max = correctFloat(max + pointRange / 2);
- // Don't render the navigator until we have data (#486, #4202, #5172).
- if (!isNumber(min) || !isNumber(max)) {
- // However, if navigator was already rendered, we may need to resize
- // it. For example hidden series, but visible navigator (#6022).
- if (rendered) {
- pxMin = 0;
- pxMax = pick(xAxis.width, scrollbarXAxis.width);
- }
- else {
- return;
- }
- }
- navigator.left = pick(xAxis.left,
- // in case of scrollbar only, without navigator
- chart.plotLeft + scrollbarHeight +
- (inverted ? chart.plotWidth : 0));
- navigator.size = zoomedMax = navigatorSize = pick(xAxis.len, (inverted ? chart.plotHeight : chart.plotWidth) -
- 2 * scrollbarHeight);
- if (inverted) {
- navigatorWidth = scrollbarHeight;
- }
- else {
- navigatorWidth = navigatorSize + 2 * scrollbarHeight;
- }
- // Get the pixel position of the handles
- pxMin = pick(pxMin, xAxis.toPixels(min, true));
- pxMax = pick(pxMax, xAxis.toPixels(max, true));
- // Verify (#1851, #2238)
- if (!isNumber(pxMin) || Math.abs(pxMin) === Infinity) {
- pxMin = 0;
- pxMax = navigatorWidth;
- }
- // Are we below the minRange? (#2618, #6191)
- newMin = xAxis.toValue(pxMin, true);
- newMax = xAxis.toValue(pxMax, true);
- currentRange = Math.abs(correctFloat(newMax - newMin));
- if (currentRange < minRange) {
- if (this.grabbedLeft) {
- pxMin = xAxis.toPixels(newMax - minRange - pointRange, true);
- }
- else if (this.grabbedRight) {
- pxMax = xAxis.toPixels(newMin + minRange + pointRange, true);
- }
- }
- else if (defined(maxRange) &&
- correctFloat(currentRange - pointRange) > maxRange) {
- if (this.grabbedLeft) {
- pxMin = xAxis.toPixels(newMax - maxRange - pointRange, true);
- }
- else if (this.grabbedRight) {
- pxMax = xAxis.toPixels(newMin + maxRange + pointRange, true);
- }
- }
- // Handles are allowed to cross, but never exceed the plot area
- navigator.zoomedMax = clamp(Math.max(pxMin, pxMax), 0, zoomedMax);
- navigator.zoomedMin = clamp(navigator.fixedWidth ?
- navigator.zoomedMax - navigator.fixedWidth :
- Math.min(pxMin, pxMax), 0, zoomedMax);
- navigator.range = navigator.zoomedMax - navigator.zoomedMin;
- zoomedMax = Math.round(navigator.zoomedMax);
- zoomedMin = Math.round(navigator.zoomedMin);
- if (navigatorEnabled) {
- navigator.navigatorGroup.attr({
- visibility: 'visible'
- });
- // Place elements
- verb = rendered && !navigator.hasDragged ? 'animate' : 'attr';
- navigator.drawMasks(zoomedMin, zoomedMax, inverted, verb);
- navigator.drawOutline(zoomedMin, zoomedMax, inverted, verb);
- if (navigator.navigatorOptions.handles.enabled) {
- navigator.drawHandle(zoomedMin, 0, inverted, verb);
- navigator.drawHandle(zoomedMax, 1, inverted, verb);
- }
- }
- if (navigator.scrollbar) {
- if (inverted) {
- scrollbarTop = navigator.top - scrollbarHeight;
- scrollbarLeft = navigator.left - scrollbarHeight +
- (navigatorEnabled || !scrollbarXAxis.opposite ? 0 :
- // Multiple axes has offsets:
- (scrollbarXAxis.titleOffset || 0) +
- // Self margin from the axis.title
- scrollbarXAxis.axisTitleMargin);
- scrollbarHeight = navigatorSize + 2 * scrollbarHeight;
- }
- else {
- scrollbarTop = navigator.top + (navigatorEnabled ?
- navigator.height :
- -scrollbarHeight);
- scrollbarLeft = navigator.left - scrollbarHeight;
- }
- // Reposition scrollbar
- navigator.scrollbar.position(scrollbarLeft, scrollbarTop, navigatorWidth, scrollbarHeight);
- // Keep scale 0-1
- navigator.scrollbar.setRange(
- // Use real value, not rounded because range can be very small
- // (#1716)
- navigator.zoomedMin / (navigatorSize || 1), navigator.zoomedMax / (navigatorSize || 1));
- }
- navigator.rendered = true;
- };
- /**
- * Set up the mouse and touch events for the navigator
- *
- * @private
- * @function Highcharts.Navigator#addMouseEvents
- */
- Navigator.prototype.addMouseEvents = function () {
- var navigator = this,
- chart = navigator.chart,
- container = chart.container,
- eventsToUnbind = [],
- mouseMoveHandler,
- mouseUpHandler;
- /**
- * Create mouse events' handlers.
- * Make them as separate functions to enable wrapping them:
- */
- navigator.mouseMoveHandler = mouseMoveHandler = function (e) {
- navigator.onMouseMove(e);
- };
- navigator.mouseUpHandler = mouseUpHandler = function (e) {
- navigator.onMouseUp(e);
- };
- // Add shades and handles mousedown events
- eventsToUnbind = navigator.getPartsEvents('mousedown');
- // Add mouse move and mouseup events. These are bind to doc/container,
- // because Navigator.grabbedSomething flags are stored in mousedown
- // events
- eventsToUnbind.push(addEvent(chart.renderTo, 'mousemove', mouseMoveHandler), addEvent(container.ownerDocument, 'mouseup', mouseUpHandler));
- // Touch events
- if (hasTouch) {
- eventsToUnbind.push(addEvent(chart.renderTo, 'touchmove', mouseMoveHandler), addEvent(container.ownerDocument, 'touchend', mouseUpHandler));
- eventsToUnbind.concat(navigator.getPartsEvents('touchstart'));
- }
- navigator.eventsToUnbind = eventsToUnbind;
- // Data events
- if (navigator.series && navigator.series[0]) {
- eventsToUnbind.push(addEvent(navigator.series[0].xAxis, 'foundExtremes', function () {
- chart.navigator.modifyNavigatorAxisExtremes();
- }));
- }
- };
- /**
- * Generate events for handles and masks
- *
- * @private
- * @function Highcharts.Navigator#getPartsEvents
- *
- * @param {string} eventName
- * Event name handler, 'mousedown' or 'touchstart'
- *
- * @return {Array<Function>}
- * An array of functions to remove navigator functions from the
- * events again.
- */
- Navigator.prototype.getPartsEvents = function (eventName) {
- var navigator = this,
- events = [];
- ['shades', 'handles'].forEach(function (name) {
- navigator[name].forEach(function (navigatorItem, index) {
- events.push(addEvent(navigatorItem.element, eventName, function (e) {
- navigator[name + 'Mousedown'](e, index);
- }));
- });
- });
- return events;
- };
- /**
- * Mousedown on a shaded mask, either:
- *
- * - will be stored for future drag&drop
- *
- * - will directly shift to a new range
- *
- * @private
- * @function Highcharts.Navigator#shadesMousedown
- *
- * @param {Highcharts.PointerEventObject} e
- * Mouse event
- *
- * @param {number} index
- * Index of a mask in Navigator.shades array
- */
- Navigator.prototype.shadesMousedown = function (e, index) {
- e = this.chart.pointer.normalize(e);
- var navigator = this,
- chart = navigator.chart,
- xAxis = navigator.xAxis,
- zoomedMin = navigator.zoomedMin,
- navigatorPosition = navigator.left,
- navigatorSize = navigator.size,
- range = navigator.range,
- chartX = e.chartX,
- fixedMax,
- fixedMin,
- ext,
- left;
- // For inverted chart, swap some options:
- if (chart.inverted) {
- chartX = e.chartY;
- navigatorPosition = navigator.top;
- }
- if (index === 1) {
- // Store information for drag&drop
- navigator.grabbedCenter = chartX;
- navigator.fixedWidth = range;
- navigator.dragOffset = chartX - zoomedMin;
- }
- else {
- // Shift the range by clicking on shaded areas
- left = chartX - navigatorPosition - range / 2;
- if (index === 0) {
- left = Math.max(0, left);
- }
- else if (index === 2 && left + range >= navigatorSize) {
- left = navigatorSize - range;
- if (navigator.reversedExtremes) {
- // #7713
- left -= range;
- fixedMin = navigator.getUnionExtremes().dataMin;
- }
- else {
- // #2293, #3543
- fixedMax = navigator.getUnionExtremes().dataMax;
- }
- }
- if (left !== zoomedMin) { // it has actually moved
- navigator.fixedWidth = range; // #1370
- ext = xAxis.navigatorAxis.toFixedRange(left, left + range, fixedMin, fixedMax);
- if (defined(ext.min)) { // #7411
- chart.xAxis[0].setExtremes(Math.min(ext.min, ext.max), Math.max(ext.min, ext.max), true, null, // auto animation
- { trigger: 'navigator' });
- }
- }
- }
- };
- /**
- * Mousedown on a handle mask.
- * Will store necessary information for drag&drop.
- *
- * @private
- * @function Highcharts.Navigator#handlesMousedown
- * @param {Highcharts.PointerEventObject} e
- * Mouse event
- * @param {number} index
- * Index of a handle in Navigator.handles array
- * @return {void}
- */
- Navigator.prototype.handlesMousedown = function (e, index) {
- e = this.chart.pointer.normalize(e);
- var navigator = this,
- chart = navigator.chart,
- baseXAxis = chart.xAxis[0],
- // For reversed axes, min and max are changed,
- // so the other extreme should be stored
- reverse = navigator.reversedExtremes;
- if (index === 0) {
- // Grab the left handle
- navigator.grabbedLeft = true;
- navigator.otherHandlePos = navigator.zoomedMax;
- navigator.fixedExtreme = reverse ? baseXAxis.min : baseXAxis.max;
- }
- else {
- // Grab the right handle
- navigator.grabbedRight = true;
- navigator.otherHandlePos = navigator.zoomedMin;
- navigator.fixedExtreme = reverse ? baseXAxis.max : baseXAxis.min;
- }
- chart.fixedRange = null;
- };
- /**
- * Mouse move event based on x/y mouse position.
- *
- * @private
- * @function Highcharts.Navigator#onMouseMove
- *
- * @param {Highcharts.PointerEventObject} e
- * Mouse event
- */
- Navigator.prototype.onMouseMove = function (e) {
- var navigator = this,
- chart = navigator.chart,
- left = navigator.left,
- navigatorSize = navigator.navigatorSize,
- range = navigator.range,
- dragOffset = navigator.dragOffset,
- inverted = chart.inverted,
- chartX;
- // In iOS, a mousemove event with e.pageX === 0 is fired when holding
- // the finger down in the center of the scrollbar. This should be
- // ignored.
- if (!e.touches || e.touches[0].pageX !== 0) { // #4696
- e = chart.pointer.normalize(e);
- chartX = e.chartX;
- // Swap some options for inverted chart
- if (inverted) {
- left = navigator.top;
- chartX = e.chartY;
- }
- // Drag left handle or top handle
- if (navigator.grabbedLeft) {
- navigator.hasDragged = true;
- navigator.render(0, 0, chartX - left, navigator.otherHandlePos);
- // Drag right handle or bottom handle
- }
- else if (navigator.grabbedRight) {
- navigator.hasDragged = true;
- navigator.render(0, 0, navigator.otherHandlePos, chartX - left);
- // Drag scrollbar or open area in navigator
- }
- else if (navigator.grabbedCenter) {
- navigator.hasDragged = true;
- if (chartX < dragOffset) { // outside left
- chartX = dragOffset;
- // outside right
- }
- else if (chartX >
- navigatorSize + dragOffset - range) {
- chartX = navigatorSize + dragOffset - range;
- }
- navigator.render(0, 0, chartX - dragOffset, chartX - dragOffset + range);
- }
- if (navigator.hasDragged &&
- navigator.scrollbar &&
- pick(navigator.scrollbar.options.liveRedraw,
- // By default, don't run live redraw on VML, on touch
- // devices or if the chart is in boost.
- H.svg && !isTouchDevice && !this.chart.isBoosting)) {
- e.DOMType = e.type; // DOMType is for IE8
- setTimeout(function () {
- navigator.onMouseUp(e);
- }, 0);
- }
- }
- };
- /**
- * Mouse up event based on x/y mouse position.
- *
- * @private
- * @function Highcharts.Navigator#onMouseUp
- * @param {Highcharts.PointerEventObject} e
- * Mouse event
- * @return {void}
- */
- Navigator.prototype.onMouseUp = function (e) {
- var navigator = this,
- chart = navigator.chart,
- xAxis = navigator.xAxis,
- scrollbar = navigator.scrollbar,
- DOMEvent = e.DOMEvent || e,
- inverted = chart.inverted,
- verb = navigator.rendered && !navigator.hasDragged ?
- 'animate' : 'attr',
- zoomedMax,
- zoomedMin,
- unionExtremes,
- fixedMin,
- fixedMax,
- ext;
- if (
- // MouseUp is called for both, navigator and scrollbar (that order),
- // which causes calling afterSetExtremes twice. Prevent first call
- // by checking if scrollbar is going to set new extremes (#6334)
- (navigator.hasDragged && (!scrollbar || !scrollbar.hasDragged)) ||
- e.trigger === 'scrollbar') {
- unionExtremes = navigator.getUnionExtremes();
- // When dragging one handle, make sure the other one doesn't change
- if (navigator.zoomedMin === navigator.otherHandlePos) {
- fixedMin = navigator.fixedExtreme;
- }
- else if (navigator.zoomedMax === navigator.otherHandlePos) {
- fixedMax = navigator.fixedExtreme;
- }
- // Snap to right edge (#4076)
- if (navigator.zoomedMax === navigator.size) {
- fixedMax = navigator.reversedExtremes ?
- unionExtremes.dataMin :
- unionExtremes.dataMax;
- }
- // Snap to left edge (#7576)
- if (navigator.zoomedMin === 0) {
- fixedMin = navigator.reversedExtremes ?
- unionExtremes.dataMax :
- unionExtremes.dataMin;
- }
- ext = xAxis.navigatorAxis.toFixedRange(navigator.zoomedMin, navigator.zoomedMax, fixedMin, fixedMax);
- if (defined(ext.min)) {
- chart.xAxis[0].setExtremes(Math.min(ext.min, ext.max), Math.max(ext.min, ext.max), true,
- // Run animation when clicking buttons, scrollbar track etc,
- // but not when dragging handles or scrollbar
- navigator.hasDragged ? false : null, {
- trigger: 'navigator',
- triggerOp: 'navigator-drag',
- DOMEvent: DOMEvent // #1838
- });
- }
- }
- if (e.DOMType !== 'mousemove' &&
- e.DOMType !== 'touchmove') {
- navigator.grabbedLeft = navigator.grabbedRight =
- navigator.grabbedCenter = navigator.fixedWidth =
- navigator.fixedExtreme = navigator.otherHandlePos =
- navigator.hasDragged = navigator.dragOffset = null;
- }
- // Update position of navigator shades, outline and handles (#12573)
- if (navigator.navigatorEnabled &&
- isNumber(navigator.zoomedMin) &&
- isNumber(navigator.zoomedMax)) {
- zoomedMin = Math.round(navigator.zoomedMin);
- zoomedMax = Math.round(navigator.zoomedMax);
- if (navigator.shades) {
- navigator.drawMasks(zoomedMin, zoomedMax, inverted, verb);
- }
- if (navigator.outline) {
- navigator.drawOutline(zoomedMin, zoomedMax, inverted, verb);
- }
- if (navigator.navigatorOptions.handles.enabled &&
- Object.keys(navigator.handles).length ===
- navigator.handles.length) {
- navigator.drawHandle(zoomedMin, 0, inverted, verb);
- navigator.drawHandle(zoomedMax, 1, inverted, verb);
- }
- }
- };
- /**
- * Removes the event handlers attached previously with addEvents.
- *
- * @private
- * @function Highcharts.Navigator#removeEvents
- * @return {void}
- */
- Navigator.prototype.removeEvents = function () {
- if (this.eventsToUnbind) {
- this.eventsToUnbind.forEach(function (unbind) {
- unbind();
- });
- this.eventsToUnbind = void 0;
- }
- this.removeBaseSeriesEvents();
- };
- /**
- * Remove data events.
- *
- * @private
- * @function Highcharts.Navigator#removeBaseSeriesEvents
- * @return {void}
- */
- Navigator.prototype.removeBaseSeriesEvents = function () {
- var baseSeries = this.baseSeries || [];
- if (this.navigatorEnabled && baseSeries[0]) {
- if (this.navigatorOptions.adaptToUpdatedData !== false) {
- baseSeries.forEach(function (series) {
- removeEvent(series, 'updatedData', this.updatedDataHandler);
- }, this);
- }
- // We only listen for extremes-events on the first baseSeries
- if (baseSeries[0].xAxis) {
- removeEvent(baseSeries[0].xAxis, 'foundExtremes', this.modifyBaseAxisExtremes);
- }
- }
- };
- /**
- * Initialize the Navigator object
- *
- * @private
- * @function Highcharts.Navigator#init
- *
- * @param {Highcharts.Chart} chart
- */
- Navigator.prototype.init = function (chart) {
- var chartOptions = chart.options,
- navigatorOptions = chartOptions.navigator,
- navigatorEnabled = navigatorOptions.enabled,
- scrollbarOptions = chartOptions.scrollbar,
- scrollbarEnabled = scrollbarOptions.enabled,
- height = navigatorEnabled ? navigatorOptions.height : 0,
- scrollbarHeight = scrollbarEnabled ?
- scrollbarOptions.height :
- 0;
- this.handles = [];
- this.shades = [];
- this.chart = chart;
- this.setBaseSeries();
- this.height = height;
- this.scrollbarHeight = scrollbarHeight;
- this.scrollbarEnabled = scrollbarEnabled;
- this.navigatorEnabled = navigatorEnabled;
- this.navigatorOptions = navigatorOptions;
- this.scrollbarOptions = scrollbarOptions;
- this.outlineHeight = height + scrollbarHeight;
- this.opposite = pick(navigatorOptions.opposite, Boolean(!navigatorEnabled && chart.inverted)); // #6262
- var navigator = this,
- baseSeries = navigator.baseSeries,
- xAxisIndex = chart.xAxis.length,
- yAxisIndex = chart.yAxis.length,
- baseXaxis = baseSeries && baseSeries[0] && baseSeries[0].xAxis ||
- chart.xAxis[0] || { options: {} };
- chart.isDirtyBox = true;
- if (navigator.navigatorEnabled) {
- // an x axis is required for scrollbar also
- navigator.xAxis = new Axis(chart, merge({
- // inherit base xAxis' break and ordinal options
- breaks: baseXaxis.options.breaks,
- ordinal: baseXaxis.options.ordinal
- }, navigatorOptions.xAxis, {
- id: 'navigator-x-axis',
- yAxis: 'navigator-y-axis',
- isX: true,
- type: 'datetime',
- index: xAxisIndex,
- isInternal: true,
- offset: 0,
- keepOrdinalPadding: true,
- startOnTick: false,
- endOnTick: false,
- minPadding: 0,
- maxPadding: 0,
- zoomEnabled: false
- }, chart.inverted ? {
- offsets: [scrollbarHeight, 0, -scrollbarHeight, 0],
- width: height
- } : {
- offsets: [0, -scrollbarHeight, 0, scrollbarHeight],
- height: height
- }));
- navigator.yAxis = new Axis(chart, merge(navigatorOptions.yAxis, {
- id: 'navigator-y-axis',
- alignTicks: false,
- offset: 0,
- index: yAxisIndex,
- isInternal: true,
- zoomEnabled: false
- }, chart.inverted ? {
- width: height
- } : {
- height: height
- }));
- // If we have a base series, initialize the navigator series
- if (baseSeries || navigatorOptions.series.data) {
- navigator.updateNavigatorSeries(false);
- // If not, set up an event to listen for added series
- }
- else if (chart.series.length === 0) {
- navigator.unbindRedraw = addEvent(chart, 'beforeRedraw', function () {
- // We've got one, now add it as base
- if (chart.series.length > 0 && !navigator.series) {
- navigator.setBaseSeries();
- navigator.unbindRedraw(); // reset
- }
- });
- }
- navigator.reversedExtremes = (chart.inverted && !navigator.xAxis.reversed) || (!chart.inverted && navigator.xAxis.reversed);
- // Render items, so we can bind events to them:
- navigator.renderElements();
- // Add mouse events
- navigator.addMouseEvents();
- // in case of scrollbar only, fake an x axis to get translation
- }
- else {
- navigator.xAxis = {
- chart: chart,
- navigatorAxis: {
- fake: true
- },
- translate: function (value, reverse) {
- var axis = chart.xAxis[0], ext = axis.getExtremes(), scrollTrackWidth = axis.len - 2 * scrollbarHeight, min = numExt('min', axis.options.min, ext.dataMin), valueRange = numExt('max', axis.options.max, ext.dataMax) - min;
- return reverse ?
- // from pixel to value
- (value * valueRange / scrollTrackWidth) + min :
- // from value to pixel
- scrollTrackWidth * (value - min) / valueRange;
- },
- toPixels: function (value) {
- return this.translate(value);
- },
- toValue: function (value) {
- return this.translate(value, true);
- }
- };
- navigator.xAxis.navigatorAxis.axis = navigator.xAxis;
- navigator.xAxis.navigatorAxis.toFixedRange = (NavigatorAxis.AdditionsClass.prototype.toFixedRange.bind(navigator.xAxis.navigatorAxis));
- }
- // Initialize the scrollbar
- if (chart.options.scrollbar.enabled) {
- chart.scrollbar = navigator.scrollbar = new Scrollbar(chart.renderer, merge(chart.options.scrollbar, {
- margin: navigator.navigatorEnabled ? 0 : 10,
- vertical: chart.inverted
- }), chart);
- addEvent(navigator.scrollbar, 'changed', function (e) {
- var range = navigator.size,
- to = range * this.to,
- from = range * this.from;
- navigator.hasDragged = navigator.scrollbar.hasDragged;
- navigator.render(0, 0, from, to);
- if (chart.options.scrollbar.liveRedraw ||
- (e.DOMType !== 'mousemove' &&
- e.DOMType !== 'touchmove')) {
- setTimeout(function () {
- navigator.onMouseUp(e);
- });
- }
- });
- }
- // Add data events
- navigator.addBaseSeriesEvents();
- // Add redraw events
- navigator.addChartEvents();
- };
- /**
- * Get the union data extremes of the chart - the outer data extremes of the
- * base X axis and the navigator axis.
- *
- * @private
- * @function Highcharts.Navigator#getUnionExtremes
- * @param {boolean} [returnFalseOnNoBaseSeries]
- * as the param says.
- * @return {Highcharts.Dictionary<(number|undefined)>|undefined}
- */
- Navigator.prototype.getUnionExtremes = function (returnFalseOnNoBaseSeries) {
- var baseAxis = this.chart.xAxis[0],
- navAxis = this.xAxis,
- navAxisOptions = navAxis.options,
- baseAxisOptions = baseAxis.options,
- ret;
- if (!returnFalseOnNoBaseSeries || baseAxis.dataMin !== null) {
- ret = {
- dataMin: pick(// #4053
- navAxisOptions && navAxisOptions.min, numExt('min', baseAxisOptions.min, baseAxis.dataMin, navAxis.dataMin, navAxis.min)),
- dataMax: pick(navAxisOptions && navAxisOptions.max, numExt('max', baseAxisOptions.max, baseAxis.dataMax, navAxis.dataMax, navAxis.max))
- };
- }
- return ret;
- };
- /**
- * Set the base series and update the navigator series from this. With a bit
- * of modification we should be able to make this an API method to be called
- * from the outside
- *
- * @private
- * @function Highcharts.Navigator#setBaseSeries
- * @param {Highcharts.SeriesOptionsType} [baseSeriesOptions]
- * Additional series options for a navigator
- * @param {boolean} [redraw]
- * Whether to redraw after update.
- * @return {void}
- */
- Navigator.prototype.setBaseSeries = function (baseSeriesOptions, redraw) {
- var chart = this.chart,
- baseSeries = this.baseSeries = [];
- baseSeriesOptions = (baseSeriesOptions ||
- chart.options && chart.options.navigator.baseSeries ||
- (chart.series.length ?
- // Find the first non-navigator series (#8430)
- find(chart.series, function (s) {
- return !s.options.isInternal;
- }).index :
- 0));
- // Iterate through series and add the ones that should be shown in
- // navigator.
- (chart.series || []).forEach(function (series, i) {
- if (
- // Don't include existing nav series
- !series.options.isInternal &&
- (series.options.showInNavigator ||
- (i === baseSeriesOptions ||
- series.options.id === baseSeriesOptions) &&
- series.options.showInNavigator !== false)) {
- baseSeries.push(series);
- }
- });
- // When run after render, this.xAxis already exists
- if (this.xAxis && !this.xAxis.navigatorAxis.fake) {
- this.updateNavigatorSeries(true, redraw);
- }
- };
- /**
- * Update series in the navigator from baseSeries, adding new if does not
- * exist.
- *
- * @private
- * @function Highcharts.Navigator.updateNavigatorSeries
- * @param {boolean} addEvents
- * @param {boolean} [redraw]
- * @return {void}
- */
- Navigator.prototype.updateNavigatorSeries = function (addEvents, redraw) {
- var navigator = this,
- chart = navigator.chart,
- baseSeries = navigator.baseSeries,
- baseOptions,
- mergedNavSeriesOptions,
- chartNavigatorSeriesOptions = navigator.navigatorOptions.series,
- baseNavigatorOptions,
- navSeriesMixin = {
- enableMouseTracking: false,
- index: null,
- linkedTo: null,
- group: 'nav',
- padXAxis: false,
- xAxis: 'navigator-x-axis',
- yAxis: 'navigator-y-axis',
- showInLegend: false,
- stacking: void 0,
- isInternal: true,
- states: {
- inactive: {
- opacity: 1
- }
- }
- },
- // Remove navigator series that are no longer in the baseSeries
- navigatorSeries = navigator.series =
- (navigator.series || []).filter(function (navSeries) {
- var base = navSeries.baseSeries;
- if (baseSeries.indexOf(base) < 0) { // Not in array
- // If there is still a base series connected to this
- // series, remove event handler and reference.
- if (base) {
- removeEvent(base, 'updatedData', navigator.updatedDataHandler);
- delete base.navigatorSeries;
- }
- // Kill the nav series. It may already have been
- // destroyed (#8715).
- if (navSeries.chart) {
- navSeries.destroy();
- }
- return false;
- }
- return true;
- });
- // Go through each base series and merge the options to create new
- // series
- if (baseSeries && baseSeries.length) {
- baseSeries.forEach(function eachBaseSeries(base) {
- var linkedNavSeries = base.navigatorSeries,
- userNavOptions = extend(
- // Grab color and visibility from base as default
- {
- color: base.color,
- visible: base.visible
- }, !isArray(chartNavigatorSeriesOptions) ?
- chartNavigatorSeriesOptions :
- defaultOptions.navigator.series);
- // Don't update if the series exists in nav and we have disabled
- // adaptToUpdatedData.
- if (linkedNavSeries &&
- navigator.navigatorOptions.adaptToUpdatedData === false) {
- return;
- }
- navSeriesMixin.name = 'Navigator ' + baseSeries.length;
- baseOptions = base.options || {};
- baseNavigatorOptions = baseOptions.navigatorOptions || {};
- mergedNavSeriesOptions = merge(baseOptions, navSeriesMixin, userNavOptions, baseNavigatorOptions);
- // Once nav series type is resolved, pick correct pointRange
- mergedNavSeriesOptions.pointRange = pick(
- // Stricte set pointRange in options
- userNavOptions.pointRange, baseNavigatorOptions.pointRange,
- // Fallback to default values, e.g. `null` for column
- defaultOptions.plotOptions[mergedNavSeriesOptions.type || 'line'].pointRange);
- // Merge data separately. Do a slice to avoid mutating the
- // navigator options from base series (#4923).
- var navigatorSeriesData = baseNavigatorOptions.data || userNavOptions.data;
- navigator.hasNavigatorData =
- navigator.hasNavigatorData || !!navigatorSeriesData;
- mergedNavSeriesOptions.data =
- navigatorSeriesData ||
- baseOptions.data && baseOptions.data.slice(0);
- // Update or add the series
- if (linkedNavSeries && linkedNavSeries.options) {
- linkedNavSeries.update(mergedNavSeriesOptions, redraw);
- }
- else {
- base.navigatorSeries = chart.initSeries(mergedNavSeriesOptions);
- base.navigatorSeries.baseSeries = base; // Store ref
- navigatorSeries.push(base.navigatorSeries);
- }
- });
- }
- // If user has defined data (and no base series) or explicitly defined
- // navigator.series as an array, we create these series on top of any
- // base series.
- if (chartNavigatorSeriesOptions.data &&
- !(baseSeries && baseSeries.length) ||
- isArray(chartNavigatorSeriesOptions)) {
- navigator.hasNavigatorData = false;
- // Allow navigator.series to be an array
- chartNavigatorSeriesOptions =
- splat(chartNavigatorSeriesOptions);
- chartNavigatorSeriesOptions.forEach(function (userSeriesOptions, i) {
- navSeriesMixin.name =
- 'Navigator ' + (navigatorSeries.length + 1);
- mergedNavSeriesOptions = merge(defaultOptions.navigator.series, {
- // Since we don't have a base series to pull color from,
- // try to fake it by using color from series with same
- // index. Otherwise pull from the colors array. We need
- // an explicit color as otherwise updates will increment
- // color counter and we'll get a new color for each
- // update of the nav series.
- color: chart.series[i] &&
- !chart.series[i].options.isInternal &&
- chart.series[i].color ||
- chart.options.colors[i] ||
- chart.options.colors[0]
- }, navSeriesMixin, userSeriesOptions);
- mergedNavSeriesOptions.data = userSeriesOptions.data;
- if (mergedNavSeriesOptions.data) {
- navigator.hasNavigatorData = true;
- navigatorSeries.push(chart.initSeries(mergedNavSeriesOptions));
- }
- });
- }
- if (addEvents) {
- this.addBaseSeriesEvents();
- }
- };
- /**
- * Add data events.
- * For example when main series is updated we need to recalculate extremes
- *
- * @private
- * @function Highcharts.Navigator#addBaseSeriesEvent
- * @return {void}
- */
- Navigator.prototype.addBaseSeriesEvents = function () {
- var navigator = this,
- baseSeries = navigator.baseSeries || [];
- // Bind modified extremes event to first base's xAxis only.
- // In event of > 1 base-xAxes, the navigator will ignore those.
- // Adding this multiple times to the same axis is no problem, as
- // duplicates should be discarded by the browser.
- if (baseSeries[0] && baseSeries[0].xAxis) {
- addEvent(baseSeries[0].xAxis, 'foundExtremes', this.modifyBaseAxisExtremes);
- }
- baseSeries.forEach(function (base) {
- // Link base series show/hide to navigator series visibility
- addEvent(base, 'show', function () {
- if (this.navigatorSeries) {
- this.navigatorSeries.setVisible(true, false);
- }
- });
- addEvent(base, 'hide', function () {
- if (this.navigatorSeries) {
- this.navigatorSeries.setVisible(false, false);
- }
- });
- // Respond to updated data in the base series, unless explicitily
- // not adapting to data changes.
- if (this.navigatorOptions.adaptToUpdatedData !== false) {
- if (base.xAxis) {
- addEvent(base, 'updatedData', this.updatedDataHandler);
- }
- }
- // Handle series removal
- addEvent(base, 'remove', function () {
- if (this.navigatorSeries) {
- erase(navigator.series, this.navigatorSeries);
- if (defined(this.navigatorSeries.options)) {
- this.navigatorSeries.remove(false);
- }
- delete this.navigatorSeries;
- }
- });
- }, this);
- };
- /**
- * Get minimum from all base series connected to the navigator
- * @private
- * @param {number} currentSeriesMin
- * Minium from the current series
- * @return {number} Minimum from all series
- */
- Navigator.prototype.getBaseSeriesMin = function (currentSeriesMin) {
- return this.baseSeries.reduce(function (min, series) {
- // (#10193)
- return Math.min(min, series.xData ? series.xData[0] : min);
- }, currentSeriesMin);
- };
- /**
- * Set the navigator x axis extremes to reflect the total. The navigator
- * extremes should always be the extremes of the union of all series in the
- * chart as well as the navigator series.
- *
- * @private
- * @function Highcharts.Navigator#modifyNavigatorAxisExtremes
- */
- Navigator.prototype.modifyNavigatorAxisExtremes = function () {
- var xAxis = this.xAxis,
- unionExtremes;
- if (typeof xAxis.getExtremes !== 'undefined') {
- unionExtremes = this.getUnionExtremes(true);
- if (unionExtremes &&
- (unionExtremes.dataMin !== xAxis.min ||
- unionExtremes.dataMax !== xAxis.max)) {
- xAxis.min = unionExtremes.dataMin;
- xAxis.max = unionExtremes.dataMax;
- }
- }
- };
- /**
- * Hook to modify the base axis extremes with information from the Navigator
- *
- * @private
- * @function Highcharts.Navigator#modifyBaseAxisExtremes
- */
- Navigator.prototype.modifyBaseAxisExtremes = function () {
- var baseXAxis = this,
- navigator = baseXAxis.chart.navigator,
- baseExtremes = baseXAxis.getExtremes(),
- baseMin = baseExtremes.min,
- baseMax = baseExtremes.max,
- baseDataMin = baseExtremes.dataMin,
- baseDataMax = baseExtremes.dataMax,
- range = baseMax - baseMin,
- stickToMin = navigator.stickToMin,
- stickToMax = navigator.stickToMax,
- overscroll = pick(baseXAxis.options.overscroll, 0),
- newMax,
- newMin,
- navigatorSeries = navigator.series && navigator.series[0],
- hasSetExtremes = !!baseXAxis.setExtremes,
- // When the extremes have been set by range selector button, don't
- // stick to min or max. The range selector buttons will handle the
- // extremes. (#5489)
- unmutable = baseXAxis.eventArgs &&
- baseXAxis.eventArgs.trigger === 'rangeSelectorButton';
- if (!unmutable) {
- // If the zoomed range is already at the min, move it to the right
- // as new data comes in
- if (stickToMin) {
- newMin = baseDataMin;
- newMax = newMin + range;
- }
- // If the zoomed range is already at the max, move it to the right
- // as new data comes in
- if (stickToMax) {
- newMax = baseDataMax + overscroll;
- // If stickToMin is true, the new min value is set above
- if (!stickToMin) {
- newMin = Math.max(baseDataMin, // don't go below data extremes (#13184)
- newMax - range, navigator.getBaseSeriesMin(navigatorSeries && navigatorSeries.xData ?
- navigatorSeries.xData[0] :
- -Number.MAX_VALUE));
- }
- }
- // Update the extremes
- if (hasSetExtremes && (stickToMin || stickToMax)) {
- if (isNumber(newMin)) {
- baseXAxis.min = baseXAxis.userMin = newMin;
- baseXAxis.max = baseXAxis.userMax = newMax;
- }
- }
- }
- // Reset
- navigator.stickToMin =
- navigator.stickToMax = null;
- };
- /**
- * Handler for updated data on the base series. When data is modified, the
- * navigator series must reflect it. This is called from the Chart.redraw
- * function before axis and series extremes are computed.
- *
- * @private
- * @function Highcharts.Navigator#updateDataHandler
- */
- Navigator.prototype.updatedDataHandler = function () {
- var navigator = this.chart.navigator,
- baseSeries = this,
- navigatorSeries = this.navigatorSeries,
- xDataMin = navigator.getBaseSeriesMin(baseSeries.xData[0]);
- // If the scrollbar is scrolled all the way to the right, keep right as
- // new data comes in.
- navigator.stickToMax = navigator.reversedExtremes ?
- Math.round(navigator.zoomedMin) === 0 :
- Math.round(navigator.zoomedMax) >= Math.round(navigator.size);
- // Detect whether the zoomed area should stick to the minimum or
- // maximum. If the current axis minimum falls outside the new updated
- // dataset, we must adjust.
- navigator.stickToMin = isNumber(baseSeries.xAxis.min) &&
- (baseSeries.xAxis.min <= xDataMin) &&
- (!this.chart.fixedRange || !navigator.stickToMax);
- // Set the navigator series data to the new data of the base series
- if (navigatorSeries && !navigator.hasNavigatorData) {
- navigatorSeries.options.pointStart = baseSeries.xData[0];
- navigatorSeries.setData(baseSeries.options.data, false, null, false); // #5414
- }
- };
- /**
- * Add chart events, like redrawing navigator, when chart requires that.
- *
- * @private
- * @function Highcharts.Navigator#addChartEvents
- * @return {void}
- */
- Navigator.prototype.addChartEvents = function () {
- if (!this.eventsToUnbind) {
- this.eventsToUnbind = [];
- }
- this.eventsToUnbind.push(
- // Move the scrollbar after redraw, like after data updata even if
- // axes don't redraw
- addEvent(this.chart, 'redraw', function () {
- var navigator = this.navigator,
- xAxis = navigator && (navigator.baseSeries &&
- navigator.baseSeries[0] &&
- navigator.baseSeries[0].xAxis ||
- this.xAxis[0]); // #5709, #13114
- if (xAxis) {
- navigator.render(xAxis.min,
- xAxis.max);
- }
- }),
- // Make room for the navigator, can be placed around the chart:
- addEvent(this.chart, 'getMargins', function () {
- var chart = this,
- navigator = chart.navigator,
- marginName = navigator.opposite ?
- 'plotTop' : 'marginBottom';
- if (chart.inverted) {
- marginName = navigator.opposite ?
- 'marginRight' : 'plotLeft';
- }
- chart[marginName] =
- (chart[marginName] || 0) + (navigator.navigatorEnabled || !chart.inverted ?
- navigator.outlineHeight :
- 0) + navigator.navigatorOptions.margin;
- }));
- };
- /**
- * Destroys allocated elements.
- *
- * @private
- * @function Highcharts.Navigator#destroy
- */
- Navigator.prototype.destroy = function () {
- // Disconnect events added in addEvents
- this.removeEvents();
- if (this.xAxis) {
- erase(this.chart.xAxis, this.xAxis);
- erase(this.chart.axes, this.xAxis);
- }
- if (this.yAxis) {
- erase(this.chart.yAxis, this.yAxis);
- erase(this.chart.axes, this.yAxis);
- }
- // Destroy series
- (this.series || []).forEach(function (s) {
- if (s.destroy) {
- s.destroy();
- }
- });
- // Destroy properties
- [
- 'series', 'xAxis', 'yAxis', 'shades', 'outline', 'scrollbarTrack',
- 'scrollbarRifles', 'scrollbarGroup', 'scrollbar', 'navigatorGroup',
- 'rendered'
- ].forEach(function (prop) {
- if (this[prop] && this[prop].destroy) {
- this[prop].destroy();
- }
- this[prop] = null;
- }, this);
- // Destroy elements in collection
- [this.handles].forEach(function (coll) {
- destroyObjectProperties(coll);
- }, this);
- };
- return Navigator;
- }());
- // End of prototype
- if (!H.Navigator) {
- H.Navigator = Navigator;
- NavigatorAxis.compose(Axis);
- // For Stock charts. For x only zooming, do not to create the zoom button
- // because X axis zooming is already allowed by the Navigator and Range
- // selector. (#9285)
- addEvent(Chart, 'beforeShowResetZoom', function () {
- var chartOptions = this.options,
- navigator = chartOptions.navigator,
- rangeSelector = chartOptions.rangeSelector;
- if (((navigator && navigator.enabled) ||
- (rangeSelector && rangeSelector.enabled)) &&
- ((!isTouchDevice && chartOptions.chart.zoomType === 'x') ||
- (isTouchDevice && chartOptions.chart.pinchType === 'x'))) {
- return false;
- }
- });
- // Initialize navigator for stock charts
- addEvent(Chart, 'beforeRender', function () {
- var options = this.options;
- if (options.navigator.enabled ||
- options.scrollbar.enabled) {
- this.scroller = this.navigator = new Navigator(this);
- }
- });
- // For stock charts, extend the Chart.setChartSize method so that we can set
- // the final top position of the navigator once the height of the chart,
- // including the legend, is determined. #367. We can't use Chart.getMargins,
- // because labels offsets are not calculated yet.
- addEvent(Chart, 'afterSetChartSize', function () {
- var legend = this.legend,
- navigator = this.navigator,
- scrollbarHeight,
- legendOptions,
- xAxis,
- yAxis;
- if (navigator) {
- legendOptions = legend && legend.options;
- xAxis = navigator.xAxis;
- yAxis = navigator.yAxis;
- scrollbarHeight = navigator.scrollbarHeight;
- // Compute the top position
- if (this.inverted) {
- navigator.left = navigator.opposite ?
- this.chartWidth - scrollbarHeight -
- navigator.height :
- this.spacing[3] + scrollbarHeight;
- navigator.top = this.plotTop + scrollbarHeight;
- }
- else {
- navigator.left = this.plotLeft + scrollbarHeight;
- navigator.top = navigator.navigatorOptions.top ||
- this.chartHeight -
- navigator.height -
- scrollbarHeight -
- this.spacing[2] -
- (this.rangeSelector && this.extraBottomMargin ?
- this.rangeSelector.getHeight() :
- 0) -
- ((legendOptions &&
- legendOptions.verticalAlign === 'bottom' &&
- legendOptions.layout !== 'proximate' && // #13392
- legendOptions.enabled &&
- !legendOptions.floating) ?
- legend.legendHeight +
- pick(legendOptions.margin, 10) :
- 0) -
- (this.titleOffset ? this.titleOffset[2] : 0);
- }
- if (xAxis && yAxis) { // false if navigator is disabled (#904)
- if (this.inverted) {
- xAxis.options.left = yAxis.options.left = navigator.left;
- }
- else {
- xAxis.options.top = yAxis.options.top = navigator.top;
- }
- xAxis.setAxisSize();
- yAxis.setAxisSize();
- }
- }
- });
- // Merge options, if no scrolling exists yet
- addEvent(Chart, 'update', function (e) {
- var navigatorOptions = (e.options.navigator || {}),
- scrollbarOptions = (e.options.scrollbar || {});
- if (!this.navigator && !this.scroller &&
- (navigatorOptions.enabled || scrollbarOptions.enabled)) {
- merge(true, this.options.navigator, navigatorOptions);
- merge(true, this.options.scrollbar, scrollbarOptions);
- delete e.options.navigator;
- delete e.options.scrollbar;
- }
- });
- // Initialize navigator, if no scrolling exists yet
- addEvent(Chart, 'afterUpdate', function (event) {
- if (!this.navigator && !this.scroller &&
- (this.options.navigator.enabled ||
- this.options.scrollbar.enabled)) {
- this.scroller = this.navigator = new Navigator(this);
- if (pick(event.redraw, true)) {
- this.redraw(event.animation); // #7067
- }
- }
- });
- // Handle adding new series
- addEvent(Chart, 'afterAddSeries', function () {
- if (this.navigator) {
- // Recompute which series should be shown in navigator, and add them
- this.navigator.setBaseSeries(null, false);
- }
- });
- // Handle updating series
- addEvent(Series, 'afterUpdate', function () {
- if (this.chart.navigator && !this.options.isInternal) {
- this.chart.navigator.setBaseSeries(null, false);
- }
- });
- Chart.prototype.callbacks.push(function (chart) {
- var extremes,
- navigator = chart.navigator;
- // Initialize the navigator
- if (navigator && chart.xAxis[0]) {
- extremes = chart.xAxis[0].getExtremes();
- navigator.render(extremes.min, extremes.max);
- }
- });
- }
- H.Navigator = Navigator;
- return H.Navigator;
- });
- _registerModule(_modules, 'masters/modules/gantt.src.js', [], function () {
- });
- _registerModule(_modules, 'masters/highcharts-gantt.src.js', [_modules['masters/highcharts.src.js']], function (Highcharts) {
- Highcharts.product = 'Highcharts Gantt';
- return Highcharts;
- });
- _modules['masters/highcharts-gantt.src.js']._modules = _modules;
- return _modules['masters/highcharts-gantt.src.js'];
- }));
|