4
0

functions.php 309 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485
  1. <?php
  2. // ===================================
  3. // Define Version
  4. define('INSTALLEDVERSION', '1.50');
  5. // ===================================
  6. use Kryptonit3\Sonarr\Sonarr;
  7. use Kryptonit3\SickRage\SickRage;
  8. // Debugging output functions
  9. function debug_out($variable, $die = false) {
  10. $trace = debug_backtrace()[0];
  11. echo "<center><img height='200px' src='images/confused.png'></center>";
  12. echo "<center>Look's like something happened, here are the errors and perhaps how to fix them:</center>";
  13. echo '<pre style="white-space: pre-line; background-color: #f2f2f2; border: 2px solid black; border-radius: 5px; padding: 5px; margin: 5px;">'.$trace['file'].':'.$trace['line']."\n\n".print_r($variable, true).'</pre>';
  14. if ($die) { http_response_code(503); die(); }
  15. }
  16. // ==== Auth Plugins START ====
  17. if (function_exists('ldap_connect')) :
  18. // Pass credentials to LDAP backend
  19. function plugin_auth_ldap($username, $password) {
  20. $ldapServers = explode(',',AUTHBACKENDHOST);
  21. foreach($ldapServers as $key => $value) {
  22. // Calculate parts
  23. $digest = parse_url(trim($value));
  24. $scheme = strtolower((isset($digest['scheme'])?$digest['scheme']:'ldap'));
  25. $host = (isset($digest['host'])?$digest['host']:(isset($digest['path'])?$digest['path']:''));
  26. $port = (isset($digest['port'])?$digest['port']:(strtolower($scheme)=='ldap'?389:636));
  27. // Reassign
  28. $ldapServers[$key] = $scheme.'://'.$host.':'.$port;
  29. }
  30. // returns true or false
  31. $ldap = ldap_connect(implode(' ',$ldapServers));
  32. if(empty(AUTHBACKENDDOMAINFORMAT)){
  33. if ($bind = ldap_bind($ldap, AUTHBACKENDDOMAIN.'\\'.$username, $password)) {
  34. writeLog("success", "LDAP authentication success");
  35. return true;
  36. } else {
  37. writeLog("error", "LDAP could not authenticate");
  38. return false;
  39. }
  40. }else{
  41. ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, 3);
  42. ldap_set_option($ldap, LDAP_OPT_REFERRALS, 0);
  43. $bind = ldap_bind($ldap, sprinf(AUTHBACKENDDOMAINFORMAT, $username), $password);
  44. if ($bind) {
  45. writeLog("success", "LDAP authentication success");
  46. return true;
  47. } else {
  48. writeLog("error", "LDPA could not authenticate");
  49. return false;
  50. }
  51. }
  52. writeLog("error", "LDAP could not authenticate");
  53. return false;
  54. }
  55. else :
  56. // Ldap Auth Missing Dependancy
  57. function plugin_auth_ldap_disabled() {
  58. return 'LDAP - Disabled (Dependancy: php-ldap missing!)';
  59. }
  60. endif;
  61. // Pass credentials to FTP backend
  62. function plugin_auth_ftp($username, $password) {
  63. // Calculate parts
  64. $digest = parse_url(AUTHBACKENDHOST);
  65. $scheme = strtolower((isset($digest['scheme'])?$digest['scheme']:(function_exists('ftp_ssl_connect')?'ftps':'ftp')));
  66. $host = (isset($digest['host'])?$digest['host']:(isset($digest['path'])?$digest['path']:''));
  67. $port = (isset($digest['port'])?$digest['port']:21);
  68. // Determine Connection Type
  69. if ($scheme == 'ftps') {
  70. $conn_id = ftp_ssl_connect($host, $port, 20);
  71. } elseif ($scheme == 'ftp') {
  72. $conn_id = ftp_connect($host, $port, 20);
  73. } else {
  74. debug_out('Invalid FTP scheme. Use ftp or ftps');
  75. writeLog("error", "invalid FTP scheme");
  76. return false;
  77. }
  78. // Check if valid FTP connection
  79. if ($conn_id) {
  80. // Attempt login
  81. @$login_result = ftp_login($conn_id, $username, $password);
  82. ftp_close($conn_id);
  83. // Return Result
  84. if ($login_result) {
  85. writeLog("success", "$username authenticated");
  86. return true;
  87. } else {
  88. writeLog("error", "$username could not authenticate");
  89. return false;
  90. }
  91. } else {
  92. return false;
  93. }
  94. return false;
  95. }
  96. // Pass credentials to Emby Backend
  97. function plugin_auth_emby_local($username, $password) {
  98. $embyAddress = qualifyURL(EMBYURL);
  99. $headers = array(
  100. 'Authorization'=> 'MediaBrowser UserId="e8837bc1-ad67-520e-8cd2-f629e3155721", Client="None", Device="Organizr", DeviceId="xxx", Version="1.0.0.0"',
  101. 'Content-Type' => 'application/json',
  102. );
  103. $body = array(
  104. 'Username' => $username,
  105. 'Password' => sha1($password),
  106. 'PasswordMd5' => md5($password),
  107. );
  108. $response = post_router($embyAddress.'/Users/AuthenticateByName', $body, $headers);
  109. if (isset($response['content'])) {
  110. $json = json_decode($response['content'], true);
  111. if (is_array($json) && isset($json['SessionInfo']) && isset($json['User']) && $json['User']['HasPassword'] == true) {
  112. // Login Success - Now Logout Emby Session As We No Longer Need It
  113. $headers = array(
  114. 'X-Mediabrowser-Token' => $json['AccessToken'],
  115. );
  116. $response = post_router($embyAddress.'/Sessions/Logout', array(), $headers);
  117. return true;
  118. }
  119. }
  120. return false;
  121. }
  122. if (function_exists('curl_version')) :
  123. // Authenticate Against Emby Local (first) and Emby Connect
  124. function plugin_auth_emby_all($username, $password) {
  125. $localResult = plugin_auth_emby_local($username, $password);
  126. if ($localResult) {
  127. return $localResult;
  128. } else {
  129. return plugin_auth_emby_connect($username, $password);
  130. }
  131. }
  132. // Authenicate against emby connect
  133. function plugin_auth_emby_connect($username, $password) {
  134. $embyAddress = qualifyURL(EMBYURL);
  135. // Get A User
  136. $connectId = '';
  137. $userIds = json_decode(file_get_contents($embyAddress.'/Users?api_key='.EMBYTOKEN),true);
  138. if (is_array($userIds)) {
  139. foreach ($userIds as $key => $value) { // Scan for this user
  140. if (isset($value['ConnectUserName']) && isset($value['ConnectUserId'])) { // Qualifty as connect account
  141. if ($value['ConnectUserName'] == $username || $value['Name'] == $username) {
  142. $connectId = $value['ConnectUserId'];
  143. break;
  144. }
  145. }
  146. }
  147. if ($connectId) {
  148. $connectURL = 'https://connect.emby.media/service/user/authenticate';
  149. $headers = array(
  150. 'Accept'=> 'application/json',
  151. 'Content-Type' => 'application/x-www-form-urlencoded',
  152. );
  153. $body = array(
  154. 'nameOrEmail' => $username,
  155. 'rawpw' => $password,
  156. );
  157. $result = curl_post($connectURL, $body, $headers);
  158. if (isset($result['content'])) {
  159. $json = json_decode($result['content'], true);
  160. if (is_array($json) && isset($json['AccessToken']) && isset($json['User']) && $json['User']['Id'] == $connectId) {
  161. return array(
  162. 'email' => $json['User']['Email'],
  163. 'image' => $json['User']['ImageUrl'],
  164. );
  165. }
  166. }
  167. }
  168. }
  169. return false;
  170. }
  171. // Pass credentials to Plex Backend
  172. function plugin_auth_plex($username, $password) {
  173. // Quick out
  174. if ((strtolower(PLEXUSERNAME) == strtolower($username)) && $password == PLEXPASSWORD) {
  175. writeLog("success", $username." authenticated by plex");
  176. return true;
  177. }
  178. //Get User List
  179. $userURL = 'https://plex.tv/pms/friends/all';
  180. $userHeaders = array(
  181. 'Authorization' => 'Basic '.base64_encode(PLEXUSERNAME.':'.PLEXPASSWORD),
  182. );
  183. libxml_use_internal_errors(true);
  184. $userXML = simplexml_load_string(curl_get($userURL, $userHeaders));
  185. if (is_array($userXML) || is_object($userXML)) {
  186. $isUser = false;
  187. $usernameLower = strtolower($username);
  188. foreach($userXML AS $child) {
  189. if(isset($child['username']) && strtolower($child['username']) == $usernameLower) {
  190. $isUser = true;
  191. writeLog("success", $usernameLower." was found in plex friends list");
  192. break;
  193. }
  194. }
  195. if ($isUser) {
  196. //Login User
  197. $connectURL = 'https://plex.tv/users/sign_in.json';
  198. $headers = array(
  199. 'Accept'=> 'application/json',
  200. 'Content-Type' => 'application/x-www-form-urlencoded',
  201. 'X-Plex-Product' => 'Organizr',
  202. 'X-Plex-Version' => '1.0',
  203. 'X-Plex-Client-Identifier' => '01010101-10101010',
  204. );
  205. $body = array(
  206. 'user[login]' => $username,
  207. 'user[password]' => $password,
  208. );
  209. $result = curl_post($connectURL, $body, $headers);
  210. if (isset($result['content'])) {
  211. $json = json_decode($result['content'], true);
  212. if (is_array($json) && isset($json['user']) && isset($json['user']['username']) && strtolower($json['user']['username']) == $usernameLower) {
  213. writeLog("success", $json['user']['username']." was logged into organizr using plex credentials");
  214. return array(
  215. 'email' => $json['user']['email'],
  216. 'image' => $json['user']['thumb']
  217. );
  218. }
  219. }
  220. }else{
  221. writeLog("error", "$username is not an authorized PLEX user or entered invalid password");
  222. }
  223. }else{
  224. writeLog("error", "error occured logging into plex might want to check curl.cainfo=/path/to/downloaded/cacert.pem in php.ini");
  225. }
  226. return false;
  227. }
  228. else :
  229. // Plex Auth Missing Dependancy
  230. function plugin_auth_plex_disabled() {
  231. return 'Plex - Disabled (Dependancy: php-curl missing!)';
  232. }
  233. // Emby Connect Auth Missing Dependancy
  234. function plugin_auth_emby_connect_disabled() {
  235. return 'Emby Connect - Disabled (Dependancy: php-curl missing!)';
  236. }
  237. // Emby Both Auth Missing Dependancy
  238. function plugin_auth_emby_both_disabled() {
  239. return 'Emby Both - Disabled (Dependancy: php-curl missing!)';
  240. }
  241. endif;
  242. // ==== Auth Plugins END ====
  243. // ==== General Class Definitions START ====
  244. $userLanguage = isset($_SERVER['HTTP_ACCEPT_LANGUAGE']) ? substr($_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2) : "en";
  245. class setLanguage {
  246. private $language = null;
  247. private $langCode = null;
  248. function __construct($language = false) {
  249. // Default
  250. if (!$language) {
  251. $language = isset($_SERVER['HTTP_ACCEPT_LANGUAGE']) ? substr($_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2) : "en";
  252. }
  253. if (!file_exists("lang/{$language}.ini")) {
  254. $language = 'en';
  255. }
  256. $this->langCode = $language;
  257. $this->language = parse_ini_file("lang/{$language}.ini", false, INI_SCANNER_RAW);
  258. if (file_exists("lang/{$language}.cust.ini")) {
  259. foreach($tmp = parse_ini_file("lang/{$language}.cust.ini", false, INI_SCANNER_RAW) as $k => $v) {
  260. $this->language[$k] = $v;
  261. }
  262. }
  263. }
  264. public function getLang() {
  265. return $this->langCode;
  266. }
  267. public function translate($originalWord) {
  268. $getArg = func_num_args();
  269. if ($getArg > 1) {
  270. $allWords = func_get_args();
  271. array_shift($allWords);
  272. } else {
  273. $allWords = array();
  274. }
  275. $translatedWord = isset($this->language[$originalWord]) ? $this->language[$originalWord] : null;
  276. if (!$translatedWord) {
  277. return ucwords(str_replace("_", " ", strtolower($originalWord)));
  278. //echo "WHOA!!!!!!! $originalWord";
  279. }
  280. $translatedWord = htmlspecialchars($translatedWord, ENT_QUOTES);
  281. return vsprintf($translatedWord, $allWords);
  282. }
  283. }
  284. $language = new setLanguage;
  285. // ==== General Class Definitions END ====
  286. // Direct request to curl if it exists, otherwise handle if not HTTPS
  287. function post_router($url, $data, $headers = array(), $referer='') {
  288. if (function_exists('curl_version')) {
  289. return curl_post($url, $data, $headers, $referer);
  290. } else {
  291. return post_request($url, $data, $headers, $referer);
  292. }
  293. }
  294. if (function_exists('curl_version')) :
  295. // Curl Post
  296. function curl_post($url, $data, $headers = array(), $referer='') {
  297. // Initiate cURL
  298. $curlReq = curl_init($url);
  299. // As post request
  300. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "POST");
  301. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  302. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  303. if(localURL($url)){
  304. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  305. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  306. }
  307. // Format Data
  308. switch (isset($headers['Content-Type'])?$headers['Content-Type']:'') {
  309. case 'application/json':
  310. curl_setopt($curlReq, CURLOPT_POSTFIELDS, json_encode($data));
  311. break;
  312. case 'application/x-www-form-urlencoded';
  313. curl_setopt($curlReq, CURLOPT_POSTFIELDS, http_build_query($data));
  314. break;
  315. default:
  316. $headers['Content-Type'] = 'application/x-www-form-urlencoded';
  317. curl_setopt($curlReq, CURLOPT_POSTFIELDS, http_build_query($data));
  318. }
  319. // Format Headers
  320. $cHeaders = array();
  321. foreach ($headers as $k => $v) {
  322. $cHeaders[] = $k.': '.$v;
  323. }
  324. if (count($cHeaders)) {
  325. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  326. }
  327. // Execute
  328. $result = curl_exec($curlReq);
  329. $httpcode = curl_getinfo($curlReq);
  330. // Close
  331. curl_close($curlReq);
  332. // Return
  333. return array('content'=>$result, 'http_code'=>$httpcode);
  334. }
  335. //Curl Get Function
  336. function curl_get($url, $headers = array()) {
  337. // Initiate cURL
  338. $curlReq = curl_init($url);
  339. // As post request
  340. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "GET");
  341. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  342. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  343. curl_setopt($curlReq, CURLOPT_CONNECTTIMEOUT, 5);
  344. if(localURL($url)){
  345. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  346. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  347. }
  348. // Format Headers
  349. $cHeaders = array();
  350. foreach ($headers as $k => $v) {
  351. $cHeaders[] = $k.': '.$v;
  352. }
  353. if (count($cHeaders)) {
  354. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  355. }
  356. // Execute
  357. $result = curl_exec($curlReq);
  358. // Close
  359. curl_close($curlReq);
  360. // Return
  361. return $result;
  362. }
  363. //Curl Delete Function
  364. function curl_delete($url, $headers = array()) {
  365. // Initiate cURL
  366. $curlReq = curl_init($url);
  367. // As post request
  368. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "DELETE");
  369. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  370. curl_setopt($curlReq, CURLOPT_CONNECTTIMEOUT, 5);
  371. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  372. if(localURL($url)){
  373. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  374. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  375. }
  376. // Format Headers
  377. $cHeaders = array();
  378. foreach ($headers as $k => $v) {
  379. $cHeaders[] = $k.': '.$v;
  380. }
  381. if (count($cHeaders)) {
  382. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  383. }
  384. // Execute
  385. $result = curl_exec($curlReq);
  386. $httpcode = curl_getinfo($curlReq);
  387. // Close
  388. curl_close($curlReq);
  389. // Return
  390. return array('content'=>$result, 'http_code'=>$httpcode);
  391. }
  392. endif;
  393. //Case-Insensitive Function
  394. function in_arrayi($needle, $haystack) {
  395. return in_array(strtolower($needle), array_map('strtolower', $haystack));
  396. }
  397. // HTTP post request (Removes need for curl, probably useless)
  398. function post_request($url, $data, $headers = array(), $referer='') {
  399. // Adapted from http://stackoverflow.com/a/28387011/6810513
  400. // Convert the data array into URL Parameters like a=b&foo=bar etc.
  401. if (isset($headers['Content-Type'])) {
  402. switch ($headers['Content-Type']) {
  403. case 'application/json':
  404. $data = json_encode($data);
  405. break;
  406. case 'application/x-www-form-urlencoded':
  407. $data = http_build_query($data);
  408. break;
  409. }
  410. } else {
  411. $headers['Content-Type'] = 'application/x-www-form-urlencoded';
  412. $data = http_build_query($data);
  413. }
  414. // parse the given URL
  415. $urlDigest = parse_url($url);
  416. // extract host and path:
  417. $host = $urlDigest['host'];
  418. $path = $urlDigest['path'];
  419. if ($urlDigest['scheme'] != 'http') {
  420. die('Error: Only HTTP request are supported, please use cURL to add HTTPS support! ('.$urlDigest['scheme'].'://'.$host.')');
  421. }
  422. // open a socket connection on port 80 - timeout: 30 sec
  423. $fp = fsockopen($host, (isset($urlDigest['port'])?':'.$urlDigest['port']:80), $errno, $errstr, 30);
  424. if ($fp){
  425. // send the request headers:
  426. fputs($fp, "POST $path HTTP/1.1\r\n");
  427. fputs($fp, "Host: $host\r\n");
  428. if ($referer != '')
  429. fputs($fp, "Referer: $referer\r\n");
  430. fputs($fp, "Content-length: ". strlen($data) ."\r\n");
  431. foreach($headers as $k => $v) {
  432. fputs($fp, $k.": ".$v."\r\n");
  433. }
  434. fputs($fp, "Connection: close\r\n\r\n");
  435. fputs($fp, $data);
  436. $result = '';
  437. while(!feof($fp)) {
  438. // receive the results of the request
  439. $result .= fgets($fp, 128);
  440. }
  441. }
  442. else {
  443. return array(
  444. 'status' => 'err',
  445. 'error' => "$errstr ($errno)"
  446. );
  447. }
  448. // close the socket connection:
  449. fclose($fp);
  450. // split the result header from the content
  451. $result = explode("\r\n\r\n", $result, 2);
  452. $header = isset($result[0]) ? $result[0] : '';
  453. $content = isset($result[1]) ? $result[1] : '';
  454. // return as structured array:
  455. return array(
  456. 'status' => 'ok',
  457. 'header' => $header,
  458. 'content' => $content,
  459. );
  460. }
  461. // Format item from Emby for Carousel
  462. function resolveEmbyItem($address, $token, $item, $nowPlaying = false, $showNames = false, $role = false, $moreInfo = false) {
  463. // Static Height
  464. $height = 444;
  465. // Get Item Details
  466. $itemDetails = json_decode(file_get_contents($address.'/Items?Ids='.$item['Id'].'&api_key='.$token),true)['Items'][0];
  467. /*if (substr_count(EMBYURL, ':') == 2) {
  468. $URL = "http://app.emby.media/itemdetails.html?id=".$itemDetails['Id'];
  469. }else{
  470. $URL = EMBYURL."/web/itemdetails.html?id=".$itemDetails['Id'];
  471. }*/
  472. $URL = EMBYURL."/web/itemdetails.html?id=".$itemDetails['Id'];
  473. switch ($itemDetails['Type']) {
  474. case 'Episode':
  475. $title = (isset($itemDetails['SeriesName'])?$itemDetails['SeriesName']:"");
  476. $imageId = (isset($itemDetails['SeriesId'])?$itemDetails['SeriesId']:$itemDetails['Id']);
  477. $width = 300;
  478. $style = '';
  479. $image = 'slick-image-tall';
  480. if(!$nowPlaying){
  481. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  482. $key = $itemDetails['Id'] . "-list";
  483. }else{
  484. $height = 281;
  485. $width = 500;
  486. $imageId = isset($itemDetails['ParentThumbItemId']) ? $itemDetails['ParentThumbItemId'] : (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : false);
  487. $imageType = isset($itemDetails['ParentThumbItemId']) ? "Thumb" : (isset($itemDetails['ParentBackdropItemId']) ? "Backdrop" : false);
  488. $key = (isset($itemDetails['ParentThumbItemId']) ? $itemDetails['ParentThumbItemId']."-np" : "none-np");
  489. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  490. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  491. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  492. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  493. $stream = $moreInfo['PlayState']['PlayMethod'];
  494. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  495. $id = $moreInfo['DeviceId'];
  496. $streamInfo = buildStream(array(
  497. 'platform' => (string) $moreInfo['Client'],
  498. 'device' => (string) $moreInfo['DeviceName'],
  499. 'stream' => streamType($stream),
  500. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  501. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  502. ));
  503. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  504. $topTitle = '<h5 class="text-center zero-m elip">'.$title.' - '.$itemDetails['Name'].'</h5>';
  505. $bottomTitle = '<small class="zero-m">S'.$itemDetails['ParentIndexNumber'].' · E'.$itemDetails['IndexNumber'].'</small>';
  506. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  507. }
  508. break;
  509. case 'MusicAlbum':
  510. case 'Audio':
  511. $title = $itemDetails['Name'];
  512. $imageId = $itemDetails['Id'];
  513. $width = 444;
  514. $style = '';
  515. $image = 'slick-image-short';
  516. if(!$nowPlaying){
  517. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  518. $key = $itemDetails['Id'] . "-list";
  519. }else{
  520. $height = 281;
  521. $width = 500;
  522. $imageId = (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : false);
  523. $imageType = (isset($itemDetails['ParentBackdropItemId']) ? "Backdrop" : false);
  524. $key = (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : "no-np") . "-np";
  525. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  526. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  527. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  528. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  529. $stream = $moreInfo['PlayState']['PlayMethod'];
  530. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  531. $id = $moreInfo['DeviceId'];
  532. $streamInfo = buildStream(array(
  533. 'platform' => (string) $moreInfo['Client'],
  534. 'device' => (string) $moreInfo['DeviceName'],
  535. 'stream' => streamType($stream),
  536. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  537. ));
  538. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  539. $topTitle = '<h5 class="text-center zero-m elip">'.$itemDetails['AlbumArtist'].' - '.$itemDetails['Album'].'</h5>';
  540. $bottomTitle = '<small class="zero-m">'.$title.'</small>';
  541. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  542. }
  543. break;
  544. case 'TvChannel':
  545. $title = $itemDetails['CurrentProgram']['Name'];
  546. $imageId = $itemDetails['Id'];
  547. $width = 300;
  548. $style = '';
  549. $image = 'slick-image-tall';
  550. if(!$nowPlaying){
  551. $imageType = "Primary";
  552. $key = $itemDetails['Id'] . "-list";
  553. }else{
  554. $height = 281;
  555. $width = 500;
  556. $imageType = "Thumb";
  557. $key = $itemDetails['Id'] . "-np";
  558. $useImage = "images/livetv.png";
  559. $watched = "0";
  560. $stream = $moreInfo['PlayState']['PlayMethod'];
  561. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  562. $id = $moreInfo['DeviceId'];
  563. $streamInfo = buildStream(array(
  564. 'platform' => (string) $moreInfo['Client'],
  565. 'device' => (string) $moreInfo['DeviceName'],
  566. 'stream' => streamType($stream),
  567. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  568. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  569. ));
  570. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  571. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  572. $bottomTitle = '<small class="zero-m">'.$itemDetails['Name'].' - '.$itemDetails['ChannelNumber'].'</small>';
  573. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  574. }
  575. break;
  576. default:
  577. $title = $itemDetails['Name'];
  578. $imageId = $itemDetails['Id'];
  579. $width = 300;
  580. $style = '';
  581. $image = 'slick-image-tall';
  582. if(!$nowPlaying){
  583. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  584. $key = $itemDetails['Id'] . "-list";
  585. }else{
  586. $height = 281;
  587. $width = 500;
  588. $imageType = isset($itemDetails['ImageTags']['Thumb']) ? "Thumb" : (isset($itemDetails['BackdropImageTags']) ? "Backdrop" : false);
  589. $key = $itemDetails['Id'] . "-np";
  590. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  591. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  592. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  593. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  594. $stream = $moreInfo['PlayState']['PlayMethod'];
  595. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  596. $id = $moreInfo['DeviceId'];
  597. $streamInfo = buildStream(array(
  598. 'platform' => (string) $moreInfo['Client'],
  599. 'device' => (string) $moreInfo['DeviceName'],
  600. 'stream' => streamType($stream),
  601. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  602. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  603. ));
  604. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  605. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  606. $bottomTitle = '<small class="zero-m">'.$moreInfo['NowPlayingItem']['ProductionYear'].'</small>';
  607. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  608. }
  609. }
  610. // If No Overview
  611. if (!isset($itemDetails['Overview'])) {
  612. $itemDetails['Overview'] = '';
  613. }
  614. if (file_exists('images/cache/'.$key.'.jpg')){ $image_url = 'images/cache/'.$key.'.jpg'; }
  615. if (file_exists('images/cache/'.$key.'.jpg') && (time() - 604800) > filemtime('images/cache/'.$key.'.jpg') || !file_exists('images/cache/'.$key.'.jpg')) {
  616. $image_url = 'ajax.php?a=emby-image&type='.$imageType.'&img='.$imageId.'&height='.$height.'&width='.$width.'&key='.$key.'';
  617. }
  618. if($nowPlaying){
  619. if(!$imageType){ $image_url = "images/no-np.png"; $key = "no-np"; }
  620. if(!$imageId){ $image_url = "images/no-np.png"; $key = "no-np"; }
  621. }else{
  622. if(!$imageType){ $image_url = "images/no-list.png"; $key = "no-list"; }
  623. if(!$imageId){ $image_url = "images/no-list.png"; $key = "no-list"; }
  624. }
  625. if(isset($useImage)){ $image_url = $useImage; }
  626. // Assemble Item And Cache Into Array
  627. if($nowPlaying){
  628. //prettyPrint($itemDetails);
  629. return '<div class="col-sm-6 col-md-3"><div class="thumbnail ultra-widget"><div style="display: none;" np="'.$id.'" class="overlay content-box small-box gray-bg">'.$streamInfo.'</div><span class="w-refresh w-p-icon gray" link="'.$id.'"><span class="fa-stack fa-lg" style="font-size: .5em"><i class="fa fa-square fa-stack-2x"></i><i class="fa fa-info-circle fa-stack-1x fa-inverse"></i></span></span><a href="'.$URL.'" target="_blank"><img style="width: 100%; display:inherit;" src="'.$image_url.'" alt="'.$itemDetails['Name'].'"></a><div class="progress progress-bar-sm zero-m"><div class="progress-bar progress-bar-success" role="progressbar" aria-valuenow="'.$watched.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$watched.'%"></div><div class="progress-bar palette-Grey-500 bg" style="width: 0%"></div></div><div class="caption"><i style="float:left" class="fa fa-'.$state.'"></i>'.$topTitle.''.$bottomTitle.'</div></div></div>';
  630. }else{
  631. return '<div class="item-'.$itemDetails['Type'].'"><a href="'.$URL.'" target="_blank"><img alt="'.$itemDetails['Name'].'" class="'.$image.'" data-lazy="'.$image_url.'"></a><small style="margin-right: 13px" class="elip">'.$title.'</small></div>';
  632. }
  633. }
  634. // Format item from Plex for Carousel
  635. function resolvePlexItem($server, $token, $item, $nowPlaying = false, $showNames = false, $role = false, $playlist = false) {
  636. // Static Height
  637. $height = 444;
  638. $playlist = ($playlist) ? " playlist-$playlist" : "";
  639. switch ($item['type']) {
  640. case 'season':
  641. $title = $item['parentTitle'];
  642. $summary = $item['parentSummary'];
  643. $width = 300;
  644. $image = 'slick-image-tall';
  645. $style = '';
  646. if(!$nowPlaying){
  647. $thumb = $item['thumb'];
  648. $key = $item['ratingKey'] . "-list";
  649. }else {
  650. $height = 281;
  651. $width = 500;
  652. $thumb = $item['art'];
  653. $key = $item['ratingKey'] . "-np";
  654. $elapsed = $item['viewOffset'];
  655. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  656. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  657. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  658. $stream = $item->Media->Part->Stream['decision'];
  659. $user = $role == "admin" ? $item->User['title'] : "";
  660. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  661. $streamInfo = buildStream(array(
  662. 'platform' => (string) $item->Player['platform'],
  663. 'device' => (string) $item->Player['device'],
  664. 'stream' => streamType($item->Media->Part['decision']),
  665. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  666. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  667. ));
  668. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  669. }
  670. break;
  671. case 'episode':
  672. $title = $item['grandparentTitle'];
  673. $summary = $item['title'];
  674. $width = 300;
  675. $image = 'slick-image-tall';
  676. $style = '';
  677. if(!$nowPlaying){
  678. $thumb = ($item['parentThumb'] ? $item['parentThumb'] : $item['grandparentThumb']);
  679. $key = $item['ratingKey'] . "-list";
  680. }else {
  681. $height = 281;
  682. $width = 500;
  683. $thumb = $item['art'];
  684. $key = $item['ratingKey'] . "-np";
  685. $elapsed = $item['viewOffset'];
  686. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  687. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  688. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  689. $stream = $item->Media->Part->Stream['decision'];
  690. $user = $role == "admin" ? $item->User['title'] : "";
  691. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  692. $streamInfo = buildStream(array(
  693. 'platform' => (string) $item->Player['platform'],
  694. 'device' => (string) $item->Player['device'],
  695. 'stream' => streamType($item->Media->Part['decision']),
  696. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  697. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  698. ));
  699. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  700. $topTitle = '<h5 class="text-center zero-m elip">'.$title.' - '.$item['title'].'</h5>';
  701. $bottomTitle = '<small class="zero-m">S'.$item['parentIndex'].' · E'.$item['index'].'</small>';
  702. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  703. }
  704. break;
  705. case 'clip':
  706. $title = $item['title'];
  707. $summary = $item['summary'];
  708. $width = 300;
  709. $image = 'slick-image-tall';
  710. $style = '';
  711. if(!$nowPlaying){
  712. $thumb = $item['thumb'];
  713. $key = $item['ratingKey'] . "-list";
  714. }else {
  715. $height = 281;
  716. $width = 500;
  717. $thumb = $item['art'];
  718. $key = isset($item['ratingKey']) ? $item['ratingKey'] . "-np" : (isset($item['live']) ? "livetv.png" : ":)");
  719. $useImage = (isset($item['live']) ? "images/livetv.png" : null);
  720. $extraInfo = isset($item['extraType']) ? "Trailer" : (isset($item['live']) ? "Live TV" : ":)");
  721. $elapsed = $item['viewOffset'];
  722. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  723. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  724. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  725. $stream = $item->Media->Part->Stream['decision'];
  726. $user = $role == "admin" ? $item->User['title'] : "";
  727. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  728. $streamInfo = buildStream(array(
  729. 'platform' => (string) $item->Player['platform'],
  730. 'device' => (string) $item->Player['device'],
  731. 'stream' => streamType($item->Media->Part['decision']),
  732. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  733. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  734. ));
  735. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  736. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  737. $bottomTitle = '<small class="zero-m">'.$extraInfo.'</small>';
  738. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  739. }
  740. break;
  741. case 'album':
  742. case 'track':
  743. $title = $item['parentTitle'];
  744. $summary = $item['title'];
  745. $image = 'slick-image-short';
  746. $style = 'left: 160px !important;';
  747. $item['ratingKey'] = $item['parentRatingKey'];
  748. if(!$nowPlaying){
  749. $width = 444;
  750. $thumb = $item['thumb'];
  751. $key = $item['ratingKey'] . "-list";
  752. }else {
  753. $height = 281;
  754. $width = 500;
  755. $thumb = $item['art'];
  756. $key = $item['ratingKey'] . "-np";
  757. $elapsed = $item['viewOffset'];
  758. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  759. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  760. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  761. $stream = $item->Media->Part->Stream['decision'];
  762. $user = $role == "admin" ? $item->User['title'] : "";
  763. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  764. $streamInfo = buildStream(array(
  765. 'platform' => (string) $item->Player['platform'],
  766. 'device' => (string) $item->Player['device'],
  767. 'stream' => streamType($item->Media->Part['decision']),
  768. 'audio' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['channels']."ch)",
  769. ));
  770. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  771. $topTitle = '<h5 class="text-center zero-m elip">'.$item['grandparentTitle'].' - '.$item['title'].'</h5>';
  772. $bottomTitle = '<small class="zero-m">'.$title.'</small>';
  773. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  774. }
  775. break;
  776. default:
  777. $title = $item['title'];
  778. $summary = $item['summary'];
  779. $image = 'slick-image-tall';
  780. $style = '';
  781. if(!$nowPlaying){
  782. $width = 300;
  783. $thumb = $item['thumb'];
  784. $key = $item['ratingKey'] . "-list";
  785. }else {
  786. $height = 281;
  787. $width = 500;
  788. $thumb = $item['art'];
  789. $key = $item['ratingKey'] . "-np";
  790. $elapsed = $item['viewOffset'];
  791. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  792. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  793. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  794. $stream = $item->Media->Part->Stream['decision'];
  795. $user = $role == "admin" ? $item->User['title'] : "";
  796. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  797. $streamInfo = buildStream(array(
  798. 'platform' => (string) $item->Player['platform'],
  799. 'device' => (string) $item->Player['device'],
  800. 'stream' => streamType($item->Media->Part['decision']),
  801. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  802. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  803. ));
  804. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  805. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  806. $bottomTitle = '<small class="zero-m">'.$item['year'].'</small>';
  807. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  808. }
  809. }
  810. if (substr_count(PLEXURL, '.') != 2) {
  811. $address = "https://app.plex.tv/web/app#!/server/$server/details?key=/library/metadata/".$item['ratingKey'];
  812. }else{
  813. $address = PLEXURL."/web/index.html#!/server/$server/details?key=/library/metadata/".$item['ratingKey'];
  814. }
  815. // If No Overview
  816. if (!isset($itemDetails['Overview'])) { $itemDetails['Overview'] = ''; }
  817. if (file_exists('images/cache/'.$key.'.jpg')){ $image_url = 'images/cache/'.$key.'.jpg'; }
  818. if (file_exists('images/cache/'.$key.'.jpg') && (time() - 604800) > filemtime('images/cache/'.$key.'.jpg') || !file_exists('images/cache/'.$key.'.jpg')) {
  819. $image_url = 'ajax.php?a=plex-image&img='.$thumb.'&height='.$height.'&width='.$width.'&key='.$key.'';
  820. }
  821. if($nowPlaying){
  822. if(!$thumb){ $image_url = "images/no-np.png"; $key = "no-np"; }
  823. }else{
  824. if(!$thumb){ $image_url = "images/no-list.png"; $key = "no-list"; }
  825. }
  826. if(isset($useImage)){ $image_url = $useImage; }
  827. $openTab = (PLEXTABNAME) ? "true" : "false";
  828. // Assemble Item And Cache Into Array
  829. if($nowPlaying){
  830. return '<div class="col-sm-6 col-md-3"><div class="thumbnail ultra-widget"><div style="display: none;" np="'.$id.'" class="overlay content-box small-box gray-bg">'.$streamInfo.'</div><span class="w-refresh w-p-icon gray" link="'.$id.'"><span class="fa-stack fa-lg" style="font-size: .5em"><i class="fa fa-square fa-stack-2x"></i><i class="fa fa-info-circle fa-stack-1x fa-inverse"></i></span></span><a class="openTab" extraTitle="'.$title.'" extraType="'.$item['type'].'" openTab="'.$openTab.'" href="'.$address.'" target="_blank"><img style="width: 100%; display:inherit;" src="'.$image_url.'" alt="'.$item['Name'].'"></a><div class="progress progress-bar-sm zero-m"><div class="progress-bar progress-bar-success" role="progressbar" aria-valuenow="'.$watched.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$watched.'%"></div><div class="progress-bar palette-Grey-500 bg" style="width: '.$transcoded.'%"></div></div><div class="caption"><i style="float:left" class="fa fa-'.$state.'"></i>'.$topTitle.''.$bottomTitle.'</div></div></div>';
  831. }else{
  832. return '<div class="item-'.$item['type'].$playlist.'"><a class="openTab" extraTitle="'.$title.'" extraType="'.$item['type'].'" openTab="'.$openTab.'" href="'.$address.'" target="_blank"><img alt="'.$item['Name'].'" class="'.$image.'" data-lazy="'.$image_url.'"></a><small style="margin-right: 13px" class="elip">'.$title.'</small></div>';
  833. }
  834. }
  835. //$hideMenu .= '<li data-filter="playlist-'.$className.'" data-name="'.$api['title'].'"><a class="js-filter-'.$className.'" href="javascript:void(0)">'.$api['title'].'</a></li>';
  836. //Recent Added
  837. function outputRecentAdded($header, $items, $script = false, $array) {
  838. $hideMenu = '<div class="pull-right"><div class="btn-group" role="group"><button type="button" class="btn waves btn-default btn-sm dropdown-toggle waves-effect waves-float" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">Filter &nbsp;<span class="caret"></span></button><ul style="right:0; left: auto" class="dropdown-menu filter-recent-event">';
  839. if(preg_grep("/item-movie/", $items)){
  840. $hideMenu .= '<li data-filter="item-movie" data-name="Movies" data-filter-on="false"><a class="js-filter-movie" href="javascript:void(0)">Movies</a></li>';
  841. }
  842. if(preg_grep("/item-season/", $items)){
  843. $hideMenu .= '<li data-filter="item-season" data-name="TV Shows" data-filter-on="false"><a class="js-filter-season" href="javascript:void(0)">Shows</a></li>';
  844. }
  845. if(preg_grep("/item-album/", $items)){
  846. $hideMenu .= '<li data-filter="item-album" data-name="Music Albums" data-filter-on="false"><a class="js-filter-album" href="javascript:void(0)">Music</a></li>';
  847. }
  848. $hideMenu .= '<li data-filter="item-all" data-name="Content" data-filter-on="false"><a class="js-filter-all" href="javascript:void(0)">All</a></li>';
  849. $hideMenu .= '</ul></div></div>';
  850. // If None Populate Empty Item
  851. if (!count($items)) {
  852. return '<div id="recentMedia" class="content-box box-shadow big-box"><h5 class="text-center">'.$header.'</h5><p class="text-center">No Media Found</p></div>';
  853. }else{
  854. $className = str_replace(' ', '', $header);
  855. return '<div id="recentMedia" class="content-box box-shadow big-box"><h5 id="recentContent-title" style="margin-bottom: -20px" class="text-center">'.$header.'</h5><div class="recentHeader inbox-pagination '.$className.'">'.$hideMenu.'</div><br/><br/><div class="recentItems-recent" data-name="'.$className.'">'.implode('',$items).'</div></div>'.($script?'<script>'.$script.'</script>':'');
  856. }
  857. }
  858. // Create Carousel
  859. function outputNowPlaying($header, $size, $type, $items, $script = false) {
  860. // If None Populate Empty Item
  861. if (!count($items)) {
  862. return '<div id=streamz></div>'.($script?'<script>'.$script.'</script>':'');
  863. }else{
  864. return '<div id=streamz><h5 class="zero-m big-box"><strong>'.$header.'</strong></h5>'.implode('',$items).'</div>'.($script?'<script>'.$script.'</script>':'');
  865. }
  866. }
  867. // Get Now Playing Streams From Emby
  868. function getEmbyStreams($size, $showNames, $role) {
  869. $address = qualifyURL(EMBYURL);
  870. $api = json_decode(@file_get_contents($address.'/Sessions?api_key='.EMBYTOKEN),true);
  871. if (!is_array($api)) { return 'Could not load!'; }
  872. $playingItems = array();
  873. foreach($api as $key => $value) {
  874. if (isset($value['NowPlayingItem'])) {
  875. $playingItems[] = resolveEmbyItem($address, EMBYTOKEN, $value['NowPlayingItem'], true, $showNames, $role, $value);
  876. }
  877. }
  878. return outputNowPlaying(translate('PLAYING_NOW_ON_EMBY'), $size, 'streams-emby', $playingItems, "
  879. setInterval(function() {
  880. $('<div></div>').load('ajax.php?a=emby-streams',function() {
  881. var element = $(this).find('[id]');
  882. var loadedID = element.attr('id');
  883. $('#'+loadedID).replaceWith(element);
  884. console.log('Loaded updated: '+loadedID);
  885. });
  886. }, 15000);
  887. ");
  888. }
  889. // Get Now Playing Streams From Plex
  890. function getPlexStreams($size, $showNames, $role){
  891. $address = qualifyURL(PLEXURL);
  892. // Perform API requests
  893. $api = @curl_get($address."/status/sessions?X-Plex-Token=".PLEXTOKEN);
  894. libxml_use_internal_errors(true);
  895. $api = simplexml_load_string($api);
  896. if (is_array($api) || is_object($api)){
  897. if (!$api->head->title){
  898. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  899. if (!$getServer) { return 'Could not load!'; }
  900. // Identify the local machine
  901. $gotServer = $getServer['machineIdentifier'];
  902. $items = array();
  903. foreach($api AS $child) {
  904. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, true, $showNames, $role);
  905. }
  906. return outputNowPlaying(translate('PLAYING_NOW_ON_PLEX')." ( ".count($items)." Streams )", $size, 'streams-plex', $items, "
  907. setInterval(function() {
  908. $('<div></div>').load('ajax.php?a=plex-streams',function() {
  909. var element = $(this).find('[id]');
  910. var loadedID = element.attr('id');
  911. $('#'+loadedID).replaceWith(element);
  912. console.log('Loaded updated: '+loadedID);
  913. });
  914. }, 15000);
  915. ");
  916. }else{
  917. writeLog("error", "PLEX STREAM ERROR: could not connect - check token - if HTTPS, is cert valid");
  918. }
  919. }else{
  920. writeLog("error", "PLEX STREAM ERROR: could not connect - check URL - if HTTPS, is cert valid");
  921. }
  922. }
  923. // Get Recent Content From Emby
  924. function getEmbyRecent($array) {
  925. $address = qualifyURL(EMBYURL);
  926. $header = translate('RECENT_CONTENT');
  927. // Currently Logged In User
  928. $username = false;
  929. if (isset($GLOBALS['USER'])) {
  930. $username = strtolower($GLOBALS['USER']->username);
  931. }
  932. // Get A User
  933. $userIds = json_decode(@file_get_contents($address.'/Users?api_key='.EMBYTOKEN),true);
  934. if (!is_array($userIds)) { return 'Could not load!'; }
  935. $showPlayed = true;
  936. foreach ($userIds as $value) { // Scan for admin user
  937. if (isset($value['Policy']) && isset($value['Policy']['IsAdministrator']) && $value['Policy']['IsAdministrator']) {
  938. $userId = $value['Id'];
  939. }
  940. if ($username && strtolower($value['Name']) == $username) {
  941. $userId = $value['Id'];
  942. $showPlayed = false;
  943. break;
  944. }
  945. }
  946. // Get the latest Items
  947. $latest = json_decode(file_get_contents($address.'/Users/'.$userId.'/Items/Latest?EnableImages=false&Limit='.EMBYRECENTITEMS.'&api_key='.EMBYTOKEN.($showPlayed?'':'&IsPlayed=false')),true);
  948. // For Each Item In Category
  949. $items = array();
  950. foreach ($latest as $k => $v) {
  951. $type = (string) $v['Type'];
  952. if(@$array[$type] == "true"){
  953. $items[] = resolveEmbyItem($address, EMBYTOKEN, $v, false, false, false);
  954. }
  955. }
  956. $array["movie"] = $array["Movie"];
  957. $array["season"] = $array["Episode"];
  958. $array["album"] = $array["MusicAlbum"];
  959. unset($array["Movie"]);
  960. unset($array["Episode"]);
  961. unset($array["MusicAlbum"]);
  962. unset($array["Series"]);
  963. return outputRecentAdded($header, $items, "", $array);
  964. }
  965. // Get Recent Content From Plex
  966. function getPlexRecent($array){
  967. $address = qualifyURL(PLEXURL);
  968. $header = translate('RECENT_CONTENT');
  969. // Perform Requests
  970. $api = @curl_get($address."/library/recentlyAdded?limit=".PLEXRECENTITEMS."&X-Plex-Token=".PLEXTOKEN);
  971. libxml_use_internal_errors(true);
  972. $api = simplexml_load_string($api);
  973. if (is_array($api) || is_object($api)){
  974. if (!$api->head->title){
  975. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  976. if (!$getServer) { return 'Could not load!'; }
  977. // Identify the local machine
  978. $gotServer = $getServer['machineIdentifier'];
  979. $items = array();
  980. foreach($api AS $child) {
  981. $type = (string) $child['type'];
  982. if($array[$type] == "true"){
  983. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, false, false, false);
  984. }
  985. }
  986. return outputRecentAdded($header, $items, "", $array);
  987. }else{
  988. writeLog("error", "PLEX RECENT-ITEMS ERROR: could not connect - check token - if HTTPS, is cert valid");
  989. }
  990. }else{
  991. writeLog("error", "PLEX RECENT-ITEMS ERROR: could not connect - check URL - if HTTPS, is cert valid");
  992. }
  993. }
  994. // Get Image From Emby
  995. function getEmbyImage() {
  996. $embyAddress = qualifyURL(EMBYURL);
  997. if (!file_exists('images/cache')) {
  998. mkdir('images/cache', 0777, true);
  999. }
  1000. $itemId = $_GET['img'];
  1001. $key = $_GET['key'];
  1002. $itemType = $_GET['type'];
  1003. $imgParams = array();
  1004. if (isset($_GET['height'])) { $imgParams['height'] = 'maxHeight='.$_GET['height']; }
  1005. if (isset($_GET['width'])) { $imgParams['width'] = 'maxWidth='.$_GET['width']; }
  1006. if(isset($itemId)) {
  1007. $image_src = $embyAddress . '/Items/'.$itemId.'/Images/'.$itemType.'?'.implode('&', $imgParams);
  1008. $cachefile = 'images/cache/'.$key.'.jpg';
  1009. $cachetime = 604800;
  1010. // Serve from the cache if it is younger than $cachetime
  1011. if (file_exists($cachefile) && time() - $cachetime < filemtime($cachefile)) {
  1012. header("Content-type: image/jpeg");
  1013. @readfile($cachefile);
  1014. exit;
  1015. }
  1016. ob_start(); // Start the output buffer
  1017. header('Content-type: image/jpeg');
  1018. @readfile($image_src);
  1019. // Cache the output to a file
  1020. $fp = fopen($cachefile, 'wb');
  1021. fwrite($fp, ob_get_contents());
  1022. fclose($fp);
  1023. ob_end_flush(); // Send the output to the browser
  1024. die();
  1025. } else {
  1026. debug_out('Invalid Request',1);
  1027. }
  1028. }
  1029. // Get Image From Plex
  1030. function getPlexImage() {
  1031. $plexAddress = qualifyURL(PLEXURL);
  1032. if (!file_exists('images/cache')) {
  1033. mkdir('images/cache', 0777, true);
  1034. }
  1035. $image_url = $_GET['img'];
  1036. $key = $_GET['key'];
  1037. $image_height = $_GET['height'];
  1038. $image_width = $_GET['width'];
  1039. if(isset($image_url) && isset($image_height) && isset($image_width)) {
  1040. $image_src = $plexAddress . '/photo/:/transcode?height='.$image_height.'&width='.$image_width.'&upscale=1&url=' . $image_url . '&X-Plex-Token=' . PLEXTOKEN;
  1041. $cachefile = 'images/cache/'.$key.'.jpg';
  1042. $cachetime = 604800;
  1043. // Serve from the cache if it is younger than $cachetime
  1044. if (file_exists($cachefile) && time() - $cachetime < filemtime($cachefile)) {
  1045. header("Content-type: image/jpeg");
  1046. @readfile($cachefile);
  1047. exit;
  1048. }
  1049. ob_start(); // Start the output buffer
  1050. header('Content-type: image/jpeg');
  1051. @readfile($image_src);
  1052. // Cache the output to a file
  1053. $fp = fopen($cachefile, 'wb');
  1054. fwrite($fp, ob_get_contents());
  1055. fclose($fp);
  1056. ob_end_flush(); // Send the output to the browser
  1057. die();
  1058. } else {
  1059. echo "Invalid Plex Request";
  1060. }
  1061. }
  1062. // Simplier access to class
  1063. function translate($string) {
  1064. if (isset($GLOBALS['language'])) {
  1065. return $GLOBALS['language']->translate($string);
  1066. } else {
  1067. return '!Translations Not Loaded!';
  1068. }
  1069. }
  1070. // Generate Random string
  1071. function randString($length = 10, $chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ') {
  1072. $tmp = '';
  1073. for ($i = 0; $i < $length; $i++) {
  1074. $tmp .= substr(str_shuffle($chars), 0, 1);
  1075. }
  1076. return $tmp;
  1077. }
  1078. // Create config file in the return syntax
  1079. function createConfig($array, $path = 'config/config.php', $nest = 0) {
  1080. // Define Initial Value
  1081. $output = array();
  1082. // Sort Items
  1083. ksort($array);
  1084. // Update the current config version
  1085. if (!$nest) {
  1086. // Inject Current Version
  1087. $output[] = "\t'CONFIG_VERSION' => '".(isset($array['apply_CONFIG_VERSION'])?$array['apply_CONFIG_VERSION']:INSTALLEDVERSION)."'";
  1088. }
  1089. unset($array['CONFIG_VERSION']);
  1090. unset($array['apply_CONFIG_VERSION']);
  1091. // Process Settings
  1092. foreach ($array as $k => $v) {
  1093. $allowCommit = true;
  1094. switch (gettype($v)) {
  1095. case 'boolean':
  1096. $item = ($v?'true':'false');
  1097. break;
  1098. case 'integer':
  1099. case 'double':
  1100. case 'integer':
  1101. case 'NULL':
  1102. $item = $v;
  1103. break;
  1104. case 'string':
  1105. $item = "'".str_replace(array('\\',"'"),array('\\\\',"\'"),$v)."'";
  1106. break;
  1107. case 'array':
  1108. $item = createConfig($v, false, $nest+1);
  1109. break;
  1110. default:
  1111. $allowCommit = false;
  1112. }
  1113. if($allowCommit) {
  1114. $output[] = str_repeat("\t",$nest+1)."'$k' => $item";
  1115. }
  1116. }
  1117. // Build output
  1118. $output = (!$nest?"<?php\nreturn ":'')."array(\n".implode(",\n",$output)."\n".str_repeat("\t",$nest).')'.(!$nest?';':'');
  1119. if (!$nest && $path) {
  1120. $pathDigest = pathinfo($path);
  1121. @mkdir($pathDigest['dirname'], 0770, true);
  1122. if (file_exists($path)) {
  1123. rename($path, $pathDigest['dirname'].'/'.$pathDigest['filename'].'.bak.php');
  1124. }
  1125. $file = fopen($path, 'w');
  1126. fwrite($file, $output);
  1127. fclose($file);
  1128. if (file_exists($path)) {
  1129. return true;
  1130. }
  1131. writeLog("error", "config was unable to write");
  1132. return false;
  1133. } else {
  1134. writeLog("success", "config was updated with new values");
  1135. return $output;
  1136. }
  1137. }
  1138. // Load a config file written in the return syntax
  1139. function loadConfig($path = 'config/config.php') {
  1140. // Adapted from http://stackoverflow.com/a/14173339/6810513
  1141. if (!is_file($path)) {
  1142. return null;
  1143. } else {
  1144. return (array) call_user_func(function() use($path) {
  1145. return include($path);
  1146. });
  1147. }
  1148. }
  1149. // Commit new values to the configuration
  1150. function updateConfig($new, $current = false) {
  1151. // Get config if not supplied
  1152. if ($current === false) {
  1153. $current = loadConfig();
  1154. } else if (is_string($current) && is_file($current)) {
  1155. $current = loadConfig($current);
  1156. }
  1157. // Inject Parts
  1158. foreach ($new as $k => $v) {
  1159. $current[$k] = $v;
  1160. }
  1161. // Return Create
  1162. return createConfig($current);
  1163. }
  1164. // Inject Defaults As Needed
  1165. function fillDefaultConfig($array, $path = 'config/configDefaults.php') {
  1166. if (is_string($path)) {
  1167. $loadedDefaults = loadConfig($path);
  1168. } else {
  1169. $loadedDefaults = $path;
  1170. }
  1171. return (is_array($loadedDefaults) ? fillDefaultConfig_recurse($array, $loadedDefaults) : false);
  1172. }
  1173. // support function for fillDefaultConfig()
  1174. function fillDefaultConfig_recurse($current, $defaults) {
  1175. foreach($defaults as $k => $v) {
  1176. if (!isset($current[$k])) {
  1177. $current[$k] = $v;
  1178. } else if (is_array($current[$k]) && is_array($v)) {
  1179. $current[$k] = fillDefaultConfig_recurse($current[$k], $v);
  1180. }
  1181. }
  1182. return $current;
  1183. };
  1184. // Define Scalar Variables (nest non-secular with underscores)
  1185. function defineConfig($array, $anyCase = true, $nest_prefix = false) {
  1186. foreach($array as $k => $v) {
  1187. if (is_scalar($v) && !defined($nest_prefix.$k)) {
  1188. define($nest_prefix.$k, $v, $anyCase);
  1189. } else if (is_array($v)) {
  1190. defineConfig($v, $anyCase, $nest_prefix.$k.'_');
  1191. }
  1192. }
  1193. }
  1194. // This function exists only because I am lazy
  1195. function configLazy($path = 'config/config.php') {
  1196. // Load config or default
  1197. if (file_exists($path)) {
  1198. $config = fillDefaultConfig(loadConfig($path));
  1199. } else {
  1200. $config = loadConfig('config/configDefaults.php');
  1201. }
  1202. if (is_array($config)) {
  1203. defineConfig($config);
  1204. }
  1205. return $config;
  1206. }
  1207. // Qualify URL
  1208. function qualifyURL($url) {
  1209. //local address?
  1210. if(substr($url, 0,1) == "/"){
  1211. if (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
  1212. $protocol = "https://";
  1213. } else {
  1214. $protocol = "http://";
  1215. }
  1216. $url = $protocol.getServer().$url;
  1217. }
  1218. // Get Digest
  1219. $digest = parse_url($url);
  1220. // http/https
  1221. if (!isset($digest['scheme'])) {
  1222. if (isset($digest['port']) && in_array($digest['port'], array(80,8080,8096,32400,7878,8989,8182,8081,6789))) {
  1223. $scheme = 'http';
  1224. } else {
  1225. $scheme = 'https';
  1226. }
  1227. } else {
  1228. $scheme = $digest['scheme'];
  1229. }
  1230. // Host
  1231. $host = (isset($digest['host'])?$digest['host']:'');
  1232. // Port
  1233. $port = (isset($digest['port'])?':'.$digest['port']:'');
  1234. // Path
  1235. $path = (isset($digest['path'])?$digest['path']:'');
  1236. // Output
  1237. return $scheme.'://'.$host.$port.$path;
  1238. }
  1239. // Function to be called at top of each to allow upgrading environment as the spec changes
  1240. function upgradeCheck() {
  1241. // Upgrade to 1.31
  1242. if (file_exists('homepageSettings.ini.php')) {
  1243. $databaseConfig = parse_ini_file('databaseLocation.ini.php', true);
  1244. $homepageConfig = parse_ini_file('homepageSettings.ini.php', true);
  1245. $databaseConfig = array_merge($databaseConfig, $homepageConfig);
  1246. $databaseData = '; <?php die("Access denied"); ?>' . "\r\n";
  1247. foreach($databaseConfig as $k => $v) {
  1248. if(substr($v, -1) == "/") : $v = rtrim($v, "/"); endif;
  1249. $databaseData .= $k . " = \"" . $v . "\"\r\n";
  1250. }
  1251. write_ini_file($databaseData, 'databaseLocation.ini.php');
  1252. unlink('homepageSettings.ini.php');
  1253. unset($databaseData);
  1254. unset($homepageConfig);
  1255. }
  1256. // Upgrade to 1.32
  1257. if (file_exists('databaseLocation.ini.php')) {
  1258. // Load Existing
  1259. $config = parse_ini_file('databaseLocation.ini.php', true);
  1260. // Refactor
  1261. $config['database_Location'] = preg_replace('/\/\/$/','/',$config['databaseLocation'].'/');
  1262. $config['user_home'] = $config['database_Location'].'users/';
  1263. unset($config['databaseLocation']);
  1264. // Turn Off Emby And Plex Recent
  1265. $config["embyURL"] = $config["embyURL"].(!empty($config["embyPort"])?':'.$config["embyPort"]:'');
  1266. unset($config["embyPort"]);
  1267. $config["plexURL"] = $config["plexURL"].(!empty($config["plexPort"])?':'.$config["plexPort"]:'');
  1268. unset($config["plexPort"]);
  1269. $config["nzbgetURL"] = $config["nzbgetURL"].(!empty($config["nzbgetPort"])?':'.$config["nzbgetPort"]:'');
  1270. unset($config["nzbgetPort"]);
  1271. $config["sabnzbdURL"] = $config["sabnzbdURL"].(!empty($config["sabnzbdPort"])?':'.$config["sabnzbdPort"]:'');
  1272. unset($config["sabnzbdPort"]);
  1273. $config["headphonesURL"] = $config["headphonesURL"].(!empty($config["headphonesPort"])?':'.$config["headphonesPort"]:'');
  1274. unset($config["headphonesPort"]);
  1275. // Write config file
  1276. $config['CONFIG_VERSION'] = '1.32';
  1277. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][pre1.32].bak.php');
  1278. $createConfigSuccess = createConfig($config);
  1279. // Create new config
  1280. if ($createConfigSuccess) {
  1281. if (file_exists('config/config.php')) {
  1282. // Remove Old ini file
  1283. unlink('databaseLocation.ini.php');
  1284. } else {
  1285. debug_out('Something is not right here!');
  1286. }
  1287. } else {
  1288. debug_out('Couldn\'t create updated configuration.' ,1);
  1289. }
  1290. }
  1291. // Upgrade to 1.33
  1292. $config = loadConfig();
  1293. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.33')) {
  1294. // Fix User Directory
  1295. $config['database_Location'] = preg_replace('/\/\/$/','/',$config['database_Location'].'/');
  1296. $config['user_home'] = $config['database_Location'].'users/';
  1297. unset($config['USER_HOME']);
  1298. // Backend auth merge
  1299. if (isset($config['authBackendPort']) && !isset(parse_url($config['authBackendHost'])['port'])) {
  1300. $config['authBackendHost'] .= ':'.$config['authBackendPort'];
  1301. }
  1302. unset($config['authBackendPort']);
  1303. // If auth is being used move it to embyURL as that is now used in auth functions
  1304. if ((isset($config['authType']) && $config['authType'] == 'true') && (isset($config['authBackendHost']) && $config['authBackendHost'] == 'true') && (isset($config['authBackend']) && in_array($config['authBackend'], array('emby_all','emby_local','emby_connect')))) {
  1305. $config['embyURL'] = $config['authBackendHost'];
  1306. }
  1307. // Upgrade database to latest version
  1308. updateSQLiteDB($config['database_Location'],'1.32');
  1309. // Update Version and Commit
  1310. $config['apply_CONFIG_VERSION'] = '1.33';
  1311. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.32].bak.php');
  1312. $createConfigSuccess = createConfig($config);
  1313. unset($config);
  1314. }
  1315. // Upgrade to 1.34
  1316. $config = loadConfig();
  1317. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.34')) {
  1318. // Upgrade database to latest version
  1319. updateSQLiteDB($config['database_Location'],'1.33');
  1320. // Update Version and Commit
  1321. $config['CONFIG_VERSION'] = '1.34';
  1322. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.33].bak.php');
  1323. $createConfigSuccess = createConfig($config);
  1324. unset($config);
  1325. }
  1326. // Upgrade to 1.40
  1327. $config = loadConfig();
  1328. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.40')) {
  1329. // Upgrade database to latest version
  1330. updateSQLiteDB($config['database_Location'],'1.38');
  1331. // Update Version and Commit
  1332. $config['CONFIG_VERSION'] = '1.40';
  1333. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.38].bak.php');
  1334. $createConfigSuccess = createConfig($config);
  1335. unset($config);
  1336. }
  1337. // Upgrade to 1.50
  1338. $config = loadConfig();
  1339. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.50')) {
  1340. // Upgrade database to latest version
  1341. updateSQLiteDB($config['database_Location'],'1.40');
  1342. // Update Version and Commit
  1343. $config['CONFIG_VERSION'] = '1.50';
  1344. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.40].bak.php');
  1345. $createConfigSuccess = createConfig($config);
  1346. unset($config);
  1347. }
  1348. return true;
  1349. }
  1350. // Get OS from server
  1351. function getOS(){
  1352. if(PHP_SHLIB_SUFFIX == "dll"){
  1353. return "win";
  1354. }else{
  1355. return "nix";
  1356. }
  1357. }
  1358. //Get Error by Server OS
  1359. function getError($os, $error){
  1360. $ini = (!empty(php_ini_loaded_file()) ? php_ini_loaded_file() : "php.ini");
  1361. $ext = (!empty(ini_get('extension_dir')) ? "uncomment ;extension_dir = and make sure it says -> extension_dir = '".ini_get('extension_dir')."'" : "uncomment ;extension_dir = and add path to 'ext' to make it like extension_dir = 'C:\nginx\php\ext'");
  1362. $errors = array(
  1363. 'pdo_sqlite' => array(
  1364. 'win' => '<b>PDO:SQLite</b> not enabled, uncomment ;extension=php_pdo_sqlite.dll in the file php.ini | '.$ext,
  1365. 'nix' => '<b>PDO:SQLite</b> not enabled, PHP7 -> run sudo apt-get install php7.0-sqlite | PHP5 -> run sudo apt-get install php5-sqlite',
  1366. ),
  1367. 'sqlite3' => array(
  1368. 'win' => '<b>SQLite3</b> not enabled, uncomment ;extension=php_sqlite3.dll in the file php.ini | uncomment ;sqlite3.extension_dir = and add "ext" to make it sqlite3.extension_dir = ext',
  1369. 'nix' => '<b>SQLite3</b> not enabled, run sudo apt-get install php-sqlite3',
  1370. ),
  1371. 'curl' => array(
  1372. 'win' => '<b>cURL</b> not enabled, uncomment ;extension=php_curl.dll in the file php.ini | '.$ext,
  1373. 'nix' => '<b>cURL</b> not enabled, PHP7 -> sudo apt-get install php-curl or sudo apt-get install php7.0-curl | PHP5 -> run sudo apt-get install php5.6-curl',
  1374. ),
  1375. 'zip' => array(
  1376. 'win' => '<b>PHP Zip</b> not enabled, uncomment ;extension=php_zip.dll in the file php.ini, if that doesn\'t work remove that line',
  1377. 'nix' => '<b>PHP Zip</b> not enabled, PHP7 -> run sudo apt-get install php7.0-zip | PHP5 -> run sudo apt-get install php5.6-zip',
  1378. ),
  1379. );
  1380. return (isset($errors[$error][$os]) ? $errors[$error][$os] : 'No Error Info Found');
  1381. }
  1382. // Check if all software dependancies are met
  1383. function dependCheck() {
  1384. $output = array();
  1385. $i = 1;
  1386. if (!extension_loaded('pdo_sqlite')) { $output["Step $i"] = getError(getOS(),'pdo_sqlite'); $i++; }
  1387. if (!extension_loaded('curl')) { $output["Step $i"] = getError(getOS(),'curl'); $i++; }
  1388. if (!extension_loaded('zip')) { $output["Step $i"] = getError(getOS(),'zip'); $i++; }
  1389. //if (!extension_loaded('sqlite3')) { $output[] = getError(getOS(),'sqlite3'); }
  1390. if ($output) {
  1391. $output["Step $i"] = "<b>Restart PHP and/or Webserver to apply changes</b>"; $i++;
  1392. $output["Step $i"] = "<b>Please visit here to also check status of necessary components after you fix them: <a href='check.php'>check.php<a/></b>"; $i++;
  1393. debug_out($output,1);
  1394. }
  1395. return true;
  1396. }
  1397. // Process file uploads
  1398. function uploadFiles($path, $ext_mask = null) {
  1399. if (isset($_FILES) && count($_FILES)) {
  1400. require_once('class.uploader.php');
  1401. $uploader = new Uploader();
  1402. $data = $uploader->upload($_FILES['files'], array(
  1403. 'limit' => 10,
  1404. 'maxSize' => 10,
  1405. 'extensions' => $ext_mask,
  1406. 'required' => false,
  1407. 'uploadDir' => str_replace('//','/',$path.'/'),
  1408. 'title' => array('name'),
  1409. 'removeFiles' => true,
  1410. 'replace' => true,
  1411. ));
  1412. if($data['isComplete']){
  1413. $files = $data['data'];
  1414. writeLog("success", $files['metas'][0]['name']." was uploaded");
  1415. echo json_encode($files['metas'][0]['name']);
  1416. }
  1417. if($data['hasErrors']){
  1418. $errors = $data['errors'];
  1419. writeLog("error", $files['metas'][0]['name']." was not able to upload");
  1420. echo json_encode($errors);
  1421. }
  1422. } else {
  1423. writeLog("error", "image was not uploaded");
  1424. echo json_encode('No files submitted!');
  1425. }
  1426. }
  1427. // Process file uploads
  1428. function uploadAvatar($path, $ext_mask = null) {
  1429. if (isset($_FILES) && count($_FILES)) {
  1430. require_once('class.uploader.php');
  1431. $uploader = new Uploader();
  1432. $data = $uploader->upload($_FILES['files'], array(
  1433. 'limit' => 10,
  1434. 'maxSize' => 10,
  1435. 'extensions' => $ext_mask,
  1436. 'required' => false,
  1437. 'uploadDir' => str_replace('//','/',$path.'/'),
  1438. 'title' => array('name'),
  1439. 'removeFiles' => true,
  1440. 'replace' => true,
  1441. ));
  1442. if($data['isComplete']){
  1443. $files = $data['data'];
  1444. writeLog("success", $files['metas'][0]['name']." was uploaded");
  1445. echo json_encode($files['metas'][0]['name']);
  1446. }
  1447. if($data['hasErrors']){
  1448. $errors = $data['errors'];
  1449. writeLog("error", $files['metas'][0]['name']." was not able to upload");
  1450. echo json_encode($errors);
  1451. }
  1452. } else {
  1453. writeLog("error", "image was not uploaded");
  1454. echo json_encode('No files submitted!');
  1455. }
  1456. }
  1457. // Remove file
  1458. function removeFiles($path) {
  1459. if(is_file($path)) {
  1460. writeLog("success", "image was removed");
  1461. unlink($path);
  1462. } else {
  1463. writeLog("error", "image was not removed");
  1464. echo json_encode('No file specified for removal!');
  1465. }
  1466. }
  1467. // Lazy select options
  1468. function resolveSelectOptions($array, $selected = '', $multi = false) {
  1469. $output = array();
  1470. $selectedArr = ($multi?explode('|', $selected):array());
  1471. foreach ($array as $key => $value) {
  1472. if (is_array($value)) {
  1473. if (isset($value['optgroup'])) {
  1474. $output[] = '<optgroup label="'.$key.'">';
  1475. foreach($value['optgroup'] as $k => $v) {
  1476. $output[] = '<option value="'.$v['value'].'"'.($selected===$v['value']||in_array($v['value'],$selectedArr)?' selected':'').(isset($v['disabled']) && $v['disabled']?' disabled':'').'>'.$k.'</option>';
  1477. }
  1478. } else {
  1479. $output[] = '<option value="'.$value['value'].'"'.($selected===$value['value']||in_array($value['value'],$selectedArr)?' selected':'').(isset($value['disabled']) && $value['disabled']?' disabled':'').'>'.$key.'</option>';
  1480. }
  1481. } else {
  1482. $output[] = '<option value="'.$value.'"'.($selected===$value||in_array($value,$selectedArr)?' selected':'').'>'.$key.'</option>';
  1483. }
  1484. }
  1485. return implode('',$output);
  1486. }
  1487. // Check if user is allowed to continue
  1488. function qualifyUser($type, $errOnFail = false) {
  1489. if (!isset($GLOBALS['USER'])) {
  1490. require_once("user.php");
  1491. $GLOBALS['USER'] = new User('registration_callback');
  1492. }
  1493. if (is_bool($type)) {
  1494. if ($type === true) {
  1495. $authorized = ($GLOBALS['USER']->authenticated == true);
  1496. } else {
  1497. $authorized = true;
  1498. }
  1499. } elseif (is_string($type) || is_array($type)) {
  1500. if ($type !== 'false') {
  1501. if (!is_array($type)) {
  1502. $type = explode('|',$type);
  1503. }
  1504. $authorized = ($GLOBALS['USER']->authenticated && in_array($GLOBALS['USER']->role,$type));
  1505. } else {
  1506. $authorized = true;
  1507. }
  1508. } else {
  1509. debug_out('Invalid Syntax!',1);
  1510. }
  1511. if (!$authorized && $errOnFail) {
  1512. if ($GLOBALS['USER']->authenticated) {
  1513. header('Location: '.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=401');
  1514. echo '<script>window.location.href = \''.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=401\'</script>';
  1515. } else {
  1516. header('Location: '.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=999');
  1517. echo '<script>window.location.href = \''.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=999\'</script>';
  1518. }
  1519. debug_out('Not Authorized' ,1);
  1520. } else {
  1521. return $authorized;
  1522. }
  1523. }
  1524. // Build an (optionally) tabbed settings page.
  1525. function buildSettings($array) {
  1526. /*
  1527. array(
  1528. 'title' => '',
  1529. 'id' => '',
  1530. 'fields' => array( See buildField() ),
  1531. 'tabs' => array(
  1532. array(
  1533. 'title' => '',
  1534. 'id' => '',
  1535. 'image' => '',
  1536. 'fields' => array( See buildField() ),
  1537. ),
  1538. ),
  1539. );
  1540. */
  1541. $notifyExplode = explode("-", NOTIFYEFFECT);
  1542. $fieldFunc = function($fieldArr) {
  1543. $fields = '<div class="row">';
  1544. foreach($fieldArr as $key => $value) {
  1545. $isSingle = isset($value['type']);
  1546. if ($isSingle) { $value = array($value); }
  1547. $tmpField = '';
  1548. $sizeLg = max(floor(12/count($value)),2);
  1549. $sizeMd = max(floor(($isSingle?12:6)/count($value)),3);
  1550. foreach($value as $k => $v) {
  1551. $tmpField .= buildField($v, 12, $sizeMd, $sizeLg);
  1552. }
  1553. $fields .= ($isSingle?$tmpField:'<div class="row col-sm-12 content-form">'.$tmpField.'</div>');
  1554. }
  1555. $fields .= '</div>';
  1556. return $fields;
  1557. };
  1558. $fields = (isset($array['fields'])?$fieldFunc($array['fields']):'');
  1559. $tabSelectors = array();
  1560. $tabContent = array();
  1561. if (isset($array['tabs'])) {
  1562. foreach($array['tabs'] as $key => $value) {
  1563. $id = (isset($value['id'])?$value['id']:randString(32));
  1564. $tabSelectors[$key] = '<li class="apps'.($tabSelectors?'':' active').'"><a href="#tab-'.$id.'" data-toggle="tab" aria-expanded="true"><img style="height:40px; width:40px;" src="'.(isset($value['image'])?$value['image']:'images/organizr.png').'"></a></li>';
  1565. $tabContent[$key] = '<div class="tab-pane big-box fade'.($tabContent?'':' active in').'" id="tab-'.$id.'">'.$fieldFunc($value['fields']).'</div>';
  1566. }
  1567. }
  1568. $pageID = (isset($array['id'])?$array['id']:str_replace(array(' ','"',"'"),array('_'),strtolower($array['id'])));
  1569. $extraClick = ($pageID == 'appearance_settings' ? "$('#advanced_settings_form_submit').click();console.log('add theme settings');" : "");
  1570. return '
  1571. <div class="email-body">
  1572. <div class="email-header gray-bg">
  1573. <button type="button" class="btn btn-danger btn-sm waves close-button"><i class="fa fa-close"></i></button>
  1574. <button id="'.$pageID.'_form_submit" class="btn waves btn-labeled btn-success btn btn-sm text-uppercase waves-effect waves-float save-btn-form">
  1575. <span class="btn-label"><i class="fa fa-floppy-o"></i></span>Save
  1576. </button>
  1577. <h1>'.$array['title'].'</h1>
  1578. </div>
  1579. <div class="email-inner small-box">
  1580. <div class="email-inner-section">
  1581. <div class="small-box fade in" id="'.$pageID.'_frame">
  1582. <div class="col-lg-12">
  1583. '.(isset($array['customBeforeForm'])?$array['customBeforeForm']:'').'
  1584. <form class="content-form" name="'.$pageID.'" id="'.$pageID.'_form" onsubmit="return false;">
  1585. '.$fields.($tabContent?'
  1586. <div class="tabbable tabs-with-bg" id="'.$pageID.'_tabs">
  1587. <ul class="nav nav-tabs apps">
  1588. '.implode('', $tabSelectors).'
  1589. </ul>
  1590. <div class="clearfix"></div>
  1591. <div class="tab-content">
  1592. '.implode('', $tabContent).'
  1593. </div>
  1594. </div>':'').'
  1595. </form>
  1596. '.(isset($array['customAfterForm'])?$array['customAfterForm']:'').'
  1597. </div>
  1598. </div>
  1599. </div>
  1600. </div>
  1601. </div>
  1602. <script>
  1603. $(document).ready(function() {
  1604. $(\'#'.$pageID.'_form\').find(\'input, select, textarea\').on(\'change\', function() { $(this).attr(\'data-changed\', \'true\'); });
  1605. var '.$pageID.'Validate = function() { if (this.value && !RegExp(\'^\'+this.pattern+\'$\').test(this.value)) { $(this).addClass(\'invalid\'); } else { $(this).removeClass(\'invalid\'); } };
  1606. $(\'#'.$pageID.'_form\').find(\'input[pattern]\').each('.$pageID.'Validate).on(\'keyup\', '.$pageID.'Validate);
  1607. $(\'#'.$pageID.'_form\').find(\'select[multiple]\').on(\'change click\', function() { $(this).attr(\'data-changed\', \'true\'); });
  1608. $(\'#'.$pageID.'_form_submit\').on(\'click\', function () {
  1609. var newVals = {};
  1610. var hasVals = false;
  1611. var errorFields = [];
  1612. $(\'#'.$pageID.'_form\').find(\'[data-changed=true][name]\').each(function() {
  1613. hasVals = true;
  1614. if (this.type == \'checkbox\') {
  1615. newVals[this.name] = this.checked;
  1616. } else if ($(this).hasClass(\'summernote\')) {
  1617. newVals[$(this).attr(\'name\')] = $(this).siblings(\'.note-editor\').find(\'.panel-body\').html();
  1618. } else {
  1619. if (this.value && this.pattern && !RegExp(\'^\'+this.pattern+\'$\').test(this.value)) { errorFields.push(this.name); }
  1620. var fieldVal = $(this).val();
  1621. if (typeof fieldVal == \'object\') {
  1622. if (typeof fieldVal.join == \'function\') {
  1623. fieldVal = fieldVal.join(\'|\');
  1624. } else {
  1625. fieldVal = JSON.stringify(fieldVal);
  1626. }
  1627. }
  1628. newVals[this.name] = fieldVal;
  1629. }
  1630. });
  1631. if (errorFields.length) {
  1632. parent.notify(\'Fields have errors: \'+errorFields.join(\', \')+\'!\', \'bullhorn\', \'error\', 5000, \''.$notifyExplode[0].'\', \''.$notifyExplode[1].'\');
  1633. } else if (hasVals) {
  1634. console.log(newVals);
  1635. ajax_request(\'POST\', \''.(isset($array['submitAction'])?$array['submitAction']:'update-config').'\', newVals, function(data, code) {
  1636. $(\'#'.$pageID.'_form\').find(\'[data-changed=true][name]\').removeAttr(\'data-changed\');
  1637. });
  1638. '.$extraClick.'
  1639. } else {
  1640. parent.notify(\'Nothing to update!\', \'bullhorn\', \'error\', 5000, \''.$notifyExplode[0].'\', \''.$notifyExplode[1].'\');
  1641. }
  1642. return false;
  1643. });
  1644. '.(isset($array['onready'])?$array['onready']:'').'
  1645. });
  1646. </script>
  1647. ';
  1648. }
  1649. // Build Settings Fields
  1650. function buildField($params, $sizeSm = 12, $sizeMd = 12, $sizeLg = 12) {
  1651. /*
  1652. array(
  1653. 'type' => '',
  1654. 'placeholder' => '',
  1655. 'label' => '',
  1656. 'labelTranslate' => '',
  1657. 'assist' => '',
  1658. 'name' => '',
  1659. 'pattern' => '',
  1660. 'options' => array( // For SELECT only
  1661. 'Display' => 'value',
  1662. ),
  1663. )
  1664. */
  1665. // Tags
  1666. $tags = array();
  1667. foreach(array('placeholder','style','disabled','readonly','pattern','min','max','required','onkeypress','onchange','onfocus','onleave','href','onclick') as $value) {
  1668. if (isset($params[$value])) {
  1669. if (is_string($params[$value])) { $tags[] = $value.'="'.$params[$value].'"';
  1670. } else if ($params[$value] === true) { $tags[] = $value; }
  1671. }
  1672. }
  1673. $format = (isset($params['format']) && in_array($params['format'],array(false,'colour','color'))?$params['format']:false);
  1674. $name = (isset($params['name'])?$params['name']:(isset($params['id'])?$params['id']:''));
  1675. $id = (isset($params['id'])?$params['id']:(isset($params['name'])?$params['name'].'_id':randString(32)));
  1676. $val = (isset($params['value'])?$params['value']:'');
  1677. $class = (isset($params['class'])?' '.$params['class']:'');
  1678. $wrapClass = (isset($params['wrapClass'])?$params['wrapClass']:'form-content');
  1679. $assist = (isset($params['assist'])?' - i.e. '.$params['assist']:'');
  1680. $label = (isset($params['labelTranslate'])?translate($params['labelTranslate']):(isset($params['label'])?$params['label']:''));
  1681. $labelOut = '<p class="help-text">'.$label.$assist.'</p>';
  1682. // Field Design
  1683. switch ($params['type']) {
  1684. case 'text':
  1685. case 'number':
  1686. case 'password':
  1687. $field = '<input id="'.$id.'" name="'.$name.'" type="'.$params['type'].'" class="form-control material input-sm'.$class.'" '.implode(' ',$tags).' autocorrect="off" autocapitalize="off" value="'.$val.'">';
  1688. break;
  1689. case 'select':
  1690. case 'dropdown':
  1691. $field = '<select id="'.$id.'" name="'.$name.'" class="form-control material input-sm" '.implode(' ',$tags).'>'.resolveSelectOptions($params['options'], $val).'</select>';
  1692. break;
  1693. case 'select-multi':
  1694. case 'dropdown-multi':
  1695. $field = '<select id="'.$id.'" name="'.$name.'" class="form-control input-sm" '.implode(' ',$tags).' multiple="multiple">'.resolveSelectOptions($params['options'], $val, true).'</select>';
  1696. break;
  1697. case 'check':
  1698. case 'checkbox':
  1699. case 'toggle':
  1700. $checked = ((is_bool($val) && $val) || trim($val) === 'true'?' checked':'');
  1701. $colour = (isset($params['colour'])?$params['colour']:'success');
  1702. $labelOut = '<label for="'.$id.'"></label>'.$label;
  1703. $field = '<input id="'.$id.'" name="'.$name.'" type="checkbox" class="switcher switcher-'.$colour.' '.$class.'" '.implode(' ',$tags).' data-value="'.$val.'"'.$checked.'>';
  1704. break;
  1705. case 'radio':
  1706. $labelOut = '';
  1707. $checked = ((is_bool($val) && $val) || ($val && trim($val) !== 'false')?' checked':'');
  1708. $bType = (isset($params['buttonType'])?$params['buttonType']:'success');
  1709. $field = '<div class="radio radio-'.$bType.'"><input id="'.$id.'" name="'.$name.'" type="radio" class="'.$class.'" '.implode(' ',$tags).' value="'.$val.'"'.$checked.'><label for="'.$id.'">'.$label.'</label></div>';
  1710. break;
  1711. case 'date':
  1712. $field = 'Unsupported, planned.';
  1713. break;
  1714. case 'hidden':
  1715. return '<input id="'.$id.'" name="'.$name.'" type="hidden" class="'.$class.'" '.implode(' ',$tags).' value="'.$val.'">';
  1716. break;
  1717. case 'header':
  1718. $labelOut = '';
  1719. $headType = (isset($params['value'])?$params['value']:3);
  1720. $field = '<h'.$headType.' class="'.$class.'" '.implode(' ',$tags).'>'.$label.'</h'.$headType.'>';
  1721. break;
  1722. case 'button':
  1723. $labelOut = '';
  1724. $icon = (isset($params['icon'])?$params['icon']:'flask');
  1725. $bType = (isset($params['buttonType'])?$params['buttonType']:'success');
  1726. $bDropdown = (isset($params['buttonDrop'])?$params['buttonDrop']:'');
  1727. $field = ($bDropdown?'<div class="btn-group">':'').'<button id="'.$id.'" type="button" class="btn waves btn-labeled btn-'.$bType.' btn-sm text-uppercase waves-effect waves-float'.$class.''.($bDropdown?' dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"':'"').' '.implode(' ',$tags).'><span class="btn-label"><i class="fa fa-'.$icon.'"></i></span><span class="btn-text">'.$label.'</span></button>'.($bDropdown?$bDropdown.'</div>':'');
  1728. break;
  1729. case 'textarea':
  1730. $rows = (isset($params['rows'])?$params['rows']:5);
  1731. $field = '<textarea id="'.$id.'" name="'.$name.'" class="form-control'.$class.'" rows="'.$rows.'" '.implode(' ',$tags).'>'.$val.'</textarea>';
  1732. break;
  1733. case 'custom':
  1734. // Settings
  1735. $settings = array(
  1736. '$id' => $id,
  1737. '$name' => $name,
  1738. '$val' => $val,
  1739. '$label' => $label,
  1740. '$labelOut' => $labelOut,
  1741. );
  1742. // Get HTML
  1743. $html = (isset($params['html'])?$params['html']:'Nothing Specified!');
  1744. // If LabelOut is in html dont print it twice
  1745. $labelOut = (strpos($html,'$label')!==false?'':$labelOut);
  1746. // Replace variables in settings
  1747. $html = preg_replace_callback('/\$\w+\b/', function ($match) use ($settings) { return (isset($settings[$match[0]])?$settings[$match[0]]:'{'.$match[0].' is undefined}'); }, $html);
  1748. // Build Field
  1749. $field = '<div id="'.$id.'_html" class="custom-field">'.$html.'</div>';
  1750. break;
  1751. case 'space':
  1752. $labelOut = '';
  1753. $field = str_repeat('<br>', (isset($params['value'])?$params['value']:1));
  1754. break;
  1755. default:
  1756. $field = 'Unsupported field type';
  1757. break;
  1758. }
  1759. // Field Formats
  1760. switch ($format) {
  1761. case 'colour': // Fuckin Eh, Canada!
  1762. case 'color':
  1763. $labelBef = '<center>'.$label.'</center>';
  1764. $wrapClass = 'gray-bg colour-field';
  1765. $labelAft = '';
  1766. $field = str_replace(' material input-sm','',$field);
  1767. break;
  1768. default:
  1769. $labelBef = '';
  1770. $labelAft = $labelOut;
  1771. }
  1772. return '<div class="'.$wrapClass.' col-sm-'.$sizeSm.' col-md-'.$sizeMd.' col-lg-'.$sizeLg.'">'.$labelBef.$field.$labelAft.'</div>';
  1773. }
  1774. // Tab Settings Generation
  1775. function printTabRow($data) {
  1776. $hidden = false;
  1777. if ($data===false) {
  1778. $hidden = true;
  1779. $data = array( // New Tab Defaults
  1780. 'id' => 'new',
  1781. 'name' => '',
  1782. 'url' => '',
  1783. 'icon' => 'fa-diamond',
  1784. 'iconurl' => '',
  1785. 'active' => 'true',
  1786. 'user' => 'true',
  1787. 'guest' => 'true',
  1788. 'window' => 'false',
  1789. 'splash' => 'true',
  1790. 'ping' => 'false',
  1791. 'ping_url' => '',
  1792. 'defaultz' => '',
  1793. );
  1794. }
  1795. $image = '<span style="font: normal normal normal 30px/1 FontAwesome;" class="fa fa-hand-paper-o"></span>';
  1796. $output = '
  1797. <li id="tab-'.$data['id'].'" class="list-group-item" style="position: relative; left: 0px; top: 0px; '.($hidden?' display: none;':'').'">
  1798. <tab class="content-form form-inline">
  1799. <div class="row">
  1800. '.buildField(array(
  1801. 'type' => 'custom',
  1802. 'html' => '<div class="action-btns tabIconView"><a style="margin-left: 0px">'.($data['iconurl']?'<img src="'.$data['iconurl'].'" height="30" width="30">':'<span style="font: normal normal normal 30px/1 FontAwesome;" class="fa '.($data['icon']?$data['icon']:'hand-paper-o').'"></span>').'</a></div>',
  1803. ),12,1,1).'
  1804. '.buildField(array(
  1805. 'type' => 'hidden',
  1806. 'id' => 'tab-'.$data['id'].'-id',
  1807. 'name' => 'id['.$data['id'].']',
  1808. 'value' => $data['id'],
  1809. ),12,2,1).'
  1810. '.buildField(array(
  1811. 'type' => 'text',
  1812. 'id' => 'tab-'.$data['id'].'-name',
  1813. 'name' => 'name['.$data['id'].']',
  1814. 'required' => true,
  1815. 'placeholder' => 'Organizr Homepage',
  1816. 'labelTranslate' => 'TAB_NAME',
  1817. 'value' => $data['name'],
  1818. 'class' => 'darkBold',
  1819. ),12,2,1).'
  1820. '.buildField(array(
  1821. 'type' => 'text',
  1822. 'id' => 'tab-'.$data['id'].'-url',
  1823. 'name' => 'url['.$data['id'].']',
  1824. 'required' => true,
  1825. 'placeholder' => 'homepage.php',
  1826. 'labelTranslate' => 'TAB_URL',
  1827. 'value' => $data['url'],
  1828. 'class' => 'darkBold',
  1829. ),12,2,2).'
  1830. '.buildField(array(
  1831. 'type' => 'text',
  1832. 'id' => 'tab-'.$data['id'].'-iconurl',
  1833. 'name' => 'iconurl['.$data['id'].']',
  1834. 'placeholder' => 'images/organizr.png',
  1835. 'labelTranslate' => 'ICON_URL',
  1836. 'value' => $data['iconurl'],
  1837. 'class' => 'darkBold',
  1838. ),12,2,1).'
  1839. '.buildField(array(
  1840. 'type' => 'custom',
  1841. 'id' => 'tab-'.$data['id'].'-icon',
  1842. 'name' => 'icon['.$data['id'].']',
  1843. 'html' => '- '.translate('OR').' - <div class="input-group"><input data-placement="bottomRight" class="form-control material icp-auto'.($hidden?'-pend':'').'" id="$id" name="$name" value="$val" type="text" /><span class="input-group-addon"></span></div>',
  1844. 'value' => $data['icon'],
  1845. ),12,1,1).'
  1846. '.buildField(array(
  1847. 'type' => 'text',
  1848. 'id' => 'tab-'.$data['id'].'-ping_url',
  1849. 'name' => 'ping_url['.$data['id'].']',
  1850. 'placeholder' => 'host:port',
  1851. 'labelTranslate' => 'PING_URL',
  1852. 'value' => $data['ping_url'],
  1853. 'class' => 'darkBold',
  1854. ),12,2,1).'
  1855. '.buildField(array(
  1856. 'type' => 'radio',
  1857. 'labelTranslate' => 'DEFAULT',
  1858. 'name' => 'defaultz['.$data['id'].']',
  1859. 'value' => $data['defaultz'],
  1860. 'onclick' => "$('[type=radio][id!=\''+this.id+'\']').each(function() { this.checked=false; });",
  1861. ),12,1,1).'
  1862. '.buildField(array(
  1863. 'type' => 'button',
  1864. 'icon' => 'chevron-down',
  1865. 'buttonType' => 'success',
  1866. 'labelTranslate' => 'MORE',
  1867. 'onclick' => "$('#tab-".$data['id']."-row').toggle();",
  1868. 'class' => 'toggleTabExtra',
  1869. ),12,1,1).'
  1870. '.buildField(array(
  1871. 'type' => 'button',
  1872. 'icon' => 'trash',
  1873. 'buttonType' => 'danger',
  1874. 'labelTranslate' => 'REMOVE',
  1875. 'onclick' => "$(this).parents('li').remove();",
  1876. ),12,1,1).'</div><div id = "tab-'.$data['id'].'-row" class = "row animated slideInUp" style = "display:none;" ><div></div>
  1877. '.buildField(array(
  1878. 'type' => 'checkbox',
  1879. 'labelTranslate' => 'ACTIVE',
  1880. 'name' => 'active['.$data['id'].']',
  1881. 'value' => $data['active'],
  1882. ),12,1,1).'
  1883. '.buildField(array(
  1884. 'type' => 'checkbox',
  1885. 'labelTranslate' => 'USER',
  1886. 'colour' => 'primary',
  1887. 'name' => 'user['.$data['id'].']',
  1888. 'value' => $data['user'],
  1889. ),12,1,1).'
  1890. '.buildField(array(
  1891. 'type' => 'checkbox',
  1892. 'labelTranslate' => 'GUEST',
  1893. 'colour' => 'warning',
  1894. 'name' => 'guest['.$data['id'].']',
  1895. 'value' => $data['guest'],
  1896. ),12,1,1).'
  1897. '.buildField(array(
  1898. 'type' => 'checkbox',
  1899. 'labelTranslate' => 'NO_IFRAME',
  1900. 'colour' => 'danger',
  1901. 'name' => 'window['.$data['id'].']',
  1902. 'value' => $data['window'],
  1903. ),12,1,1).'
  1904. '.buildField(array(
  1905. 'type' => 'checkbox',
  1906. 'labelTranslate' => 'SPLASH',
  1907. 'colour' => 'success',
  1908. 'name' => 'splash['.$data['id'].']',
  1909. 'value' => $data['splash'],
  1910. ),12,1,1).'
  1911. '.buildField(array(
  1912. 'type' => 'checkbox',
  1913. 'labelTranslate' => 'PING',
  1914. 'colour' => 'success',
  1915. 'name' => 'ping['.$data['id'].']',
  1916. 'value' => $data['ping'],
  1917. ),12,1,1).'
  1918. </div>
  1919. </tab>
  1920. </li>
  1921. ';
  1922. return $output;
  1923. }
  1924. // Timezone array
  1925. function timezoneOptions() {
  1926. $output = array();
  1927. $timezones = array();
  1928. $regions = array(
  1929. 'Africa' => DateTimeZone::AFRICA,
  1930. 'America' => DateTimeZone::AMERICA,
  1931. 'Antarctica' => DateTimeZone::ANTARCTICA,
  1932. 'Arctic' => DateTimeZone::ARCTIC,
  1933. 'Asia' => DateTimeZone::ASIA,
  1934. 'Atlantic' => DateTimeZone::ATLANTIC,
  1935. 'Australia' => DateTimeZone::AUSTRALIA,
  1936. 'Europe' => DateTimeZone::EUROPE,
  1937. 'Indian' => DateTimeZone::INDIAN,
  1938. 'Pacific' => DateTimeZone::PACIFIC
  1939. );
  1940. foreach ($regions as $name => $mask) {
  1941. $zones = DateTimeZone::listIdentifiers($mask);
  1942. foreach($zones as $timezone) {
  1943. $time = new DateTime(NULL, new DateTimeZone($timezone));
  1944. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  1945. $output[$name]['optgroup'][substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm]['value'] = $timezone;
  1946. }
  1947. }
  1948. return $output;
  1949. }
  1950. // Build Database
  1951. function createSQLiteDB($path = false) {
  1952. if ($path === false) {
  1953. if (DATABASE_LOCATION){
  1954. $path = DATABASE_LOCATION;
  1955. } else {
  1956. debug_out('No Path Specified!');
  1957. }
  1958. }
  1959. if (!is_file($path.'users.db') || filesize($path.'users.db') <= 0) {
  1960. if (!isset($GLOBALS['file_db'])) {
  1961. $GLOBALS['file_db'] = new PDO('sqlite:'.$path.'users.db');
  1962. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  1963. }
  1964. // Create Users
  1965. $users = $GLOBALS['file_db']->query('CREATE TABLE `users` (
  1966. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1967. `username` TEXT UNIQUE,
  1968. `password` TEXT,
  1969. `email` TEXT,
  1970. `token` TEXT,
  1971. `role` TEXT,
  1972. `active` TEXT,
  1973. `last` TEXT,
  1974. `auth_service` TEXT DEFAULT \'internal\'
  1975. );');
  1976. // Create Tabs
  1977. $tabs = $GLOBALS['file_db']->query('CREATE TABLE `tabs` (
  1978. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1979. `order` INTEGER,
  1980. `users_id` INTEGER,
  1981. `name` TEXT,
  1982. `url` TEXT,
  1983. `defaultz` TEXT,
  1984. `active` TEXT,
  1985. `user` TEXT,
  1986. `guest` TEXT,
  1987. `icon` TEXT,
  1988. `iconurl` TEXT,
  1989. `window` TEXT,
  1990. `splash` TEXT,
  1991. `ping` TEXT,
  1992. `ping_url` TEXT
  1993. );');
  1994. // Create Options
  1995. $options = $GLOBALS['file_db']->query('CREATE TABLE `options` (
  1996. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1997. `users_id` INTEGER UNIQUE,
  1998. `title` TEXT UNIQUE,
  1999. `topbar` TEXT,
  2000. `bottombar` TEXT,
  2001. `sidebar` TEXT,
  2002. `hoverbg` TEXT,
  2003. `topbartext` TEXT,
  2004. `activetabBG` TEXT,
  2005. `activetabicon` TEXT,
  2006. `activetabtext` TEXT,
  2007. `inactiveicon` TEXT,
  2008. `inactivetext` TEXT,
  2009. `loading` TEXT,
  2010. `hovertext` TEXT
  2011. );');
  2012. // Create Invites
  2013. $invites = $GLOBALS['file_db']->query('CREATE TABLE `invites` (
  2014. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  2015. `code` TEXT UNIQUE,
  2016. `date` TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
  2017. `email` TEXT,
  2018. `username` TEXT,
  2019. `dateused` TIMESTAMP,
  2020. `usedby` TEXT,
  2021. `ip` TEXT,
  2022. `valid` TEXT
  2023. );');
  2024. writeLog("success", "database created/saved");
  2025. return $users && $tabs && $options && $invites;
  2026. } else {
  2027. writeLog("error", "database was unable to be created/saved");
  2028. return false;
  2029. }
  2030. }
  2031. // Upgrade Database
  2032. function updateSQLiteDB($db_path = false, $oldVerNum = false) {
  2033. if (!$db_path) {
  2034. if (defined('DATABASE_LOCATION')) {
  2035. $db_path = DATABASE_LOCATION;
  2036. } else {
  2037. debug_out('No Path Specified',1);
  2038. }
  2039. }
  2040. if (!isset($GLOBALS['file_db'])) {
  2041. $GLOBALS['file_db'] = new PDO('sqlite:'.$db_path.'users.db');
  2042. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2043. }
  2044. // Cache current DB
  2045. $cache = array();
  2046. foreach($GLOBALS['file_db']->query('SELECT name FROM sqlite_master WHERE type="table";') as $table) {
  2047. foreach($GLOBALS['file_db']->query('SELECT * FROM '.$table['name'].';') as $key => $row) {
  2048. foreach($row as $k => $v) {
  2049. if (is_string($k)) {
  2050. $cache[$table['name']][$key][$k] = $v;
  2051. }
  2052. }
  2053. }
  2054. }
  2055. // Remove Current Database
  2056. $GLOBALS['file_db'] = null;
  2057. $pathDigest = pathinfo($db_path.'users.db');
  2058. if (file_exists($db_path.'users.db')) {
  2059. rename($db_path.'users.db', $pathDigest['dirname'].'/'.$pathDigest['filename'].'['.date('Y-m-d_H-i-s').']'.($oldVerNum?'['.$oldVerNum.']':'').'.bak.db');
  2060. }
  2061. // Create New Database
  2062. $success = createSQLiteDB($db_path);
  2063. // Restore Items
  2064. if ($success) {
  2065. foreach($cache as $table => $tableData) {
  2066. if ($tableData) {
  2067. $queryBase = 'INSERT INTO '.$table.' (`'.implode('`,`',array_keys(current($tableData))).'`) values ';
  2068. $insertValues = array();
  2069. reset($tableData);
  2070. foreach($tableData as $key => $value) {
  2071. $insertValues[] = '('.implode(',',array_map(function($d) {
  2072. return (isset($d)?$GLOBALS['file_db']->quote($d):'null');
  2073. }, $value)).')';
  2074. }
  2075. $GLOBALS['file_db']->query($queryBase.implode(',',$insertValues).';');
  2076. }
  2077. }
  2078. writeLog("success", "database values have been updated");
  2079. return true;
  2080. } else {
  2081. writeLog("error", "database values unable to be updated");
  2082. return false;
  2083. }
  2084. }
  2085. // Commit colours to database
  2086. function updateDBOptions($values) {
  2087. if (!isset($GLOBALS['file_db'])) {
  2088. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2089. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2090. }
  2091. // Commit new values to database
  2092. if ($GLOBALS['file_db']->query('UPDATE options SET '.implode(',',array_map(function($d, $k) {
  2093. return '`'.$k.'` = '.(isset($d)?"'".addslashes($d)."'":'null');
  2094. }, $values, array_keys($values))).';')->rowCount()) {
  2095. return true;
  2096. } else if ($GLOBALS['file_db']->query('INSERT OR IGNORE INTO options (`'.implode('`,`',array_keys($values)).'`) VALUES (\''.implode("','",$values).'\');')->rowCount()) {
  2097. writeLog("success", "database values for options table have been updated");
  2098. return true;
  2099. } else {
  2100. writeLog("error", "database values for options table unable to be updated");
  2101. return false;
  2102. }
  2103. }
  2104. // Send AJAX notification
  2105. function sendNotification($success, $message = false, $send = true) {
  2106. $notifyExplode = explode("-", NOTIFYEFFECT);
  2107. if ($success) {
  2108. $msg = array(
  2109. 'html' => ($message?''.$message:'<strong>'.translate("SETTINGS_SAVED").'</strong>'),
  2110. 'icon' => 'floppy-o',
  2111. 'type' => 'success',
  2112. 'length' => '5000',
  2113. 'layout' => $notifyExplode[0],
  2114. 'effect' => $notifyExplode[1],
  2115. );
  2116. } else {
  2117. $msg = array(
  2118. 'html' => ($message?''.$message:'<strong>'.translate("SETTINGS_NOT_SAVED").'</strong>'),
  2119. 'icon' => 'floppy-o',
  2120. 'type' => 'failed',
  2121. 'length' => '5000',
  2122. 'layout' => $notifyExplode[0],
  2123. 'effect' => $notifyExplode[1],
  2124. );
  2125. }
  2126. // Send and kill script?
  2127. if ($send) {
  2128. header('Content-Type: application/json');
  2129. echo json_encode(array('notify'=>$msg));
  2130. die();
  2131. }
  2132. return $msg;
  2133. }
  2134. // Load colours from the database
  2135. function loadAppearance() {
  2136. // Defaults
  2137. $defaults = array(
  2138. 'title' => 'Organizr',
  2139. 'topbartext' => '#66D9EF',
  2140. 'topbar' => '#333333',
  2141. 'bottombar' => '#333333',
  2142. 'sidebar' => '#393939',
  2143. 'hoverbg' => '#AD80FD',
  2144. 'activetabBG' => '#F92671',
  2145. 'activetabicon' => '#FFFFFF',
  2146. 'activetabtext' => '#FFFFFF',
  2147. 'inactiveicon' => '#66D9EF',
  2148. 'inactivetext' => '#66D9EF',
  2149. 'loading' => '#66D9EF',
  2150. 'hovertext' => '#000000',
  2151. );
  2152. if (DATABASE_LOCATION) {
  2153. if(is_file(DATABASE_LOCATION.'users.db') && filesize(DATABASE_LOCATION.'users.db') > 0){
  2154. if (!isset($GLOBALS['file_db'])) {
  2155. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2156. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2157. }
  2158. // Database Lookup
  2159. $options = $GLOBALS['file_db']->query('SELECT * FROM options');
  2160. // Replace defaults with filled options
  2161. foreach($options as $row) {
  2162. foreach($defaults as $key => $value) {
  2163. if (isset($row[$key]) && $row[$key]) {
  2164. $defaults[$key] = $row[$key];
  2165. }
  2166. }
  2167. }
  2168. }
  2169. }
  2170. // Return the Results
  2171. return $defaults;
  2172. }
  2173. // Delete Database
  2174. function deleteDatabase() {
  2175. unset($_COOKIE['Organizr']);
  2176. setcookie('Organizr', '', time() - 3600, '/');
  2177. unset($_COOKIE['OrganizrU']);
  2178. setcookie('OrganizrU', '', time() - 3600, '/');
  2179. $GLOBALS['file_db'] = null;
  2180. unlink(DATABASE_LOCATION.'users.db');
  2181. foreach(glob(substr_replace($userdirpath, "", -1).'/*') as $file) {
  2182. if(is_dir($file)) {
  2183. rmdir($file);
  2184. } elseif (!is_dir($file)) {
  2185. unlink($file);
  2186. }
  2187. }
  2188. rmdir($userdirpath);
  2189. writeLog("success", "database has been deleted");
  2190. return true;
  2191. }
  2192. // Upgrade the installation
  2193. function upgradeInstall($branch = 'master') {
  2194. function downloadFile($url, $path){
  2195. ini_set('max_execution_time',0);
  2196. $folderPath = "upgrade/";
  2197. if(!mkdir($folderPath)){
  2198. writeLog("error", "organizr could not create upgrade folder");
  2199. }
  2200. $newfname = $folderPath . $path;
  2201. $file = fopen ($url, 'rb');
  2202. if ($file) {
  2203. $newf = fopen ($newfname, 'wb');
  2204. if ($newf) {
  2205. while(!feof($file)) {
  2206. fwrite($newf, fread($file, 1024 * 8), 1024 * 8);
  2207. }
  2208. }
  2209. }else{
  2210. writeLog("error", "organizr could not download $url");
  2211. }
  2212. if ($file) {
  2213. fclose($file);
  2214. writeLog("success", "organizr finished downloading the github zip file");
  2215. }else{
  2216. writeLog("error", "organizr could not download the github zip file");
  2217. }
  2218. if ($newf) {
  2219. fclose($newf);
  2220. writeLog("success", "organizr created upgrade zip file from github zip file");
  2221. }else{
  2222. writeLog("error", "organizr could not create upgrade zip file from github zip file");
  2223. }
  2224. }
  2225. function unzipFile($zipFile){
  2226. $zip = new ZipArchive;
  2227. $extractPath = "upgrade/";
  2228. if($zip->open($extractPath . $zipFile) != "true"){
  2229. writeLog("error", "organizr could not unzip upgrade.zip");
  2230. }else{
  2231. writeLog("success", "organizr unzipped upgrade.zip");
  2232. }
  2233. /* Extract Zip File */
  2234. $zip->extractTo($extractPath);
  2235. $zip->close();
  2236. }
  2237. // Function to remove folders and files
  2238. function rrmdir($dir) {
  2239. if (is_dir($dir)) {
  2240. $files = scandir($dir);
  2241. foreach ($files as $file)
  2242. if ($file != "." && $file != "..") rrmdir("$dir/$file");
  2243. rmdir($dir);
  2244. }
  2245. else if (file_exists($dir)) unlink($dir);
  2246. }
  2247. // Function to Copy folders and files
  2248. function rcopy($src, $dst) {
  2249. if (is_dir ( $src )) {
  2250. if (!file_exists($dst)) : mkdir ( $dst ); endif;
  2251. $files = scandir ( $src );
  2252. foreach ( $files as $file )
  2253. if ($file != "." && $file != "..")
  2254. rcopy ( "$src/$file", "$dst/$file" );
  2255. } else if (file_exists ( $src ))
  2256. copy ( $src, $dst );
  2257. }
  2258. $url = 'https://github.com/causefx/Organizr/archive/'.$branch.'.zip';
  2259. $file = "upgrade.zip";
  2260. $source = __DIR__ . '/upgrade/Organizr-'.$branch.'/';
  2261. $cleanup = __DIR__ . "/upgrade/";
  2262. $destination = __DIR__ . "/";
  2263. writeLog("success", "starting organizr upgrade process");
  2264. downloadFile($url, $file);
  2265. unzipFile($file);
  2266. rcopy($source, $destination);
  2267. writeLog("success", "new organizr files copied");
  2268. rrmdir($cleanup);
  2269. writeLog("success", "organizr upgrade folder removed");
  2270. writeLog("success", "organizr has been updated");
  2271. return true;
  2272. }
  2273. // NzbGET Items
  2274. function nzbgetConnect($list = 'listgroups') {
  2275. $url = qualifyURL(NZBGETURL);
  2276. $api = curl_get($url.'/'.NZBGETUSERNAME.':'.NZBGETPASSWORD.'/jsonrpc/'.$list);
  2277. $api = json_decode($api, true);
  2278. $gotNZB = array();
  2279. if (is_array($api) || is_object($api)){
  2280. foreach ($api['result'] AS $child) {
  2281. $downloadName = htmlentities($child['NZBName'], ENT_QUOTES);
  2282. $downloadStatus = $child['Status'];
  2283. $downloadCategory = $child['Category'];
  2284. if($list == "history"){ $downloadPercent = "100"; $progressBar = ""; }
  2285. if($list == "listgroups"){ $downloadPercent = (($child['FileSizeMB'] - $child['RemainingSizeMB']) / $child['FileSizeMB']) * 100; $progressBar = "progress-bar-striped active"; }
  2286. if($child['Health'] <= "750"){
  2287. $downloadHealth = "danger";
  2288. }elseif($child['Health'] <= "900"){
  2289. $downloadHealth = "warning";
  2290. }elseif($child['Health'] <= "1000"){
  2291. $downloadHealth = "success";
  2292. }
  2293. $gotNZB[] = '<tr>
  2294. <td class="col-xs-7 nzbtable-file-row">'.$downloadName.'</td>
  2295. <td class="col-xs-2 nzbtable nzbtable-row">'.$downloadStatus.'</td>
  2296. <td class="col-xs-1 nzbtable nzbtable-row">'.$downloadCategory.'</td>
  2297. <td class="col-xs-2 nzbtable nzbtable-row">
  2298. <div class="progress">
  2299. <div class="progress-bar progress-bar-'.$downloadHealth.' '.$progressBar.'" role="progressbar" aria-valuenow="'.$downloadPercent.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$downloadPercent.'%">
  2300. <p class="text-center">'.round($downloadPercent).'%</p>
  2301. <span class="sr-only">'.$downloadPercent.'% Complete</span>
  2302. </div>
  2303. </div>
  2304. </td>
  2305. </tr>';
  2306. }
  2307. if ($gotNZB) {
  2308. return implode('',$gotNZB);
  2309. } else {
  2310. return '<tr><td colspan="4"><p class="text-center">No Results</p></td></tr>';
  2311. }
  2312. }else{
  2313. writeLog("error", "NZBGET ERROR: could not connect - check URL and/or check token and/or Username and Password - if HTTPS, is cert valid");
  2314. }
  2315. }
  2316. // Sabnzbd Items
  2317. function sabnzbdConnect($list = 'queue') {
  2318. $url = qualifyURL(SABNZBDURL);
  2319. $api = file_get_contents($url.'/api?mode='.$list.'&output=json&apikey='.SABNZBDKEY);
  2320. $api = json_decode($api, true);
  2321. $gotNZB = array();
  2322. foreach ($api[$list]['slots'] AS $child) {
  2323. if($list == "queue"){ $downloadName = $child['filename']; $downloadCategory = $child['cat']; $downloadPercent = (($child['mb'] - $child['mbleft']) / $child['mb']) * 100; $progressBar = "progress-bar-striped active"; }
  2324. if($list == "history"){ $downloadName = $child['name']; $downloadCategory = $child['category']; $downloadPercent = "100"; $progressBar = ""; }
  2325. $downloadStatus = $child['status'];
  2326. $gotNZB[] = '<tr>
  2327. <td class="col-xs-7 nzbtable-file-row">'.$downloadName.'</td>
  2328. <td class="col-xs-2 nzbtable nzbtable-row">'.$downloadStatus.'</td>
  2329. <td class="col-xs-1 nzbtable nzbtable-row">'.$downloadCategory.'</td>
  2330. <td class="col-xs-2 nzbtable nzbtable-row">
  2331. <div class="progress">
  2332. <div class="progress-bar progress-bar-success '.$progressBar.'" role="progressbar" aria-valuenow="'.$downloadPercent.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$downloadPercent.'%">
  2333. <p class="text-center">'.round($downloadPercent).'%</p>
  2334. <span class="sr-only">'.$downloadPercent.'% Complete</span>
  2335. </div>
  2336. </div>
  2337. </td>
  2338. </tr>';
  2339. }
  2340. if ($gotNZB) {
  2341. return implode('',$gotNZB);
  2342. } else {
  2343. return '<tr><td colspan="4"><p class="text-center">No Results</p></td></tr>';
  2344. }
  2345. }
  2346. // Apply new tab settings
  2347. function updateTabs($tabs) {
  2348. if (!isset($GLOBALS['file_db'])) {
  2349. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2350. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2351. }
  2352. // Validate
  2353. if (!isset($tabs['defaultz'])) { $tabs['defaultz'][current(array_keys($tabs['name']))] = 'true'; }
  2354. if (isset($tabs['name']) && isset($tabs['url']) && is_array($tabs['name'])) {
  2355. // Clear Existing Tabs
  2356. $GLOBALS['file_db']->query("DELETE FROM tabs");
  2357. // Process New Tabs
  2358. $totalValid = 0;
  2359. foreach ($tabs['name'] as $key => $value) {
  2360. // Qualify
  2361. if (!$value || !isset($tabs['url']) || !$tabs['url'][$key]) { continue; }
  2362. $totalValid++;
  2363. $fields = array();
  2364. foreach(array('id','name','url','icon','iconurl','order','ping_url') as $v) {
  2365. if (isset($tabs[$v]) && isset($tabs[$v][$key])) { $fields[$v] = $tabs[$v][$key]; }
  2366. }
  2367. foreach(array('active','user','guest','defaultz','window','splash','ping') as $v) {
  2368. if (isset($tabs[$v]) && isset($tabs[$v][$key])) { $fields[$v] = ($tabs[$v][$key]!=='false'?'true':'false'); }
  2369. }
  2370. $GLOBALS['file_db']->query('INSERT INTO tabs (`'.implode('`,`',array_keys($fields)).'`) VALUES (\''.implode("','",$fields).'\');');
  2371. }
  2372. writeLog("success", "tabs successfully saved");
  2373. return $totalValid;
  2374. } else {
  2375. writeLog("error", "tabs could not save");
  2376. return false;
  2377. }
  2378. writeLog("error", "tabs could not save");
  2379. return false;
  2380. }
  2381. // ==============
  2382. function clean($strin) {
  2383. $strout = null;
  2384. for ($i = 0; $i < strlen($strin); $i++) {
  2385. $ord = ord($strin[$i]);
  2386. if (($ord > 0 && $ord < 32) || ($ord >= 127)) {
  2387. $strout .= "&amp;#{$ord};";
  2388. }
  2389. else {
  2390. switch ($strin[$i]) {
  2391. case '<':
  2392. $strout .= '&lt;';
  2393. break;
  2394. case '>':
  2395. $strout .= '&gt;';
  2396. break;
  2397. case '&':
  2398. $strout .= '&amp;';
  2399. break;
  2400. case '"':
  2401. $strout .= '&quot;';
  2402. break;
  2403. default:
  2404. $strout .= $strin[$i];
  2405. }
  2406. }
  2407. }
  2408. return $strout;
  2409. }
  2410. function registration_callback($username, $email, $userdir){
  2411. global $data;
  2412. $data = array($username, $email, $userdir);
  2413. }
  2414. function printArray($arrayName){
  2415. $messageCount = count($arrayName);
  2416. $i = 0;
  2417. foreach ( $arrayName as $item ) :
  2418. $i++;
  2419. if($i < $messageCount) :
  2420. echo "<small class='text-uppercase'>" . $item . "</small> & ";
  2421. elseif($i = $messageCount) :
  2422. echo "<small class='text-uppercase'>" . $item . "</small>";
  2423. endif;
  2424. endforeach;
  2425. }
  2426. function write_ini_file($content, $path) {
  2427. if (!$handle = fopen($path, 'w')) {
  2428. return false;
  2429. }
  2430. $success = fwrite($handle, trim($content));
  2431. fclose($handle);
  2432. return $success;
  2433. }
  2434. function gotTimezone(){
  2435. $regions = array(
  2436. 'Africa' => DateTimeZone::AFRICA,
  2437. 'America' => DateTimeZone::AMERICA,
  2438. 'Antarctica' => DateTimeZone::ANTARCTICA,
  2439. 'Arctic' => DateTimeZone::ARCTIC,
  2440. 'Asia' => DateTimeZone::ASIA,
  2441. 'Atlantic' => DateTimeZone::ATLANTIC,
  2442. 'Australia' => DateTimeZone::AUSTRALIA,
  2443. 'Europe' => DateTimeZone::EUROPE,
  2444. 'Indian' => DateTimeZone::INDIAN,
  2445. 'Pacific' => DateTimeZone::PACIFIC
  2446. );
  2447. $timezones = array();
  2448. foreach ($regions as $name => $mask) {
  2449. $zones = DateTimeZone::listIdentifiers($mask);
  2450. foreach($zones as $timezone) {
  2451. $time = new DateTime(NULL, new DateTimeZone($timezone));
  2452. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  2453. $timezones[$name][$timezone] = substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm;
  2454. }
  2455. }
  2456. print '<select name="timezone" id="timezone" class="form-control material input-sm" required>';
  2457. foreach($timezones as $region => $list) {
  2458. print '<optgroup label="' . $region . '">' . "\n";
  2459. foreach($list as $timezone => $name) {
  2460. if($timezone == TIMEZONE) : $selected = " selected"; else : $selected = ""; endif;
  2461. print '<option value="' . $timezone . '"' . $selected . '>' . $name . '</option>' . "\n";
  2462. }
  2463. print '</optgroup>' . "\n";
  2464. }
  2465. print '</select>';
  2466. }
  2467. function getTimezone(){
  2468. $regions = array(
  2469. 'Africa' => DateTimeZone::AFRICA,
  2470. 'America' => DateTimeZone::AMERICA,
  2471. 'Antarctica' => DateTimeZone::ANTARCTICA,
  2472. 'Arctic' => DateTimeZone::ARCTIC,
  2473. 'Asia' => DateTimeZone::ASIA,
  2474. 'Atlantic' => DateTimeZone::ATLANTIC,
  2475. 'Australia' => DateTimeZone::AUSTRALIA,
  2476. 'Europe' => DateTimeZone::EUROPE,
  2477. 'Indian' => DateTimeZone::INDIAN,
  2478. 'Pacific' => DateTimeZone::PACIFIC
  2479. );
  2480. $timezones = array();
  2481. foreach ($regions as $name => $mask) {
  2482. $zones = DateTimeZone::listIdentifiers($mask);
  2483. foreach($zones as $timezone) {
  2484. $time = new DateTime(NULL, new DateTimeZone($timezone));
  2485. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  2486. $timezones[$name][$timezone] = substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm;
  2487. }
  2488. }
  2489. print '<select name="timezone" id="timezone" class="form-control material" required>';
  2490. foreach($timezones as $region => $list) {
  2491. print '<optgroup label="' . $region . '">' . "\n";
  2492. foreach($list as $timezone => $name) {
  2493. print '<option value="' . $timezone . '">' . $name . '</option>' . "\n";
  2494. }
  2495. print '</optgroup>' . "\n";
  2496. }
  2497. print '</select>';
  2498. }
  2499. function explosion($string, $position){
  2500. $getWord = explode("|", $string);
  2501. return $getWord[$position];
  2502. }
  2503. function getServerPath() {
  2504. if (isset($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] == "https"){
  2505. $protocol = "https://";
  2506. }elseif (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
  2507. $protocol = "https://";
  2508. } else {
  2509. $protocol = "http://";
  2510. }
  2511. $domain = '';
  2512. if (isset($_SERVER['SERVER_NAME']) && strpos($_SERVER['SERVER_NAME'], '.') !== false){
  2513. $domain = $_SERVER['SERVER_NAME'];
  2514. }elseif(isset($_SERVER['HTTP_HOST'])){
  2515. if (strpos($_SERVER['HTTP_HOST'], ':') !== false) {
  2516. $domain = explode(':', $_SERVER['HTTP_HOST'])[0];
  2517. $port = explode(':', $_SERVER['HTTP_HOST'])[1];
  2518. if ($port == "80" || $port == "443"){
  2519. $domain = $domain;
  2520. }else{
  2521. $domain = $_SERVER['HTTP_HOST'];
  2522. }
  2523. }else{
  2524. $domain = $_SERVER['HTTP_HOST'];
  2525. }
  2526. }
  2527. return $protocol . $domain . dirname($_SERVER['REQUEST_URI']);
  2528. }
  2529. function get_browser_name() {
  2530. $user_agent = $_SERVER['HTTP_USER_AGENT'];
  2531. if (strpos($user_agent, 'Opera') || strpos($user_agent, 'OPR/')) return 'Opera';
  2532. elseif (strpos($user_agent, 'Edge')) return 'Edge';
  2533. elseif (strpos($user_agent, 'Chrome')) return 'Chrome';
  2534. elseif (strpos($user_agent, 'Safari')) return 'Safari';
  2535. elseif (strpos($user_agent, 'Firefox')) return 'Firefox';
  2536. elseif (strpos($user_agent, 'MSIE') || strpos($user_agent, 'Trident/7')) return 'Internet Explorer';
  2537. return 'Other';
  2538. }
  2539. function getSickrageCalendarWanted($array){
  2540. $array = json_decode($array, true);
  2541. //$gotCalendar = "";
  2542. $gotCalendar = array();
  2543. $i = 0;
  2544. foreach($array['data']['missed'] AS $child) {
  2545. $i++;
  2546. $seriesName = $child['show_name'];
  2547. $episodeID = $child['tvdbid'];
  2548. $episodeAirDate = $child['airdate'];
  2549. $episodeAirDateTime = explode(" ",$child['airs']);
  2550. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2551. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2552. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2553. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2554. $downloaded = "0";
  2555. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2556. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2557. array_push($gotCalendar, array(
  2558. "id" => "Sick-Miss-".$i,
  2559. "title" => $seriesName,
  2560. "start" => $episodeAirDate,
  2561. "className" => $downloaded." tvID--".$episodeID,
  2562. "imagetype" => "tv",
  2563. ));
  2564. }
  2565. foreach($array['data']['today'] AS $child) {
  2566. $i++;
  2567. $seriesName = $child['show_name'];
  2568. $episodeID = $child['tvdbid'];
  2569. $episodeAirDate = $child['airdate'];
  2570. $episodeAirDateTime = explode(" ",$child['airs']);
  2571. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2572. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2573. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2574. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2575. $downloaded = "0";
  2576. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2577. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2578. array_push($gotCalendar, array(
  2579. "id" => "Sick-Today-".$i,
  2580. "title" => $seriesName,
  2581. "start" => $episodeAirDate,
  2582. "className" => $downloaded." tvID--".$episodeID,
  2583. "imagetype" => "tv",
  2584. ));
  2585. }
  2586. foreach($array['data']['soon'] AS $child) {
  2587. $i++;
  2588. $seriesName = $child['show_name'];
  2589. $episodeID = $child['tvdbid'];
  2590. $episodeAirDate = $child['airdate'];
  2591. $episodeAirDateTime = explode(" ",$child['airs']);
  2592. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2593. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2594. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2595. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2596. $downloaded = "0";
  2597. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2598. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2599. array_push($gotCalendar, array(
  2600. "id" => "Sick-Soon-".$i,
  2601. "title" => $seriesName,
  2602. "start" => $episodeAirDate,
  2603. "className" => $downloaded." tvID--".$episodeID,
  2604. "imagetype" => "tv",
  2605. ));
  2606. }
  2607. foreach($array['data']['later'] AS $child) {
  2608. $i++;
  2609. $seriesName = $child['show_name'];
  2610. $episodeID = $child['tvdbid'];
  2611. $episodeAirDate = $child['airdate'];
  2612. $episodeAirDateTime = explode(" ",$child['airs']);
  2613. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2614. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2615. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2616. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2617. $downloaded = "0";
  2618. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2619. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2620. array_push($gotCalendar, array(
  2621. "id" => "Sick-Later-".$i,
  2622. "title" => $seriesName,
  2623. "start" => $episodeAirDate,
  2624. "className" => $downloaded." tvID--".$episodeID,
  2625. "imagetype" => "tv",
  2626. ));
  2627. }
  2628. if ($i != 0){ return $gotCalendar; }
  2629. }
  2630. function getSickrageCalendarHistory($array){
  2631. $array = json_decode($array, true);
  2632. //$gotCalendar = "";
  2633. $gotCalendar = array();
  2634. $i = 0;
  2635. foreach($array['data'] AS $child) {
  2636. $i++;
  2637. $seriesName = $child['show_name'];
  2638. $episodeID = $child['tvdbid'];
  2639. $episodeAirDate = $child['date'];
  2640. $downloaded = "green-bg";
  2641. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2642. array_push($gotCalendar, array(
  2643. "id" => "Sick-History-".$i,
  2644. "title" => $seriesName,
  2645. "start" => $episodeAirDate,
  2646. "className" => $downloaded." tvID--".$episodeID,
  2647. "imagetype" => "tv",
  2648. ));
  2649. }
  2650. if ($i != 0){ return $gotCalendar; }
  2651. }
  2652. function getSonarrCalendar($array){
  2653. $array = json_decode($array, true);
  2654. //$gotCalendar = "";
  2655. $gotCalendar = array();
  2656. $i = 0;
  2657. foreach($array AS $child) {
  2658. $i++;
  2659. $seriesName = $child['series']['title'];
  2660. $episodeID = $child['series']['tvdbId'];
  2661. if(!isset($episodeID)){ $episodeID = ""; }
  2662. $episodeName = htmlentities($child['title'], ENT_QUOTES);
  2663. if($child['episodeNumber'] == "1"){ $episodePremier = "true"; }else{ $episodePremier = "false"; }
  2664. $episodeAirDate = $child['airDateUtc'];
  2665. $episodeAirDate = strtotime($episodeAirDate);
  2666. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2667. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2668. $downloaded = $child['hasFile'];
  2669. if($downloaded == "0" && isset($unaired) && $episodePremier == "true"){ $downloaded = "light-blue-bg"; }elseif($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2670. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2671. array_push($gotCalendar, array(
  2672. "id" => "Sonarr-".$i,
  2673. "title" => $seriesName,
  2674. "start" => $episodeAirDate,
  2675. "className" => $downloaded." tvID--".$episodeID,
  2676. "imagetype" => "tv",
  2677. ));
  2678. }
  2679. if ($i != 0){ return $gotCalendar; }
  2680. }
  2681. function getCouchCalendar(){
  2682. $url = qualifyURL(COUCHURL);
  2683. $api = curl_get($url."/api/".COUCHAPI."/media.list");
  2684. $api = json_decode($api, true);
  2685. $i = 0;
  2686. $gotCalendar = array();
  2687. if (is_array($api) || is_object($api)){
  2688. foreach($api['movies'] AS $child) {
  2689. if($child['status'] == "active" || $child['status'] == "done" ){
  2690. $i++;
  2691. $movieName = $child['info']['original_title'];
  2692. $movieID = $child['info']['tmdb_id'];
  2693. if(!isset($movieID)){ $movieID = ""; }
  2694. $physicalRelease = (isset($child['info']['released']) ? $child['info']['released'] : null);
  2695. $backupRelease = (isset($child['info']['release_date']['theater']) ? $child['info']['release_date']['theater'] : null);
  2696. $physicalRelease = (isset($physicalRelease) ? $physicalRelease : $backupRelease);
  2697. $physicalRelease = strtotime($physicalRelease);
  2698. $physicalRelease = date("Y-m-d", $physicalRelease);
  2699. if (new DateTime() < new DateTime($physicalRelease)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2700. $downloaded = ($child['status'] == "active") ? "0" : "1";
  2701. if($downloaded == "0" && $notReleased == "true"){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg"; }else{ $downloaded = "red-bg"; }
  2702. array_push($gotCalendar, array(
  2703. "id" => "CouchPotato-".$i,
  2704. "title" => $movieName,
  2705. "start" => $physicalRelease,
  2706. "className" => $downloaded." movieID--".$movieID,
  2707. "imagetype" => "film",
  2708. ));
  2709. }
  2710. }
  2711. if ($i != 0){ return $gotCalendar; }
  2712. }else{
  2713. writeLog("error", "CouchPotato ERROR: could not connect - check URL and/or check API key - if HTTPS, is cert valid");
  2714. }
  2715. }
  2716. function getRadarrCalendar($array){
  2717. $array = json_decode($array, true);
  2718. $gotCalendar = array();
  2719. $i = 0;
  2720. foreach($array AS $child) {
  2721. if(isset($child['physicalRelease'])){
  2722. $i++;
  2723. $movieName = $child['title'];
  2724. $movieID = $child['tmdbId'];
  2725. if(!isset($movieID)){ $movieID = ""; }
  2726. $physicalRelease = $child['physicalRelease'];
  2727. $physicalRelease = strtotime($physicalRelease);
  2728. $physicalRelease = date("Y-m-d", $physicalRelease);
  2729. if (new DateTime() < new DateTime($physicalRelease)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2730. $downloaded = $child['hasFile'];
  2731. if($downloaded == "0" && $notReleased == "true"){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg"; }else{ $downloaded = "red-bg"; }
  2732. array_push($gotCalendar, array(
  2733. "id" => "Radarr-".$i,
  2734. "title" => $movieName,
  2735. "start" => $physicalRelease,
  2736. "className" => $downloaded." movieID--".$movieID,
  2737. "imagetype" => "film",
  2738. ));
  2739. }
  2740. }
  2741. if ($i != 0){ return $gotCalendar; }
  2742. }
  2743. function getHeadphonesCalendar($url, $key, $list){
  2744. $url = qualifyURL(HEADPHONESURL);
  2745. $api = curl_get($url."/api?apikey=".$key."&cmd=$list");
  2746. $api = json_decode($api, true);
  2747. $i = 0;
  2748. //$gotCalendar = "";
  2749. $gotCalendar = array();;
  2750. if (is_array($api) || is_object($api)){
  2751. foreach($api AS $child) {
  2752. if($child['Status'] == "Wanted" && $list == "getWanted" && $child['ReleaseDate']){
  2753. $i++;
  2754. $albumName = addslashes($child['AlbumTitle']);
  2755. $albumArtist = htmlentities($child['ArtistName'], ENT_QUOTES);
  2756. $albumDate = (strlen($child['ReleaseDate']) > 4) ? $child['ReleaseDate'] : $child['ReleaseDate']."-01-01";
  2757. $albumID = $child['AlbumID'];
  2758. $albumDate = strtotime($albumDate);
  2759. $albumDate = date("Y-m-d", $albumDate);
  2760. $albumStatus = $child['Status'];
  2761. if (new DateTime() < new DateTime($albumDate)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2762. if($albumStatus == "Wanted" && $notReleased == "true"){ $albumStatusColor = "indigo-bg"; }elseif($albumStatus == "Downloaded"){ $albumStatusColor = "green-bg"; }else{ $albumStatusColor = "red-bg"; }
  2763. //$gotCalendar .= "{ title: \"$albumArtist - $albumName\", start: \"$albumDate\", className: \"$albumStatusColor\", imagetype: \"music\", url: \"https://musicbrainz.org/release-group/$albumID\" }, \n";
  2764. array_push($gotCalendar, array(
  2765. "id" => "Headphones-".$i,
  2766. "title" => $albumArtist.' - '.$albumName,
  2767. "start" => $albumDate,
  2768. "className" => $albumStatusColor,
  2769. "imagetype" => "music",
  2770. 'url' => "https://musicbrainz.org/release-group/".$albumID,
  2771. ));
  2772. }
  2773. if($child['Status'] == "Processed" && $list == "getHistory"){
  2774. $i++;
  2775. $find = array('_','[', ']', '\n');
  2776. $replace = array(' ','(', ')', ' ');
  2777. $albumName = addslashes(str_replace($find,$replace,$child['FolderName']));
  2778. $albumDate = $child['DateAdded'];
  2779. $albumID = $child['AlbumID'];
  2780. $albumDate = strtotime($albumDate);
  2781. $albumDate = date("Y-m-d", $albumDate);
  2782. $albumStatusColor = "green-bg";
  2783. if (new DateTime() < new DateTime($albumDate)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2784. //$gotCalendar .= "{ title: \"$albumName\", start: \"$albumDate\", className: \"$albumStatusColor\", imagetype: \"music\", url: \"https://musicbrainz.org/release-group/$albumID\" }, \n";
  2785. array_push($gotCalendar, array(
  2786. "id" => "Headphones-".$i,
  2787. "title" => $albumName,
  2788. "start" => $albumDate,
  2789. "className" => $albumStatusColor,
  2790. "imagetype" => "music",
  2791. 'url' => "https://musicbrainz.org/release-group/".$albumID,
  2792. ));
  2793. }
  2794. }
  2795. if ($i != 0){ return $gotCalendar; }
  2796. }else{
  2797. writeLog("error", "HEADPHONES $list ERROR: could not connect - check URL and/or check API key - if HTTPS, is cert valid");
  2798. }
  2799. }
  2800. function checkRootPath($string){
  2801. if($string == "\\" || $string == "/"){
  2802. return "/";
  2803. }else{
  2804. return str_replace("\\", "/", $string) . "/";
  2805. }
  2806. }
  2807. function strip($string){
  2808. $string = strip_tags($string);
  2809. return preg_replace('/[ \t]+/', ' ', preg_replace('/\s*$^\s*/m', "\n", $string));
  2810. }
  2811. function writeLog($type, $message){
  2812. if(filesize("org.log") > 500000){
  2813. rename('org.log','org['.date('Y-m-d').'].log');
  2814. $message2 = date("Y-m-d H:i:s")."|".$type."|".strip("ORG LOG: Creating backup of org.log to org[".date('Y-m-d')."].log ")."\n";
  2815. file_put_contents("org.log", $message2, FILE_APPEND | LOCK_EX);
  2816. }
  2817. $message = date("Y-m-d H:i:s")."|".$type."|".strip($message)."\n";
  2818. file_put_contents("org.log", $message, FILE_APPEND | LOCK_EX);
  2819. }
  2820. function readLog(){
  2821. $log = file("org.log",FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
  2822. $log = array_reverse($log);
  2823. foreach($log as $line){
  2824. if(substr_count($line, '|') == 2){
  2825. $line = explode("|", strip($line));
  2826. $line[1] = ($line[1] == "error") ? '<span class="label label-danger">Error</span>' : '<span class="label label-primary">Success</span>';
  2827. echo "<tr><td>".$line[0]."</td><td>".$line[2]."</td><td>".$line[1]."</td></tr>";
  2828. }
  2829. }
  2830. }
  2831. function buildStream($array){
  2832. $result = "";
  2833. if (array_key_exists('platform', $array)) {
  2834. $result .= '<div class="reg-info" style="margin-top:0; padding-left:0; position: absolute; bottom: 10px; left: 10px;"><div style="margin-right: 0;" class="item pull-left text-center"><img alt="'.$array['platform'].'" class="img-circle" height="55px" src="images/platforms/'.getPlatform($array['platform']).'"></div></div><div class="clearfix"></div>';
  2835. }
  2836. if (array_key_exists('device', $array)) {
  2837. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-laptop fa-fw"></i>'.$array['device'].'</span></div></div><div class="clearfix"></div>';
  2838. }
  2839. if (array_key_exists('stream', $array)) {
  2840. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-play fa-fw"></i>'.$array['stream'].'</span></div></div><div class="clearfix"></div>';
  2841. }
  2842. if (array_key_exists('video', $array)) {
  2843. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-film fa-fw"></i>'.$array['video'].'</span></div></div><div class="clearfix"></div>';
  2844. }
  2845. if (array_key_exists('audio', $array)) {
  2846. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-volume-up fa-fw"></i>'.$array['audio'].'</span></div></div><div class="clearfix"></div>';
  2847. }
  2848. return $result;
  2849. }
  2850. function streamType($value){
  2851. if($value == "transcode" || $value == "Transcode"){
  2852. return "Transcode";
  2853. }elseif($value == "copy" || $value == "DirectStream"){
  2854. return "Direct Stream";
  2855. }elseif($value == "directplay" || $value == "DirectPlay"){
  2856. return "Direct Play";
  2857. }else{
  2858. return "Direct Play";
  2859. }
  2860. }
  2861. function getPlatform($platform){
  2862. $allPlatforms = array(
  2863. "Chrome" => "chrome.png",
  2864. "tvOS" => "atv.png",
  2865. "iOS" => "ios.png",
  2866. "Xbox One" => "xbox.png",
  2867. "Mystery 4" => "playstation.png",
  2868. "Samsung" => "samsung.png",
  2869. "Roku" => "roku.png",
  2870. "Emby for iOS" => "ios.png",
  2871. "Emby Mobile" => "emby.png",
  2872. "Emby Theater" => "emby.png",
  2873. "Emby Classic" => "emby.png",
  2874. "Safari" => "safari.png",
  2875. "Android" => "android.png",
  2876. "AndroidTv" => "android.png",
  2877. "Chromecast" => "chromecast.png",
  2878. "Dashboard" => "emby.png",
  2879. "Dlna" => "dlna.png",
  2880. "Windows Phone" => "wp.png",
  2881. "Windows RT" => "win8.png",
  2882. "Kodi" => "kodi.png",
  2883. );
  2884. if (array_key_exists($platform, $allPlatforms)) {
  2885. return $allPlatforms[$platform];
  2886. }else{
  2887. return "pmp.png";
  2888. }
  2889. }
  2890. function getServer(){
  2891. $server = isset($_SERVER["HTTP_HOST"]) ? $_SERVER["HTTP_HOST"] : $_SERVER["SERVER_NAME"];
  2892. return $server;
  2893. }
  2894. function prettyPrint($array) {
  2895. echo "<pre>";
  2896. print_r($array);
  2897. echo "</pre>";
  2898. echo "<br/>";
  2899. }
  2900. function checkFrame($array, $url){
  2901. if(array_key_exists("x-frame-options", $array)){
  2902. if($array['x-frame-options'] == "deny"){
  2903. return false;
  2904. }elseif($array['x-frame-options'] == "sameorgin"){
  2905. $digest = parse_url($url);
  2906. $host = (isset($digest['host'])?$digest['host']:'');
  2907. if(getServer() == $host){
  2908. return true;
  2909. }else{
  2910. return false;
  2911. }
  2912. }
  2913. }else{
  2914. if(!$array){
  2915. return false;
  2916. }
  2917. return true;
  2918. }
  2919. }
  2920. function frameTest($url){
  2921. $array = array_change_key_case(get_headers(qualifyURL($url), 1));
  2922. $url = qualifyURL($url);
  2923. if(checkFrame($array, $url)){
  2924. return true;
  2925. }else{
  2926. return false;
  2927. }
  2928. }
  2929. function sendResult($result, $icon = "floppy-o", $message = false, $success = "WAS_SUCCESSFUL", $fail = "HAS_FAILED", $send = true) {
  2930. $notifyExplode = explode("-", NOTIFYEFFECT);
  2931. if ($result) {
  2932. $msg = array(
  2933. 'html' => ($message?''.$message.' <strong>'.translate($success).'</strong>':'<strong>'.translate($success).'</strong>'),
  2934. 'icon' => $icon,
  2935. 'type' => 'success',
  2936. 'length' => '5000',
  2937. 'layout' => $notifyExplode[0],
  2938. 'effect' => $notifyExplode[1],
  2939. );
  2940. } else {
  2941. $msg = array(
  2942. 'html' => ($message?''.$message.' <strong>'.translate($fail).'</strong>':'<strong>'.translate($fail).'</strong>'),
  2943. 'icon' => $icon,
  2944. 'type' => 'error',
  2945. 'length' => '5000',
  2946. 'layout' => $notifyExplode[0],
  2947. 'effect' => $notifyExplode[1],
  2948. );
  2949. }
  2950. // Send and kill script?
  2951. if ($send) {
  2952. header('Content-Type: application/json');
  2953. echo json_encode(array('notify'=>$msg));
  2954. die();
  2955. }
  2956. return $msg;
  2957. }
  2958. function buildHomepageNotice($layout, $type, $title, $message){
  2959. switch ($layout) {
  2960. case 'elegant':
  2961. return '
  2962. <div id="homepageNotice" class="row">
  2963. <div class="col-lg-12">
  2964. <div class="content-box big-box box-shadow panel-box panel-'.$type.'">
  2965. <div class="content-title i-block">
  2966. <h4 class="zero-m"><strong>'.$title.'</strong></h4>
  2967. <div class="content-tools i-block pull-right">
  2968. <a class="close-btn">
  2969. <i class="fa fa-times"></i>
  2970. </a>
  2971. </div>
  2972. </div>
  2973. '.$message.'
  2974. </div>
  2975. </div>
  2976. </div>
  2977. ';
  2978. break;
  2979. case 'basic':
  2980. return '
  2981. <div id="homepageNotice" class="row">
  2982. <div class="col-lg-12">
  2983. <div class="panel panel-'.$type.'">
  2984. <div class="panel-heading">
  2985. <h3 class="panel-title">'.$title.'</h3>
  2986. </div>
  2987. <div class="panel-body">
  2988. '.$message.'
  2989. </div>
  2990. </div>
  2991. </div>
  2992. </div>
  2993. ';
  2994. break;
  2995. case 'jumbotron';
  2996. return '
  2997. <div id="homepageNotice" class="row">
  2998. <div class="col-lg-12">
  2999. <div class="jumbotron">
  3000. <div class="container">
  3001. <h1>'.$title.'</h1>
  3002. <p>'.$message.'</p>
  3003. </div>
  3004. </div>
  3005. </div>
  3006. </div>
  3007. ';
  3008. }
  3009. }
  3010. function embyArray($array, $type) {
  3011. $key = ($type == "video" ? "Height" : "Channels");
  3012. if (array_key_exists($key, $array)) {
  3013. switch ($type) {
  3014. case "video":
  3015. $codec = $array["Codec"];
  3016. $height = $array["Height"];
  3017. $width = $array["Width"];
  3018. break;
  3019. default:
  3020. $codec = $array["Codec"];
  3021. $channels = $array["Channels"];
  3022. }
  3023. return ($type == "video" ? "(".$codec.") (".$width."x".$height.")" : "(".$codec.") (".$channels."ch)");
  3024. }
  3025. foreach ($array as $element) {
  3026. if (is_array($element)) {
  3027. if (embyArray($element, $type)) {
  3028. return embyArray($element, $type);
  3029. }
  3030. }
  3031. }
  3032. }
  3033. // Get Now Playing Streams From Plex
  3034. function searchPlex($query){
  3035. $address = qualifyURL(PLEXURL);
  3036. $openTab = (PLEXTABNAME) ? "true" : "false";
  3037. // Perform API requests
  3038. $api = @curl_get($address."/search?query=".rawurlencode($query)."&X-Plex-Token=".PLEXTOKEN);
  3039. libxml_use_internal_errors(true);
  3040. $api = simplexml_load_string($api);
  3041. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3042. if (!$getServer) { return 'Could not load!'; }
  3043. // Identify the local machine
  3044. $server = $getServer['machineIdentifier'];
  3045. $pre = "<table class=\"table table-hover table-stripped\"><thead><tr><th>Cover</th><th>Title</th><th>Genre</th><th>Year</th><th>Type</th><th>Added</th><th>Extra Info</th></tr></thead><tbody>";
  3046. $items = "";
  3047. $albums = $movies = $shows = 0;
  3048. $style = 'style="vertical-align: middle"';
  3049. foreach($api AS $child) {
  3050. if($child['type'] != "artist" && $child['type'] != "episode" && isset($child['librarySectionID'])){
  3051. $time = (string)$child['addedAt'];
  3052. $time = new DateTime("@$time");
  3053. $results = array(
  3054. "title" => (string)$child['title'],
  3055. "image" => (string)$child['thumb'],
  3056. "type" => (string)ucwords($child['type']),
  3057. "year" => (string)$child['year'],
  3058. "key" => (string)$child['ratingKey']."-search",
  3059. "ratingkey" => (string)$child['ratingKey'],
  3060. "genre" => (string)$child->Genre['tag'],
  3061. "added" => $time->format('Y-m-d'),
  3062. "extra" => "",
  3063. );
  3064. switch ($child['type']){
  3065. case "album":
  3066. $push = array(
  3067. "title" => (string)$child['parentTitle']." - ".(string)$child['title'],
  3068. );
  3069. $results = array_replace($results,$push);
  3070. $albums++;
  3071. break;
  3072. case "movie":
  3073. $push = array(
  3074. "extra" => "Content Rating: ".(string)$child['contentRating']."<br/>Movie Rating: ".(string)$child['rating'],
  3075. );
  3076. $results = array_replace($results,$push);
  3077. $movies++;
  3078. break;
  3079. case "show":
  3080. $push = array(
  3081. "extra" => "Seasons: ".(string)$child['childCount']."<br/>Episodes: ".(string)$child['leafCount'],
  3082. );
  3083. $results = array_replace($results,$push);
  3084. $shows++;
  3085. break;
  3086. }
  3087. if (file_exists('images/cache/'.$results['key'].'.jpg')){ $image_url = 'images/cache/'.$results['key'].'.jpg'; }
  3088. if (file_exists('images/cache/'.$results['key'].'.jpg') && (time() - 604800) > filemtime('images/cache/'.$results['key'].'.jpg') || !file_exists('images/cache/'.$results['key'].'.jpg')) {
  3089. $image_url = 'ajax.php?a=plex-image&img='.$results['image'].'&height=150&width=100&key='.$results['key'];
  3090. }
  3091. if(!$results['image']){ $image_url = "images/no-search.png"; $key = "no-search"; }
  3092. if (substr_count(PLEXURL, '.') != 2) {
  3093. $link = "https://app.plex.tv/web/app#!/server/$server/details?key=/library/metadata/".$results['ratingkey'];
  3094. }else{
  3095. $link = PLEXURL."/web/index.html#!/server/$server/details?key=/library/metadata/".$results['ratingkey'];
  3096. }
  3097. $items .= '<tr style="cursor: pointer;" class="openTab" extraTitle="'.$results['title'].'" extraType="'.$child['type'].'" openTab="'.$openTab.'" href="'.$link.'">
  3098. <th scope="row"><img src="'.$image_url.'"></th>
  3099. <td class="col-xs-2 nzbtable nzbtable-row"'.$style.'>'.$results['title'].'</td>
  3100. <td class="col-xs-3 nzbtable nzbtable-row"'.$style.'>'.$results['genre'].'</td>
  3101. <td class="col-xs-1 nzbtable nzbtable-row"'.$style.'>'.$results['year'].'</td>
  3102. <td class="col-xs-1 nzbtable nzbtable-row"'.$style.'>'.$results['type'].'</td>
  3103. <td class="col-xs-3 nzbtable nzbtable-row"'.$style.'>'.$results['added'].'</td>
  3104. <td class="col-xs-2 nzbtable nzbtable-row"'.$style.'>'.$results['extra'].'</td>
  3105. </tr>';
  3106. }
  3107. }
  3108. $totals = '<div style="margin: 10px;" class="sort-todo pull-right">
  3109. <span class="badge gray-bg"><i class="fa fa-film fa-2x white"></i><strong style="
  3110. font-size: 23px;
  3111. ">&nbsp;'.$movies.'</strong></span>
  3112. <span class="badge gray-bg"><i class="fa fa-tv fa-2x white"></i><strong style="
  3113. font-size: 23px;
  3114. ">&nbsp;'.$shows.'</strong></span>
  3115. <span class="badge gray-bg"><i class="fa fa-music fa-2x white"></i><strong style="
  3116. font-size: 23px;
  3117. ">&nbsp;'.$albums.'</strong></span>
  3118. </div>';
  3119. return (!empty($items) ? $totals.$pre.$items."</div></table>" : "<h2 class='text-center'>No Results for $query</h2>" );
  3120. }
  3121. function getBannedUsers($string){
  3122. if (strpos($string, ',') !== false) {
  3123. $banned = explode(",", $string);
  3124. }else{
  3125. $banned = array($string);
  3126. }
  3127. return $banned;
  3128. }
  3129. function getWhitelist($string){
  3130. if (strpos($string, ',') !== false) {
  3131. $whitelist = explode(",", $string);
  3132. }else{
  3133. $whitelist = array($string);
  3134. }
  3135. foreach($whitelist as &$ip){
  3136. $ip = is_numeric(substr($ip, 0, 1)) ? $ip : gethostbyname($ip);
  3137. }
  3138. return $whitelist;
  3139. }
  3140. function get_client_ip() {
  3141. $ipaddress = '';
  3142. if (isset($_SERVER['HTTP_CLIENT_IP']))
  3143. $ipaddress = $_SERVER['HTTP_CLIENT_IP'];
  3144. else if(isset($_SERVER['HTTP_X_FORWARDED_FOR']))
  3145. $ipaddress = $_SERVER['HTTP_X_FORWARDED_FOR'];
  3146. else if(isset($_SERVER['HTTP_X_FORWARDED']))
  3147. $ipaddress = $_SERVER['HTTP_X_FORWARDED'];
  3148. else if(isset($_SERVER['HTTP_FORWARDED_FOR']))
  3149. $ipaddress = $_SERVER['HTTP_FORWARDED_FOR'];
  3150. else if(isset($_SERVER['HTTP_FORWARDED']))
  3151. $ipaddress = $_SERVER['HTTP_FORWARDED'];
  3152. else if(isset($_SERVER['REMOTE_ADDR']))
  3153. $ipaddress = $_SERVER['REMOTE_ADDR'];
  3154. else
  3155. $ipaddress = 'UNKNOWN';
  3156. return $ipaddress;
  3157. }
  3158. //EMAIL SHIT
  3159. function sendEmail($email, $username = "Organizr User", $subject, $body, $cc = null, $bcc = null){
  3160. $mail = new PHPMailer;
  3161. $mail->isSMTP();
  3162. $mail->Host = SMTPHOST;
  3163. $mail->SMTPAuth = SMTPHOSTAUTH;
  3164. $mail->Username = SMTPHOSTUSERNAME;
  3165. $mail->Password = SMTPHOSTPASSWORD;
  3166. $mail->SMTPSecure = SMTPHOSTTYPE;
  3167. $mail->Port = SMTPHOSTPORT;
  3168. $mail->setFrom(SMTPHOSTSENDEREMAIL, SMTPHOSTSENDERNAME);
  3169. $mail->addReplyTo(SMTPHOSTSENDEREMAIL, SMTPHOSTSENDERNAME);
  3170. $mail->isHTML(true);
  3171. if($email){
  3172. $mail->addAddress($email, $username);
  3173. }
  3174. if($cc){
  3175. $mail->addCC($cc);
  3176. }
  3177. if($bcc){
  3178. if(strpos($bcc , ',') === false){
  3179. $mail->addBCC($bcc);
  3180. }else{
  3181. $allEmails = explode(",",$bcc);
  3182. foreach($allEmails as $gotEmail){
  3183. $mail->addBCC($gotEmail);
  3184. }
  3185. }
  3186. }
  3187. $mail->Subject = $subject;
  3188. $mail->Body = $body;
  3189. //$mail->send();
  3190. if(!$mail->send()) {
  3191. writeLog("error", "mail failed to send");
  3192. } else {
  3193. writeLog("success", "mail has been sent");
  3194. }
  3195. }
  3196. //EMAIL SHIT
  3197. function sendTestEmail($to, $from, $host, $auth, $username, $password, $type, $port, $sendername){
  3198. $mail = new PHPMailer;
  3199. $mail->isSMTP();
  3200. $mail->Host = $host;
  3201. $mail->SMTPAuth = $auth;
  3202. $mail->Username = $username;
  3203. $mail->Password = $password;
  3204. $mail->SMTPSecure = $type;
  3205. $mail->Port = $port;
  3206. $mail->setFrom($from, $sendername);
  3207. $mail->addReplyTo($from, $sendername);
  3208. $mail->isHTML(true);
  3209. $mail->addAddress($to, "Organizr Admin");
  3210. $mail->Subject = "Organizr Test E-Mail";
  3211. $mail->Body = "This was just a test!";
  3212. //$mail->send();
  3213. if(!$mail->send()) {
  3214. writeLog("error", "EMAIL TEST: mail failed to send - Error:".$mail->ErrorInfo);
  3215. return false;
  3216. } else {
  3217. writeLog("success", "EMAIL TEST: mail has been sent successfully");
  3218. return true;
  3219. }
  3220. }
  3221. function libraryList(){
  3222. $address = qualifyURL(PLEXURL);
  3223. $headers = array(
  3224. "Accept" => "application/json",
  3225. "X-Plex-Token" => PLEXTOKEN
  3226. );
  3227. libxml_use_internal_errors(true);
  3228. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3229. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3230. $api = simplexml_load_string(@curl_get("https://plex.tv/api/servers/$gotServer/shared_servers", $headers));
  3231. $libraryList = array();
  3232. foreach($api->SharedServer->Section AS $child) {
  3233. $libraryList['libraries'][(string)$child['title']] = (string)$child['id'];
  3234. }
  3235. foreach($api->SharedServer AS $child) {
  3236. if(!empty($child['username'])){
  3237. $username = (string)strtolower($child['username']);
  3238. $email = (string)strtolower($child['email']);
  3239. $libraryList['users'][$username] = (string)$child['id'];
  3240. $libraryList['emails'][$email] = (string)$child['id'];
  3241. $libraryList['both'][$username] = $email;
  3242. }
  3243. }
  3244. return (!empty($libraryList) ? array_change_key_case($libraryList,CASE_LOWER) : null );
  3245. }
  3246. function plexUserShare($username){
  3247. $address = qualifyURL(PLEXURL);
  3248. $headers = array(
  3249. "Accept" => "application/json",
  3250. "Content-Type" => "application/json",
  3251. "X-Plex-Token" => PLEXTOKEN
  3252. );
  3253. libxml_use_internal_errors(true);
  3254. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3255. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3256. $json = array(
  3257. "server_id" => $gotServer,
  3258. "shared_server" => array(
  3259. //"library_section_ids" => "[26527637]",
  3260. "invited_email" => $username
  3261. )
  3262. );
  3263. $api = curl_post("https://plex.tv/api/servers/$gotServer/shared_servers/", $json, $headers);
  3264. switch ($api['http_code']['http_code']){
  3265. case 400:
  3266. writeLog("error", "PLEX INVITE: $username already has access to the shared libraries");
  3267. $result = "$username already has access to the shared libraries";
  3268. break;
  3269. case 401:
  3270. writeLog("error", "PLEX INVITE: Invalid Plex Token");
  3271. $result = "Invalid Plex Token";
  3272. break;
  3273. case 200:
  3274. writeLog("success", "PLEX INVITE: $username now has access to your Plex Library");
  3275. $result = "$username now has access to your Plex Library";
  3276. break;
  3277. default:
  3278. writeLog("error", "PLEX INVITE: unknown error");
  3279. $result = false;
  3280. }
  3281. return (!empty($result) ? $result : null );
  3282. }
  3283. function plexUserDelete($username){
  3284. $address = qualifyURL(PLEXURL);
  3285. $headers = array(
  3286. "Accept" => "application/json",
  3287. "Content-Type" => "application/json",
  3288. "X-Plex-Token" => PLEXTOKEN
  3289. );
  3290. libxml_use_internal_errors(true);
  3291. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3292. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3293. $id = (is_numeric($username) ? $id : convertPlexName($username, "id"));
  3294. $api = curl_delete("https://plex.tv/api/servers/$gotServer/shared_servers/$id", $headers);
  3295. switch ($api['http_code']['http_code']){
  3296. case 401:
  3297. writeLog("error", "PLEX INVITE: Invalid Plex Token");
  3298. $result = "Invalid Plex Token";
  3299. break;
  3300. case 200:
  3301. writeLog("success", "PLEX INVITE: $username doesn't have access to your Plex Library anymore");
  3302. $result = "$username doesn't have access to your Plex Library anymore";
  3303. break;
  3304. default:
  3305. writeLog("error", "PLEX INVITE: unknown error");
  3306. $result = false;
  3307. }
  3308. return (!empty($result) ? $result : null );
  3309. }
  3310. function convertPlexName($user, $type){
  3311. $array = libraryList();
  3312. switch ($type){
  3313. case "username":
  3314. $plexUser = array_search ($user, $array['users']);
  3315. break;
  3316. case "id":
  3317. if (array_key_exists(strtolower($user), $array['users'])) {
  3318. $plexUser = $array['users'][strtolower($user)];
  3319. }
  3320. break;
  3321. default:
  3322. $plexUser = false;
  3323. }
  3324. return (!empty($plexUser) ? $plexUser : null );
  3325. }
  3326. function randomCode($length = 5, $type = null) {
  3327. switch ($type){
  3328. case "alpha":
  3329. $legend = array_merge(range('A', 'Z'));
  3330. break;
  3331. case "numeric":
  3332. $legend = array_merge(range(0,9));
  3333. break;
  3334. default:
  3335. $legend = array_merge(range(0,9),range('A', 'Z'));
  3336. }
  3337. $code = "";
  3338. for($i=0; $i < $length; $i++) {
  3339. $code .= $legend[mt_rand(0, count($legend) - 1)];
  3340. }
  3341. return $code;
  3342. }
  3343. function inviteCodes($action, $code = null, $usedBy = null) {
  3344. if (!isset($GLOBALS['file_db'])) {
  3345. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  3346. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  3347. }
  3348. $now = date("Y-m-d H:i:s");
  3349. switch ($action) {
  3350. case "get":
  3351. // Start Array
  3352. $result = array();
  3353. // Database Lookup
  3354. $invites = $GLOBALS['file_db']->query('SELECT * FROM invites WHERE valid = "Yes"');
  3355. // Get Codes
  3356. foreach($invites as $row) {
  3357. array_push($result, $row['code']);
  3358. }
  3359. // Return the Results
  3360. return (!empty($result) ? $result : false );
  3361. break;
  3362. case "check":
  3363. // Start Array
  3364. $result = array();
  3365. // Database Lookup
  3366. $invites = $GLOBALS['file_db']->query('SELECT * FROM invites WHERE valid = "Yes" AND code = "'.$code.'"');
  3367. // Get Codes
  3368. foreach($invites as $row) {
  3369. $result = $row['code'];
  3370. }
  3371. // Return the Results
  3372. return (!empty($result) ? $result : false );
  3373. break;
  3374. case "use":
  3375. $currentIP = get_client_ip();
  3376. $invites = $GLOBALS['file_db']->query('UPDATE invites SET valid = "No", usedby = "'.$usedBy.'", dateused = "'.$now.'", ip = "'.$currentIP.'" WHERE code = "'.$code.'"');
  3377. if(ENABLEMAIL){
  3378. if (!isset($GLOBALS['USER'])) {
  3379. require_once("user.php");
  3380. $GLOBALS['USER'] = new User('registration_callback');
  3381. }
  3382. sendEmail($GLOBALS['USER']->adminEmail, "Admin", "Plex Invite Used", orgEmail("PLEX Invite Used", "Look who joined the cool club", "Admin", "Hey, The User: $usedBy has redeemd their invite code: [$code], their IP Address was: $currentIP", null, null, "What Next?", "Well, That is up to you. You can go check on them if you like."));
  3383. }
  3384. return (!empty($invites) ? true : false );
  3385. break;
  3386. }
  3387. }
  3388. function plexJoin($username, $email, $password){
  3389. $connectURL = 'https://plex.tv/users.json';
  3390. $headers = array(
  3391. 'Accept'=> 'application/json',
  3392. 'Content-Type' => 'application/x-www-form-urlencoded',
  3393. 'X-Plex-Product' => 'Organizr',
  3394. 'X-Plex-Version' => '1.0',
  3395. 'X-Plex-Client-Identifier' => '01010101-10101010',
  3396. );
  3397. $body = array(
  3398. 'user[email]' => $email,
  3399. 'user[username]' => $username,
  3400. 'user[password]' => $password,
  3401. );
  3402. $api = curl_post($connectURL, $body, $headers);
  3403. $json = json_decode($api['content'], true);
  3404. $errors = (!empty($json['errors']) ? true : false);
  3405. $success = (!empty($json['user']) ? true : false);
  3406. //Use This for later
  3407. $usernameError = (!empty($json['errors']['username']) ? $json['errors']['username'][0] : false);
  3408. $emailError = (!empty($json['errors']['email']) ? $json['errors']['email'][0] : false);
  3409. $passwordError = (!empty($json['errors']['password']) ? $json['errors']['password'][0] : false);
  3410. $errorMessage = "";
  3411. if($errors){
  3412. if($usernameError){ $errorMessage .= "[Username Error: ". $usernameError ."]"; }
  3413. if($emailError){ $errorMessage .= "[Email Error: ". $emailError ."]"; }
  3414. if($passwordError){ $errorMessage .= "[Password Error: ". $passwordError ."]"; }
  3415. }
  3416. switch ($api['http_code']['http_code']){
  3417. case 400:
  3418. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." $username already has access to the shared libraries $errorMessage");
  3419. break;
  3420. case 401:
  3421. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." invalid Plex Token $errorMessage");
  3422. break;
  3423. case 422:
  3424. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." user info error $errorMessage");
  3425. break;
  3426. case 429:
  3427. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." too many requests to plex.tv please try later $errorMessage");
  3428. break;
  3429. case 200:
  3430. case 201:
  3431. writeLog("success", "PLEX JOIN: $username now has access to your Plex Library");
  3432. break;
  3433. default:
  3434. writeLog("error", "PLEX JOIN: unknown error, $errorMessage Error: ".$api['http_code']['http_code']);
  3435. }
  3436. //prettyPrint($api);
  3437. //prettyPrint(json_decode($api['content'], true));
  3438. return (!empty($success) && empty($errors) ? true : false );
  3439. }
  3440. function getCert(){
  3441. $url = "http://curl.haxx.se/ca/cacert.pem";
  3442. $file = getcwd()."/config/cacert.pem";
  3443. $directory = getcwd()."/config/";
  3444. @mkdir($directory, 0770, true);
  3445. if(!file_exists($file)){
  3446. file_put_contents( $file, fopen($url, 'r'));
  3447. writeLog("success", "CERT PEM: pem file created");
  3448. }elseif (file_exists($file) && time() - 2592000 > filemtime($file)) {
  3449. file_put_contents( $file, fopen($url, 'r'));
  3450. writeLog("success", "CERT PEM: downloaded new pem file");
  3451. }
  3452. return $file;
  3453. }
  3454. function customCSS(){
  3455. if(CUSTOMCSS == "true") {
  3456. $template_file = "custom.css";
  3457. $file_handle = fopen($template_file, "rb");
  3458. echo "\n";
  3459. echo fread($file_handle, filesize($template_file));
  3460. fclose($file_handle);
  3461. echo "\n";
  3462. }
  3463. }
  3464. function tvdbToken(){
  3465. $headers = array(
  3466. "Accept" => "application/json",
  3467. "Content-Type" => "application/json"
  3468. );
  3469. $json = array(
  3470. "apikey" => "FBE7B62621F4CAD7",
  3471. "userkey" => "328BB46EB1E9A0F5",
  3472. "username" => "causefx"
  3473. );
  3474. $api = curl_post("https://api.thetvdb.com/login", $json, $headers);
  3475. return json_decode($api['content'], true)['token'];
  3476. }
  3477. function tvdbGet($id){
  3478. $headers = array(
  3479. "Accept" => "application/json",
  3480. "Authorization" => "Bearer ".tvdbToken(),
  3481. "trakt-api-key" => "4502cfdf8f7282fe454878ff8583f5636392cdc5fcac30d0cc4565f7173bf443",
  3482. "trakt-api-version" => "2"
  3483. );
  3484. $trakt = curl_get("https://api.trakt.tv/search/tvdb/$id?type=show", $headers);
  3485. @$api['trakt'] = json_decode($trakt, true)[0]['show']['ids'];
  3486. if(empty($api['trakt'])){
  3487. $series = curl_get("https://api.thetvdb.com/series/$id", $headers);
  3488. $poster = curl_get("https://api.thetvdb.com/series/$id/images/query?keyType=poster", $headers);
  3489. $backdrop = curl_get("https://api.thetvdb.com/series/$id/images/query?keyType=fanart", $headers);
  3490. $api['series'] = json_decode($series, true)['data'];
  3491. $api['poster'] = json_decode($poster, true)['data'];
  3492. $api['backdrop'] = json_decode($backdrop, true)['data'];
  3493. }
  3494. return $api;
  3495. }
  3496. function tvdbSearch($name, $type){
  3497. $name = rawurlencode(preg_replace("/\(([^()]*+|(?R))*\)/","", $name));
  3498. $headers = array(
  3499. "Accept" => "application/json",
  3500. "Authorization" => "Bearer ".tvdbToken(),
  3501. "trakt-api-key" => "4502cfdf8f7282fe454878ff8583f5636392cdc5fcac30d0cc4565f7173bf443",
  3502. "trakt-api-version" => "2"
  3503. );
  3504. $trakt = curl_get("https://api.trakt.tv/search/$type?query=$name", $headers);
  3505. @$api['trakt'] = json_decode($trakt, true)[0][$type]['ids'];
  3506. return $api;
  3507. }
  3508. function getPlexPlaylists(){
  3509. $address = qualifyURL(PLEXURL);
  3510. // Perform API requests
  3511. $api = @curl_get($address."/playlists?X-Plex-Token=".PLEXTOKEN);
  3512. libxml_use_internal_errors(true);
  3513. $api = simplexml_load_string($api);
  3514. if (is_array($api) || is_object($api)){
  3515. if (!$api->head->title){
  3516. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3517. if (!$getServer) { return 'Could not load!'; }
  3518. // Identify the local machine
  3519. $gotServer = $getServer['machineIdentifier'];
  3520. $output = "";
  3521. $hideMenu = '<div class="pull-right"><div class="btn-group" role="group"><button type="button" id="playlist-Name" class="btn waves btn-default btn-sm dropdown-toggle waves-effect waves-float" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">Choose A Playlist &nbsp;<span class="caret"></span></button><ul style="right:0; left: auto; height: 200px;" class="dropdown-menu filter-recent-playlist playlist-listing">';
  3522. foreach($api AS $child) {
  3523. $items = array();
  3524. if ($child['playlistType'] == "video" && strpos(strtolower($child['title']) , 'private') === false){
  3525. $api = @curl_get($address.$child['key']."?X-Plex-Token=".PLEXTOKEN);
  3526. $api = simplexml_load_string($api);
  3527. if (is_array($api) || is_object($api)){
  3528. if (!$api->head->title){
  3529. $className = preg_replace("/(\W)+/", "", $api['title']);
  3530. $hideMenu .= '<li data-filter="playlist-'.$className.'" data-name="'.$api['title'].'"><a class="js-filter-'.$className.'" href="javascript:void(0)">'.$api['title'].'</a></li>';
  3531. foreach($api->Video AS $child){
  3532. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, false, false,false,$className);
  3533. }
  3534. if (count($items)) {
  3535. $output .= ''.implode('',$items).'';
  3536. }
  3537. }
  3538. }
  3539. }
  3540. }
  3541. $hideMenu .= '</ul></div></div>';
  3542. return '<div id="playlist-all" class="content-box box-shadow big-box"><h5 id="playlist-title" style="margin-bottom: -20px" class="text-center">All Playlists</h5><div class="recentHeader inbox-pagination all">'.$hideMenu.'</div><br/><br/><div class="recentItems-playlists" data-name="all">'.$output.'</div></div>';
  3543. }else{
  3544. writeLog("error", "PLEX PLAYLIST ERROR: could not connect - check token - if HTTPS, is cert valid");
  3545. }
  3546. }else{
  3547. writeLog("error", "PLEX PLAYLIST ERROR: could not connect - check URL - if HTTPS, is cert valid");
  3548. }
  3549. }
  3550. function readExternalLog($type,$filename,$name = null){
  3551. $log = file($filename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
  3552. $log = array_reverse($log);
  3553. foreach($log as $line){
  3554. if(!empty($line) && $line[0] != " "){
  3555. $line = strip($line);
  3556. if($type == "single"){
  3557. if( strpos( strtolower($line), "ror" ) !== false ) {
  3558. echo "<tr><td class='red-bg'>".$line."</td></tr>";
  3559. }else{
  3560. echo "<tr><td>".$line."</td></tr>";
  3561. }
  3562. }elseif($type == "all"){
  3563. if( strpos( strtolower($line), "ror" ) !== false ) {
  3564. echo "<tr><td class='red-bg'>".$name."</td>";
  3565. echo "<td class='red-bg'>".$line."</td></tr>";
  3566. }else{
  3567. echo "<tr><td>".$name."</td>";
  3568. echo "<td>".$line."</td></tr>";
  3569. }
  3570. }
  3571. }
  3572. }
  3573. }
  3574. function getLogs(){
  3575. $path = __DIR__ ."/logs/";
  3576. @mkdir($path, 0770, true);
  3577. $logs = array();
  3578. $files = array_diff(scandir($path), array('.', '..'));
  3579. foreach($files as $v){
  3580. $title = explode(".", $v)[0];
  3581. $logs[$title] = $path.$v;
  3582. }
  3583. return $logs;
  3584. }
  3585. function getBackups(){
  3586. $path = DATABASE_LOCATION ."backups/";
  3587. @mkdir($path, 0770, true);
  3588. $backups = array();
  3589. $files = array_diff(scandir($path), array('.', '..'));
  3590. return array_reverse($files);
  3591. }
  3592. function getExtension($string) {
  3593. return preg_replace("#(.+)?\.(\w+)(\?.+)?#", "$2", $string);
  3594. }
  3595. function showFile(){
  3596. $file = $_GET['file'];
  3597. $fileType = getExtension($file);
  3598. if($fileType != 'php'){
  3599. header("Content-type: ".mimeTypes()[$fileType]);
  3600. @readfile($file);
  3601. }
  3602. }
  3603. function getCalendar(){
  3604. $sonarr = new Sonarr(SONARRURL, SONARRKEY);
  3605. $radarr = new Sonarr(RADARRURL, RADARRKEY);
  3606. $sickrage = new SickRage(SICKRAGEURL, SICKRAGEKEY);
  3607. $startDate = date('Y-m-d',strtotime("-".CALENDARSTARTDAY." days"));
  3608. $endDate = date('Y-m-d',strtotime("+".CALENDARENDDAY." days"));
  3609. $calendarItems = array();
  3610. if (SONARRURL != "" && qualifyUser(SONARRHOMEAUTH)){
  3611. try {
  3612. $sonarrCalendar = getSonarrCalendar($sonarr->getCalendar($startDate, $endDate));
  3613. if(!empty($sonarrCalendar)) { $calendarItems = array_merge($calendarItems, $sonarrCalendar); }
  3614. } catch (Exception $e) {
  3615. writeLog("error", "SONARR ERROR: ".strip($e->getMessage()));
  3616. }
  3617. }
  3618. if (RADARRURL != "" && qualifyUser(RADARRHOMEAUTH)){
  3619. try {
  3620. $radarrCalendar = getRadarrCalendar($radarr->getCalendar($startDate, $endDate));
  3621. if(!empty($radarrCalendar)) { $calendarItems = array_merge($calendarItems, $radarrCalendar); }
  3622. } catch (Exception $e) {
  3623. writeLog("error", "RADARR ERROR: ".strip($e->getMessage()));
  3624. }
  3625. }
  3626. if (COUCHURL != "" && qualifyUser(COUCHHOMEAUTH)){
  3627. $couchCalendar = getCouchCalendar();
  3628. if(!empty($couchCalendar)) { $calendarItems = array_merge($calendarItems, $couchCalendar); }
  3629. }
  3630. if (HEADPHONESURL != "" && qualifyUser(HEADPHONESHOMEAUTH)){
  3631. $headphonesHistory = getHeadphonesCalendar(HEADPHONESURL, HEADPHONESKEY, "getHistory");
  3632. $headphonesWanted = getHeadphonesCalendar(HEADPHONESURL, HEADPHONESKEY, "getWanted");
  3633. if(!empty($headphonesHistory)) { $calendarItems = array_merge($calendarItems, $headphonesHistory); }
  3634. if(!empty($headphonesWanted)) { $calendarItems = array_merge($calendarItems, $headphonesWanted); }
  3635. }
  3636. if (SICKRAGEURL != "" && qualifyUser(SICKRAGEHOMEAUTH)){
  3637. try {
  3638. $sickrageFuture = getSickrageCalendarWanted($sickrage->future());
  3639. if(!empty($sickrageFuture)) { $calendarItems = array_merge($calendarItems, $sickrageFuture); }
  3640. } catch (Exception $e) {
  3641. writeLog("error", "SICKRAGE/BEARD ERROR: ".strip($e->getMessage()));
  3642. } try {
  3643. $sickrageHistory = getSickrageCalendarHistory($sickrage->history("100","downloaded"));
  3644. if(!empty($sickrageHistory)) { $calendarItems = array_merge($calendarItems, $sickrageHistory); }
  3645. } catch (Exception $e) {
  3646. writeLog("error", "SICKRAGE/BEARD ERROR: ".strip($e->getMessage()));
  3647. }
  3648. }
  3649. return $calendarItems;
  3650. }
  3651. function localURL($url){
  3652. if (strpos($url, 'https') !== false) {
  3653. preg_match("/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/", $url, $result);
  3654. $result = (!empty($result) ? true : false);
  3655. return $result;
  3656. }
  3657. }
  3658. function fileArray($files){
  3659. foreach($files as $file){
  3660. if(file_exists($file)){
  3661. $list[] = $file;
  3662. }
  3663. }
  3664. if(!empty($list)){ return $list; }
  3665. }
  3666. function backupDB(){
  3667. if (extension_loaded('ZIP')) {
  3668. $directory = DATABASE_LOCATION."backups/";
  3669. @mkdir($directory, 0770, true);
  3670. $orgFiles = array(
  3671. 'css' => 'custom.css',
  3672. 'temp' => 'cus.sd',
  3673. 'orgLog' => 'org.log',
  3674. 'loginLog' => 'loginLog.json',
  3675. 'chatDB' => 'chatpack.db',
  3676. 'config' => 'config/config.php',
  3677. 'database' => DATABASE_LOCATION.'users.db'
  3678. );
  3679. $files = fileArray($orgFiles);
  3680. if(!empty($files)){
  3681. writeLog("success", "BACKUP: backup process started");
  3682. $zipname = $directory.'backup['.date('Y-m-d_H-i').']['.INSTALLEDVERSION.'].zip';
  3683. $zip = new ZipArchive;
  3684. $zip->open($zipname, ZipArchive::CREATE);
  3685. foreach ($files as $file) {
  3686. $zip->addFile($file);
  3687. }
  3688. $zip->close();
  3689. writeLog("success", "BACKUP: backup process finished");
  3690. return true;
  3691. }else{
  3692. return false;
  3693. }
  3694. }else{
  3695. return false;
  3696. }
  3697. }
  3698. class Ping {
  3699. private $host;
  3700. private $ttl;
  3701. private $timeout;
  3702. private $port = 80;
  3703. private $data = 'Ping';
  3704. private $commandOutput;
  3705. /**
  3706. * Called when the Ping object is created.
  3707. *
  3708. * @param string $host
  3709. * The host to be pinged.
  3710. * @param int $ttl
  3711. * Time-to-live (TTL) (You may get a 'Time to live exceeded' error if this
  3712. * value is set too low. The TTL value indicates the scope or range in which
  3713. * a packet may be forwarded. By convention:
  3714. * - 0 = same host
  3715. * - 1 = same subnet
  3716. * - 32 = same site
  3717. * - 64 = same region
  3718. * - 128 = same continent
  3719. * - 255 = unrestricted
  3720. * @param int $timeout
  3721. * Timeout (in seconds) used for ping and fsockopen().
  3722. * @throws \Exception if the host is not set.
  3723. */
  3724. public function __construct($host, $ttl = 255, $timeout = 10) {
  3725. if (!isset($host)) {
  3726. throw new \Exception("Error: Host name not supplied.");
  3727. }
  3728. $this->host = $host;
  3729. $this->ttl = $ttl;
  3730. $this->timeout = $timeout;
  3731. }
  3732. /**
  3733. * Set the ttl (in hops).
  3734. *
  3735. * @param int $ttl
  3736. * TTL in hops.
  3737. */
  3738. public function setTtl($ttl) {
  3739. $this->ttl = $ttl;
  3740. }
  3741. /**
  3742. * Get the ttl.
  3743. *
  3744. * @return int
  3745. * The current ttl for Ping.
  3746. */
  3747. public function getTtl() {
  3748. return $this->ttl;
  3749. }
  3750. /**
  3751. * Set the timeout.
  3752. *
  3753. * @param int $timeout
  3754. * Time to wait in seconds.
  3755. */
  3756. public function setTimeout($timeout) {
  3757. $this->timeout = $timeout;
  3758. }
  3759. /**
  3760. * Get the timeout.
  3761. *
  3762. * @return int
  3763. * Current timeout for Ping.
  3764. */
  3765. public function getTimeout() {
  3766. return $this->timeout;
  3767. }
  3768. /**
  3769. * Set the host.
  3770. *
  3771. * @param string $host
  3772. * Host name or IP address.
  3773. */
  3774. public function setHost($host) {
  3775. $this->host = $host;
  3776. }
  3777. /**
  3778. * Get the host.
  3779. *
  3780. * @return string
  3781. * The current hostname for Ping.
  3782. */
  3783. public function getHost() {
  3784. return $this->host;
  3785. }
  3786. /**
  3787. * Set the port (only used for fsockopen method).
  3788. *
  3789. * Since regular pings use ICMP and don't need to worry about the concept of
  3790. * 'ports', this is only used for the fsockopen method, which pings servers by
  3791. * checking port 80 (by default).
  3792. *
  3793. * @param int $port
  3794. * Port to use for fsockopen ping (defaults to 80 if not set).
  3795. */
  3796. public function setPort($port) {
  3797. $this->port = $port;
  3798. }
  3799. /**
  3800. * Get the port (only used for fsockopen method).
  3801. *
  3802. * @return int
  3803. * The port used by fsockopen pings.
  3804. */
  3805. public function getPort() {
  3806. return $this->port;
  3807. }
  3808. /**
  3809. * Return the command output when method=exec.
  3810. * @return string
  3811. */
  3812. public function getCommandOutput(){
  3813. return $this->commandOutput;
  3814. }
  3815. /**
  3816. * Matches an IP on command output and returns.
  3817. * @return string
  3818. */
  3819. public function getIpAddress() {
  3820. $out = array();
  3821. if (preg_match('/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/', $this->commandOutput, $out)){
  3822. return $out[0];
  3823. }
  3824. return null;
  3825. }
  3826. /**
  3827. * Ping a host.
  3828. *
  3829. * @param string $method
  3830. * Method to use when pinging:
  3831. * - exec (default): Pings through the system ping command. Fast and
  3832. * robust, but a security risk if you pass through user-submitted data.
  3833. * - fsockopen: Pings a server on port 80.
  3834. * - socket: Creates a RAW network socket. Only usable in some
  3835. * environments, as creating a SOCK_RAW socket requires root privileges.
  3836. *
  3837. * @throws InvalidArgumentException if $method is not supported.
  3838. *
  3839. * @return mixed
  3840. * Latency as integer, in ms, if host is reachable or FALSE if host is down.
  3841. */
  3842. public function ping($method = 'exec') {
  3843. $latency = false;
  3844. switch ($method) {
  3845. case 'exec':
  3846. $latency = $this->pingExec();
  3847. break;
  3848. case 'fsockopen':
  3849. $latency = $this->pingFsockopen();
  3850. break;
  3851. case 'socket':
  3852. $latency = $this->pingSocket();
  3853. break;
  3854. default:
  3855. throw new \InvalidArgumentException('Unsupported ping method.');
  3856. }
  3857. // Return the latency.
  3858. return $latency;
  3859. }
  3860. /**
  3861. * The exec method uses the possibly insecure exec() function, which passes
  3862. * the input to the system. This is potentially VERY dangerous if you pass in
  3863. * any user-submitted data. Be SURE you sanitize your inputs!
  3864. *
  3865. * @return int
  3866. * Latency, in ms.
  3867. */
  3868. private function pingExec() {
  3869. $latency = false;
  3870. $ttl = escapeshellcmd($this->ttl);
  3871. $timeout = escapeshellcmd($this->timeout);
  3872. $host = escapeshellcmd($this->host);
  3873. // Exec string for Windows-based systems.
  3874. if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
  3875. // -n = number of pings; -i = ttl; -w = timeout (in milliseconds).
  3876. $exec_string = 'ping -n 1 -i ' . $ttl . ' -w ' . ($timeout * 1000) . ' ' . $host;
  3877. }
  3878. // Exec string for Darwin based systems (OS X).
  3879. else if(strtoupper(PHP_OS) === 'DARWIN') {
  3880. // -n = numeric output; -c = number of pings; -m = ttl; -t = timeout.
  3881. $exec_string = 'ping -n -c 1 -m ' . $ttl . ' -t ' . $timeout . ' ' . $host;
  3882. }
  3883. // Exec string for other UNIX-based systems (Linux).
  3884. else {
  3885. // -n = numeric output; -c = number of pings; -t = ttl; -W = timeout
  3886. $exec_string = 'ping -n -c 1 -t ' . $ttl . ' -W ' . $timeout . ' ' . $host . ' 2>&1';
  3887. }
  3888. exec($exec_string, $output, $return);
  3889. // Strip empty lines and reorder the indexes from 0 (to make results more
  3890. // uniform across OS versions).
  3891. $this->commandOutput = implode($output, '');
  3892. $output = array_values(array_filter($output));
  3893. // If the result line in the output is not empty, parse it.
  3894. if (!empty($output[1])) {
  3895. // Search for a 'time' value in the result line.
  3896. $response = preg_match("/time(?:=|<)(?<time>[\.0-9]+)(?:|\s)ms/", $output[1], $matches);
  3897. // If there's a result and it's greater than 0, return the latency.
  3898. if ($response > 0 && isset($matches['time'])) {
  3899. $latency = round($matches['time'], 2);
  3900. }
  3901. }
  3902. return $latency;
  3903. }
  3904. /**
  3905. * The fsockopen method simply tries to reach the host on a port. This method
  3906. * is often the fastest, but not necessarily the most reliable. Even if a host
  3907. * doesn't respond, fsockopen may still make a connection.
  3908. *
  3909. * @return int
  3910. * Latency, in ms.
  3911. */
  3912. private function pingFsockopen() {
  3913. $start = microtime(true);
  3914. // fsockopen prints a bunch of errors if a host is unreachable. Hide those
  3915. // irrelevant errors and deal with the results instead.
  3916. $fp = @fsockopen($this->host, $this->port, $errno, $errstr, $this->timeout);
  3917. if (!$fp) {
  3918. $latency = false;
  3919. }
  3920. else {
  3921. $latency = microtime(true) - $start;
  3922. $latency = round($latency * 1000, 2);
  3923. }
  3924. return $latency;
  3925. }
  3926. /**
  3927. * The socket method uses raw network packet data to try sending an ICMP ping
  3928. * packet to a server, then measures the response time. Using this method
  3929. * requires the script to be run with root privileges, though, so this method
  3930. * only works reliably on Windows systems and on Linux servers where the
  3931. * script is not being run as a web user.
  3932. *
  3933. * @return int
  3934. * Latency, in ms.
  3935. */
  3936. private function pingSocket() {
  3937. // Create a package.
  3938. $type = "\x08";
  3939. $code = "\x00";
  3940. $checksum = "\x00\x00";
  3941. $identifier = "\x00\x00";
  3942. $seq_number = "\x00\x00";
  3943. $package = $type . $code . $checksum . $identifier . $seq_number . $this->data;
  3944. // Calculate the checksum.
  3945. $checksum = $this->calculateChecksum($package);
  3946. // Finalize the package.
  3947. $package = $type . $code . $checksum . $identifier . $seq_number . $this->data;
  3948. // Create a socket, connect to server, then read socket and calculate.
  3949. if ($socket = socket_create(AF_INET, SOCK_RAW, 1)) {
  3950. socket_set_option($socket, SOL_SOCKET, SO_RCVTIMEO, array(
  3951. 'sec' => 10,
  3952. 'usec' => 0,
  3953. ));
  3954. // Prevent errors from being printed when host is unreachable.
  3955. @socket_connect($socket, $this->host, null);
  3956. $start = microtime(true);
  3957. // Send the package.
  3958. @socket_send($socket, $package, strlen($package), 0);
  3959. if (socket_read($socket, 255) !== false) {
  3960. $latency = microtime(true) - $start;
  3961. $latency = round($latency * 1000, 2);
  3962. }
  3963. else {
  3964. $latency = false;
  3965. }
  3966. }
  3967. else {
  3968. $latency = false;
  3969. }
  3970. // Close the socket.
  3971. socket_close($socket);
  3972. return $latency;
  3973. }
  3974. /**
  3975. * Calculate a checksum.
  3976. *
  3977. * @param string $data
  3978. * Data for which checksum will be calculated.
  3979. *
  3980. * @return string
  3981. * Binary string checksum of $data.
  3982. */
  3983. private function calculateChecksum($data) {
  3984. if (strlen($data) % 2) {
  3985. $data .= "\x00";
  3986. }
  3987. $bit = unpack('n*', $data);
  3988. $sum = array_sum($bit);
  3989. while ($sum >> 16) {
  3990. $sum = ($sum >> 16) + ($sum & 0xffff);
  3991. }
  3992. return pack('n*', ~$sum);
  3993. }
  3994. }
  3995. function ping($pings, $type = "string") {
  3996. $ping = new Ping("");
  3997. $ping->setTtl(128);
  3998. $ping->setTimeout(2);
  3999. switch ($type){
  4000. case "array":
  4001. $results = [];
  4002. foreach ($pings as $k => $v) {
  4003. if(strpos($v, ':') !== false){
  4004. $domain = explode(':', $v)[0];
  4005. $port = explode(':', $v)[1];
  4006. $ping->setHost($domain);
  4007. $ping->setPort($port);
  4008. $latency = $ping->ping('fsockopen');
  4009. }else{
  4010. $ping->setHost($v);
  4011. $latency = $ping->ping();
  4012. }
  4013. if ($latency || $latency === 0) {
  4014. $results[$k] = $latency;
  4015. } else {
  4016. $results[$k] = 0;
  4017. }
  4018. }
  4019. break;
  4020. case "string":
  4021. if(strpos($pings, ':') !== false){
  4022. $domain = explode(':', $pings)[0];
  4023. $port = explode(':', $pings)[1];
  4024. $ping->setHost($domain);
  4025. $ping->setPort($port);
  4026. $latency = $ping->ping('fsockopen');
  4027. }else{
  4028. $ping->setHost($pings);
  4029. $latency = $ping->ping();
  4030. }
  4031. if ($latency || $latency === 0) {
  4032. $results = $latency;
  4033. } else {
  4034. $results = 0;
  4035. }
  4036. break;
  4037. }
  4038. return $results;
  4039. }
  4040. function getPing($url, $style, $refresh = null){
  4041. if(ping($url) !== 0){
  4042. $class = 'success';
  4043. if(!$refresh){
  4044. $class .= " animated slideInLeft";
  4045. }
  4046. }else{
  4047. $class = "warning";
  4048. if(!$refresh){
  4049. $class .= " animated flash loop-animation-timeout";
  4050. }
  4051. }
  4052. echo '<span class="badge ping-'.$class.'" style="position: absolute;z-index: 100;right: 5px; padding: 0px 0px;'.$style.';font-size: 10px;">&nbsp;</span>';
  4053. }
  4054. function speedTestData(){
  4055. $file_db = DATABASE_LOCATION."speedtest.db";
  4056. if(file_exists($file_db)){
  4057. $conn = new PDO("sqlite:$file_db") or die("1");
  4058. $result = $conn->query('SELECT * FROM speedtest_users');
  4059. $conn = null;
  4060. if (is_array($result) || is_object($result)){
  4061. foreach($result as $k => $v){
  4062. $return[$k] = $v;
  4063. }
  4064. return $return;
  4065. }
  4066. }
  4067. }
  4068. function speedTestDisplay($array, $output){
  4069. if (is_array($array) || is_object($array)){
  4070. if($output == "graph"){
  4071. $result = "Morris.Line({element: 'morris-line',data: [";
  4072. foreach($array as $k => $v){
  4073. $result .= "{ y: '".substr($v['timestamp'],0,10)."', a: ".$v['ul'].", b: ".$v['dl'].", c: ".$v['ping']." },";
  4074. }
  4075. $result .= "],xkey: 'y',ykeys: ['a', 'b', 'c'],labels: ['Upload', 'Download', 'Ping'],hideHover: 'auto',resize: true,lineColors: ['#63A8EB','#ccc','#000'] });";
  4076. }elseif($output == "table"){
  4077. $result = "";
  4078. foreach($array as $k => $v){
  4079. $result .= "<tr><td>".$v['timestamp']."</td><td>".$v['ip']."</td><td>".$v['dl']."</td><td>".$v['ul']."</td><td>".$v['ping']."</td><td>".$v['jitter']."</td></tr>";
  4080. }
  4081. }
  4082. return $result;
  4083. }
  4084. }
  4085. function buildMenuPhone($array){
  4086. if (is_array($array) || is_object($array)){
  4087. $result = '
  4088. <div class="content-box profile-sidebar box-shadow">
  4089. <img src="images/organizr-logo-h-d.png" width="100%" style="margin-top: -10px;">
  4090. <div class="profile-usermenu">
  4091. <ul class="nav" id="settings-list">
  4092. ';
  4093. foreach($array as $k => $v){
  4094. if($v['id'] == 'open-invites' && empty(PLEXURL)){
  4095. continue;
  4096. }
  4097. if($v['id'] == 'open-email' && ENABLEMAIL !== "true"){
  4098. continue;
  4099. }
  4100. /*$result .= '
  4101. <li>
  4102. <a id="'.$v['id'].'" box="'.$v['box'].'">'.$v['name'].'
  4103. <span class="fa-stack fa-fw pull-right" style="margin-top: -5px;margin-right: -10px;">
  4104. <i class="fa fa-'.$v['icon_1'].' fa-stack-2x '.$v['color'].'" style="font-size:null;"></i>
  4105. <i class="fa fa-'.$v['icon_2'].' fa-stack-1x fa-inverse"></i>
  4106. </span>
  4107. </a>
  4108. </li>
  4109. ';*/
  4110. $result .= '<li><a id="'.$v['id'].'" box="'.$v['box'].'"><i class="fa fa-'.$v['icon_2'].' '.$v['color'].' fa-fw pull-right"></i>'.$v['name'].'</a></li>';
  4111. }
  4112. $result .= '</ul></div></div>';
  4113. return $result;
  4114. }
  4115. }
  4116. function buildMenu($array){
  4117. if (is_array($array) || is_object($array)){
  4118. $result = '<div class="settingsList">';
  4119. foreach($array as $k => $v){
  4120. if($v['id'] == 'open-invites' && empty(PLEXURL)){
  4121. continue;
  4122. }
  4123. if($v['id'] == 'open-email' && ENABLEMAIL !== "true"){
  4124. continue;
  4125. }
  4126. $result .= '
  4127. <button id="'.$v['id'].'" box="'.$v['box'].'" type="button" style="border-radius: 0px !important; -webkit-border-radius: 0px !important;margin-bottom: 3px;margin-left:5px;color:white;" class="btn '.$v['color2'].' btn-icon waves waves-circle waves-effect waves-float settingsMenu">
  4128. <i class="mdi mdi-'.$v['icon_1'].' fa-fw pull-left" style="padding-left: '.$v['padding'].'px;font-size: 30px"></i>
  4129. <p class="" style="text-align: center;direction: rtl;display:none;margin: 2px;"><strong>'.$v['name'].'</strong></p>
  4130. </button>
  4131. ';
  4132. }
  4133. $result .= '</div>';
  4134. return $result;
  4135. }
  4136. }
  4137. function requestInvite($email, $username){
  4138. //sendEmail($email, $username = "Organizr User", $subject, $body, $cc = null){
  4139. //orgEmail($header = "Message From Admin", $title = "Important Message", $user = "Organizr User", $mainMessage = "", $button = null, $buttonURL = null, $subTitle = "", $subMessage = ""){
  4140. sendEmail($GLOBALS['USER']->adminEmail, "Admin", "Plex Invite Request", orgEmail("PLEX Invite Request", "Look who wants to join the cool club", "Admin", "Hey, The User: $user has requested access to your Plex Library.", "Generate Invite", null, "What Next?", "Well, That is up to you. You can go check on them if you like."));
  4141. }
  4142. function errormessage($msg) {
  4143. echo "<div style=\"margin-top: 50px;\">";
  4144. echo "<span style=\"color:#d89334;\">error </span>";
  4145. echo $msg;
  4146. echo "</div>";
  4147. }
  4148. function getOrgUsers(){
  4149. $file_db = DATABASE_LOCATION."users.db";
  4150. if(file_exists($file_db)){
  4151. $conn = new PDO("sqlite:$file_db") or die("1");
  4152. $result = $conn->query('SELECT * FROM users');
  4153. $conn = null;
  4154. if (is_array($result) || is_object($result)){
  4155. foreach($result as $k => $v){
  4156. $return[$v['username']] = $v['email'];
  4157. }
  4158. return $return;
  4159. }
  4160. }
  4161. }
  4162. function getEmails($type = 'org'){
  4163. if($type == 'plex'){
  4164. $emails = array_merge(libraryList()['both'],getOrgUsers());
  4165. }elseif($type == 'emby'){
  4166. $emails = getOrgUsers();
  4167. }else{
  4168. $emails = getOrgUsers();
  4169. }
  4170. return $emails;
  4171. }
  4172. function printEmails($emails){
  4173. $result = '';
  4174. foreach($emails as $k => $v){
  4175. $result .= '<option value="'.$v.'">'.$k.'</option>';
  4176. }
  4177. return $result;
  4178. }
  4179. function massEmail($to, $subject, $message){
  4180. if (!isset($GLOBALS['file_db'])) {
  4181. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  4182. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  4183. }
  4184. sendEmail(null, null, $subject, orgEmail("Message From Admin", "Important Information", "There", $message, null, null, "Thank You!", "Thanks for taking the time to read this message from me."),$GLOBALS['USER']->adminEmail,$to);
  4185. }
  4186. function q2a($q){
  4187. if (is_array($q) || is_object($q)){
  4188. foreach ($q as $k => $v){
  4189. $a[$k] = $v;
  4190. }
  4191. if(!empty($a)){
  4192. return $a;
  4193. }
  4194. }
  4195. }
  4196. class Mobile_Detect
  4197. {
  4198. /**
  4199. * Mobile detection type.
  4200. *
  4201. * @deprecated since version 2.6.9
  4202. */
  4203. const DETECTION_TYPE_MOBILE = 'mobile';
  4204. /**
  4205. * Extended detection type.
  4206. *
  4207. * @deprecated since version 2.6.9
  4208. */
  4209. const DETECTION_TYPE_EXTENDED = 'extended';
  4210. /**
  4211. * A frequently used regular expression to extract version #s.
  4212. *
  4213. * @deprecated since version 2.6.9
  4214. */
  4215. const VER = '([\w._\+]+)';
  4216. /**
  4217. * Top-level device.
  4218. */
  4219. const MOBILE_GRADE_A = 'A';
  4220. /**
  4221. * Mid-level device.
  4222. */
  4223. const MOBILE_GRADE_B = 'B';
  4224. /**
  4225. * Low-level device.
  4226. */
  4227. const MOBILE_GRADE_C = 'C';
  4228. /**
  4229. * Stores the version number of the current release.
  4230. */
  4231. const VERSION = '2.8.26';
  4232. /**
  4233. * A type for the version() method indicating a string return value.
  4234. */
  4235. const VERSION_TYPE_STRING = 'text';
  4236. /**
  4237. * A type for the version() method indicating a float return value.
  4238. */
  4239. const VERSION_TYPE_FLOAT = 'float';
  4240. /**
  4241. * A cache for resolved matches
  4242. * @var array
  4243. */
  4244. protected $cache = array();
  4245. /**
  4246. * The User-Agent HTTP header is stored in here.
  4247. * @var string
  4248. */
  4249. protected $userAgent = null;
  4250. /**
  4251. * HTTP headers in the PHP-flavor. So HTTP_USER_AGENT and SERVER_SOFTWARE.
  4252. * @var array
  4253. */
  4254. protected $httpHeaders = array();
  4255. /**
  4256. * CloudFront headers. E.g. CloudFront-Is-Desktop-Viewer, CloudFront-Is-Mobile-Viewer & CloudFront-Is-Tablet-Viewer.
  4257. * @var array
  4258. */
  4259. protected $cloudfrontHeaders = array();
  4260. /**
  4261. * The matching Regex.
  4262. * This is good for debug.
  4263. * @var string
  4264. */
  4265. protected $matchingRegex = null;
  4266. /**
  4267. * The matches extracted from the regex expression.
  4268. * This is good for debug.
  4269. * @var string
  4270. */
  4271. protected $matchesArray = null;
  4272. /**
  4273. * The detection type, using self::DETECTION_TYPE_MOBILE or self::DETECTION_TYPE_EXTENDED.
  4274. *
  4275. * @deprecated since version 2.6.9
  4276. *
  4277. * @var string
  4278. */
  4279. protected $detectionType = self::DETECTION_TYPE_MOBILE;
  4280. /**
  4281. * HTTP headers that trigger the 'isMobile' detection
  4282. * to be true.
  4283. *
  4284. * @var array
  4285. */
  4286. protected static $mobileHeaders = array(
  4287. 'HTTP_ACCEPT' => array('matches' => array(
  4288. // Opera Mini; @reference: http://dev.opera.com/articles/view/opera-binary-markup-language/
  4289. 'application/x-obml2d',
  4290. // BlackBerry devices.
  4291. 'application/vnd.rim.html',
  4292. 'text/vnd.wap.wml',
  4293. 'application/vnd.wap.xhtml+xml'
  4294. )),
  4295. 'HTTP_X_WAP_PROFILE' => null,
  4296. 'HTTP_X_WAP_CLIENTID' => null,
  4297. 'HTTP_WAP_CONNECTION' => null,
  4298. 'HTTP_PROFILE' => null,
  4299. // Reported by Opera on Nokia devices (eg. C3).
  4300. 'HTTP_X_OPERAMINI_PHONE_UA' => null,
  4301. 'HTTP_X_NOKIA_GATEWAY_ID' => null,
  4302. 'HTTP_X_ORANGE_ID' => null,
  4303. 'HTTP_X_VODAFONE_3GPDPCONTEXT' => null,
  4304. 'HTTP_X_HUAWEI_USERID' => null,
  4305. // Reported by Windows Smartphones.
  4306. 'HTTP_UA_OS' => null,
  4307. // Reported by Verizon, Vodafone proxy system.
  4308. 'HTTP_X_MOBILE_GATEWAY' => null,
  4309. // Seen this on HTC Sensation. SensationXE_Beats_Z715e.
  4310. 'HTTP_X_ATT_DEVICEID' => null,
  4311. // Seen this on a HTC.
  4312. 'HTTP_UA_CPU' => array('matches' => array('ARM')),
  4313. );
  4314. /**
  4315. * List of mobile devices (phones).
  4316. *
  4317. * @var array
  4318. */
  4319. protected static $phoneDevices = array(
  4320. 'iPhone' => '\biPhone\b|\biPod\b', // |\biTunes
  4321. 'BlackBerry' => 'BlackBerry|\bBB10\b|rim[0-9]+',
  4322. 'HTC' => 'HTC|HTC.*(Sensation|Evo|Vision|Explorer|6800|8100|8900|A7272|S510e|C110e|Legend|Desire|T8282)|APX515CKT|Qtek9090|APA9292KT|HD_mini|Sensation.*Z710e|PG86100|Z715e|Desire.*(A8181|HD)|ADR6200|ADR6400L|ADR6425|001HT|Inspire 4G|Android.*\bEVO\b|T-Mobile G1|Z520m',
  4323. 'Nexus' => 'Nexus One|Nexus S|Galaxy.*Nexus|Android.*Nexus.*Mobile|Nexus 4|Nexus 5|Nexus 6',
  4324. // @todo: Is 'Dell Streak' a tablet or a phone? ;)
  4325. 'Dell' => 'Dell.*Streak|Dell.*Aero|Dell.*Venue|DELL.*Venue Pro|Dell Flash|Dell Smoke|Dell Mini 3iX|XCD28|XCD35|\b001DL\b|\b101DL\b|\bGS01\b',
  4326. 'Motorola' => 'Motorola|DROIDX|DROID BIONIC|\bDroid\b.*Build|Android.*Xoom|HRI39|MOT-|A1260|A1680|A555|A853|A855|A953|A955|A956|Motorola.*ELECTRIFY|Motorola.*i1|i867|i940|MB200|MB300|MB501|MB502|MB508|MB511|MB520|MB525|MB526|MB611|MB612|MB632|MB810|MB855|MB860|MB861|MB865|MB870|ME501|ME502|ME511|ME525|ME600|ME632|ME722|ME811|ME860|ME863|ME865|MT620|MT710|MT716|MT720|MT810|MT870|MT917|Motorola.*TITANIUM|WX435|WX445|XT300|XT301|XT311|XT316|XT317|XT319|XT320|XT390|XT502|XT530|XT531|XT532|XT535|XT603|XT610|XT611|XT615|XT681|XT701|XT702|XT711|XT720|XT800|XT806|XT860|XT862|XT875|XT882|XT883|XT894|XT901|XT907|XT909|XT910|XT912|XT928|XT926|XT915|XT919|XT925|XT1021|\bMoto E\b',
  4327. 'Samsung' => '\bSamsung\b|SM-G9250|GT-19300|SGH-I337|BGT-S5230|GT-B2100|GT-B2700|GT-B2710|GT-B3210|GT-B3310|GT-B3410|GT-B3730|GT-B3740|GT-B5510|GT-B5512|GT-B5722|GT-B6520|GT-B7300|GT-B7320|GT-B7330|GT-B7350|GT-B7510|GT-B7722|GT-B7800|GT-C3010|GT-C3011|GT-C3060|GT-C3200|GT-C3212|GT-C3212I|GT-C3262|GT-C3222|GT-C3300|GT-C3300K|GT-C3303|GT-C3303K|GT-C3310|GT-C3322|GT-C3330|GT-C3350|GT-C3500|GT-C3510|GT-C3530|GT-C3630|GT-C3780|GT-C5010|GT-C5212|GT-C6620|GT-C6625|GT-C6712|GT-E1050|GT-E1070|GT-E1075|GT-E1080|GT-E1081|GT-E1085|GT-E1087|GT-E1100|GT-E1107|GT-E1110|GT-E1120|GT-E1125|GT-E1130|GT-E1160|GT-E1170|GT-E1175|GT-E1180|GT-E1182|GT-E1200|GT-E1210|GT-E1225|GT-E1230|GT-E1390|GT-E2100|GT-E2120|GT-E2121|GT-E2152|GT-E2220|GT-E2222|GT-E2230|GT-E2232|GT-E2250|GT-E2370|GT-E2550|GT-E2652|GT-E3210|GT-E3213|GT-I5500|GT-I5503|GT-I5700|GT-I5800|GT-I5801|GT-I6410|GT-I6420|GT-I7110|GT-I7410|GT-I7500|GT-I8000|GT-I8150|GT-I8160|GT-I8190|GT-I8320|GT-I8330|GT-I8350|GT-I8530|GT-I8700|GT-I8703|GT-I8910|GT-I9000|GT-I9001|GT-I9003|GT-I9010|GT-I9020|GT-I9023|GT-I9070|GT-I9082|GT-I9100|GT-I9103|GT-I9220|GT-I9250|GT-I9300|GT-I9305|GT-I9500|GT-I9505|GT-M3510|GT-M5650|GT-M7500|GT-M7600|GT-M7603|GT-M8800|GT-M8910|GT-N7000|GT-S3110|GT-S3310|GT-S3350|GT-S3353|GT-S3370|GT-S3650|GT-S3653|GT-S3770|GT-S3850|GT-S5210|GT-S5220|GT-S5229|GT-S5230|GT-S5233|GT-S5250|GT-S5253|GT-S5260|GT-S5263|GT-S5270|GT-S5300|GT-S5330|GT-S5350|GT-S5360|GT-S5363|GT-S5369|GT-S5380|GT-S5380D|GT-S5560|GT-S5570|GT-S5600|GT-S5603|GT-S5610|GT-S5620|GT-S5660|GT-S5670|GT-S5690|GT-S5750|GT-S5780|GT-S5830|GT-S5839|GT-S6102|GT-S6500|GT-S7070|GT-S7200|GT-S7220|GT-S7230|GT-S7233|GT-S7250|GT-S7500|GT-S7530|GT-S7550|GT-S7562|GT-S7710|GT-S8000|GT-S8003|GT-S8500|GT-S8530|GT-S8600|SCH-A310|SCH-A530|SCH-A570|SCH-A610|SCH-A630|SCH-A650|SCH-A790|SCH-A795|SCH-A850|SCH-A870|SCH-A890|SCH-A930|SCH-A950|SCH-A970|SCH-A990|SCH-I100|SCH-I110|SCH-I400|SCH-I405|SCH-I500|SCH-I510|SCH-I515|SCH-I600|SCH-I730|SCH-I760|SCH-I770|SCH-I830|SCH-I910|SCH-I920|SCH-I959|SCH-LC11|SCH-N150|SCH-N300|SCH-R100|SCH-R300|SCH-R351|SCH-R400|SCH-R410|SCH-T300|SCH-U310|SCH-U320|SCH-U350|SCH-U360|SCH-U365|SCH-U370|SCH-U380|SCH-U410|SCH-U430|SCH-U450|SCH-U460|SCH-U470|SCH-U490|SCH-U540|SCH-U550|SCH-U620|SCH-U640|SCH-U650|SCH-U660|SCH-U700|SCH-U740|SCH-U750|SCH-U810|SCH-U820|SCH-U900|SCH-U940|SCH-U960|SCS-26UC|SGH-A107|SGH-A117|SGH-A127|SGH-A137|SGH-A157|SGH-A167|SGH-A177|SGH-A187|SGH-A197|SGH-A227|SGH-A237|SGH-A257|SGH-A437|SGH-A517|SGH-A597|SGH-A637|SGH-A657|SGH-A667|SGH-A687|SGH-A697|SGH-A707|SGH-A717|SGH-A727|SGH-A737|SGH-A747|SGH-A767|SGH-A777|SGH-A797|SGH-A817|SGH-A827|SGH-A837|SGH-A847|SGH-A867|SGH-A877|SGH-A887|SGH-A897|SGH-A927|SGH-B100|SGH-B130|SGH-B200|SGH-B220|SGH-C100|SGH-C110|SGH-C120|SGH-C130|SGH-C140|SGH-C160|SGH-C170|SGH-C180|SGH-C200|SGH-C207|SGH-C210|SGH-C225|SGH-C230|SGH-C417|SGH-C450|SGH-D307|SGH-D347|SGH-D357|SGH-D407|SGH-D415|SGH-D780|SGH-D807|SGH-D980|SGH-E105|SGH-E200|SGH-E315|SGH-E316|SGH-E317|SGH-E335|SGH-E590|SGH-E635|SGH-E715|SGH-E890|SGH-F300|SGH-F480|SGH-I200|SGH-I300|SGH-I320|SGH-I550|SGH-I577|SGH-I600|SGH-I607|SGH-I617|SGH-I627|SGH-I637|SGH-I677|SGH-I700|SGH-I717|SGH-I727|SGH-i747M|SGH-I777|SGH-I780|SGH-I827|SGH-I847|SGH-I857|SGH-I896|SGH-I897|SGH-I900|SGH-I907|SGH-I917|SGH-I927|SGH-I937|SGH-I997|SGH-J150|SGH-J200|SGH-L170|SGH-L700|SGH-M110|SGH-M150|SGH-M200|SGH-N105|SGH-N500|SGH-N600|SGH-N620|SGH-N625|SGH-N700|SGH-N710|SGH-P107|SGH-P207|SGH-P300|SGH-P310|SGH-P520|SGH-P735|SGH-P777|SGH-Q105|SGH-R210|SGH-R220|SGH-R225|SGH-S105|SGH-S307|SGH-T109|SGH-T119|SGH-T139|SGH-T209|SGH-T219|SGH-T229|SGH-T239|SGH-T249|SGH-T259|SGH-T309|SGH-T319|SGH-T329|SGH-T339|SGH-T349|SGH-T359|SGH-T369|SGH-T379|SGH-T409|SGH-T429|SGH-T439|SGH-T459|SGH-T469|SGH-T479|SGH-T499|SGH-T509|SGH-T519|SGH-T539|SGH-T559|SGH-T589|SGH-T609|SGH-T619|SGH-T629|SGH-T639|SGH-T659|SGH-T669|SGH-T679|SGH-T709|SGH-T719|SGH-T729|SGH-T739|SGH-T746|SGH-T749|SGH-T759|SGH-T769|SGH-T809|SGH-T819|SGH-T839|SGH-T919|SGH-T929|SGH-T939|SGH-T959|SGH-T989|SGH-U100|SGH-U200|SGH-U800|SGH-V205|SGH-V206|SGH-X100|SGH-X105|SGH-X120|SGH-X140|SGH-X426|SGH-X427|SGH-X475|SGH-X495|SGH-X497|SGH-X507|SGH-X600|SGH-X610|SGH-X620|SGH-X630|SGH-X700|SGH-X820|SGH-X890|SGH-Z130|SGH-Z150|SGH-Z170|SGH-ZX10|SGH-ZX20|SHW-M110|SPH-A120|SPH-A400|SPH-A420|SPH-A460|SPH-A500|SPH-A560|SPH-A600|SPH-A620|SPH-A660|SPH-A700|SPH-A740|SPH-A760|SPH-A790|SPH-A800|SPH-A820|SPH-A840|SPH-A880|SPH-A900|SPH-A940|SPH-A960|SPH-D600|SPH-D700|SPH-D710|SPH-D720|SPH-I300|SPH-I325|SPH-I330|SPH-I350|SPH-I500|SPH-I600|SPH-I700|SPH-L700|SPH-M100|SPH-M220|SPH-M240|SPH-M300|SPH-M305|SPH-M320|SPH-M330|SPH-M350|SPH-M360|SPH-M370|SPH-M380|SPH-M510|SPH-M540|SPH-M550|SPH-M560|SPH-M570|SPH-M580|SPH-M610|SPH-M620|SPH-M630|SPH-M800|SPH-M810|SPH-M850|SPH-M900|SPH-M910|SPH-M920|SPH-M930|SPH-N100|SPH-N200|SPH-N240|SPH-N300|SPH-N400|SPH-Z400|SWC-E100|SCH-i909|GT-N7100|GT-N7105|SCH-I535|SM-N900A|SGH-I317|SGH-T999L|GT-S5360B|GT-I8262|GT-S6802|GT-S6312|GT-S6310|GT-S5312|GT-S5310|GT-I9105|GT-I8510|GT-S6790N|SM-G7105|SM-N9005|GT-S5301|GT-I9295|GT-I9195|SM-C101|GT-S7392|GT-S7560|GT-B7610|GT-I5510|GT-S7582|GT-S7530E|GT-I8750|SM-G9006V|SM-G9008V|SM-G9009D|SM-G900A|SM-G900D|SM-G900F|SM-G900H|SM-G900I|SM-G900J|SM-G900K|SM-G900L|SM-G900M|SM-G900P|SM-G900R4|SM-G900S|SM-G900T|SM-G900V|SM-G900W8|SHV-E160K|SCH-P709|SCH-P729|SM-T2558|GT-I9205|SM-G9350|SM-J120F|SM-G920F|SM-G920V|SM-G930F|SM-N910C',
  4328. 'LG' => '\bLG\b;|LG[- ]?(C800|C900|E400|E610|E900|E-900|F160|F180K|F180L|F180S|730|855|L160|LS740|LS840|LS970|LU6200|MS690|MS695|MS770|MS840|MS870|MS910|P500|P700|P705|VM696|AS680|AS695|AX840|C729|E970|GS505|272|C395|E739BK|E960|L55C|L75C|LS696|LS860|P769BK|P350|P500|P509|P870|UN272|US730|VS840|VS950|LN272|LN510|LS670|LS855|LW690|MN270|MN510|P509|P769|P930|UN200|UN270|UN510|UN610|US670|US740|US760|UX265|UX840|VN271|VN530|VS660|VS700|VS740|VS750|VS910|VS920|VS930|VX9200|VX11000|AX840A|LW770|P506|P925|P999|E612|D955|D802|MS323)',
  4329. 'Sony' => 'SonyST|SonyLT|SonyEricsson|SonyEricssonLT15iv|LT18i|E10i|LT28h|LT26w|SonyEricssonMT27i|C5303|C6902|C6903|C6906|C6943|D2533',
  4330. 'Asus' => 'Asus.*Galaxy|PadFone.*Mobile',
  4331. 'NokiaLumia' => 'Lumia [0-9]{3,4}',
  4332. // http://www.micromaxinfo.com/mobiles/smartphones
  4333. // Added because the codes might conflict with Acer Tablets.
  4334. 'Micromax' => 'Micromax.*\b(A210|A92|A88|A72|A111|A110Q|A115|A116|A110|A90S|A26|A51|A35|A54|A25|A27|A89|A68|A65|A57|A90)\b',
  4335. // @todo Complete the regex.
  4336. 'Palm' => 'PalmSource|Palm', // avantgo|blazer|elaine|hiptop|plucker|xiino ;
  4337. 'Vertu' => 'Vertu|Vertu.*Ltd|Vertu.*Ascent|Vertu.*Ayxta|Vertu.*Constellation(F|Quest)?|Vertu.*Monika|Vertu.*Signature', // Just for fun ;)
  4338. // http://www.pantech.co.kr/en/prod/prodList.do?gbrand=VEGA (PANTECH)
  4339. // Most of the VEGA devices are legacy. PANTECH seem to be newer devices based on Android.
  4340. 'Pantech' => 'PANTECH|IM-A850S|IM-A840S|IM-A830L|IM-A830K|IM-A830S|IM-A820L|IM-A810K|IM-A810S|IM-A800S|IM-T100K|IM-A725L|IM-A780L|IM-A775C|IM-A770K|IM-A760S|IM-A750K|IM-A740S|IM-A730S|IM-A720L|IM-A710K|IM-A690L|IM-A690S|IM-A650S|IM-A630K|IM-A600S|VEGA PTL21|PT003|P8010|ADR910L|P6030|P6020|P9070|P4100|P9060|P5000|CDM8992|TXT8045|ADR8995|IS11PT|P2030|P6010|P8000|PT002|IS06|CDM8999|P9050|PT001|TXT8040|P2020|P9020|P2000|P7040|P7000|C790',
  4341. // http://www.fly-phone.com/devices/smartphones/ ; Included only smartphones.
  4342. 'Fly' => 'IQ230|IQ444|IQ450|IQ440|IQ442|IQ441|IQ245|IQ256|IQ236|IQ255|IQ235|IQ245|IQ275|IQ240|IQ285|IQ280|IQ270|IQ260|IQ250',
  4343. // http://fr.wikomobile.com
  4344. 'Wiko' => 'KITE 4G|HIGHWAY|GETAWAY|STAIRWAY|DARKSIDE|DARKFULL|DARKNIGHT|DARKMOON|SLIDE|WAX 4G|RAINBOW|BLOOM|SUNSET|GOA(?!nna)|LENNY|BARRY|IGGY|OZZY|CINK FIVE|CINK PEAX|CINK PEAX 2|CINK SLIM|CINK SLIM 2|CINK +|CINK KING|CINK PEAX|CINK SLIM|SUBLIM',
  4345. 'iMobile' => 'i-mobile (IQ|i-STYLE|idea|ZAA|Hitz)',
  4346. // Added simvalley mobile just for fun. They have some interesting devices.
  4347. // http://www.simvalley.fr/telephonie---gps-_22_telephonie-mobile_telephones_.html
  4348. 'SimValley' => '\b(SP-80|XT-930|SX-340|XT-930|SX-310|SP-360|SP60|SPT-800|SP-120|SPT-800|SP-140|SPX-5|SPX-8|SP-100|SPX-8|SPX-12)\b',
  4349. // Wolfgang - a brand that is sold by Aldi supermarkets.
  4350. // http://www.wolfgangmobile.com/
  4351. 'Wolfgang' => 'AT-B24D|AT-AS50HD|AT-AS40W|AT-AS55HD|AT-AS45q2|AT-B26D|AT-AS50Q',
  4352. 'Alcatel' => 'Alcatel',
  4353. 'Nintendo' => 'Nintendo 3DS',
  4354. // http://en.wikipedia.org/wiki/Amoi
  4355. 'Amoi' => 'Amoi',
  4356. // http://en.wikipedia.org/wiki/INQ
  4357. 'INQ' => 'INQ',
  4358. // @Tapatalk is a mobile app; http://support.tapatalk.com/threads/smf-2-0-2-os-and-browser-detection-plugin-and-tapatalk.15565/#post-79039
  4359. 'GenericPhone' => 'Tapatalk|PDA;|SAGEM|\bmmp\b|pocket|\bpsp\b|symbian|Smartphone|smartfon|treo|up.browser|up.link|vodafone|\bwap\b|nokia|Series40|Series60|S60|SonyEricsson|N900|MAUI.*WAP.*Browser',
  4360. );
  4361. /**
  4362. * List of tablet devices.
  4363. *
  4364. * @var array
  4365. */
  4366. protected static $tabletDevices = array(
  4367. // @todo: check for mobile friendly emails topic.
  4368. 'iPad' => 'iPad|iPad.*Mobile',
  4369. // Removed |^.*Android.*Nexus(?!(?:Mobile).)*$
  4370. // @see #442
  4371. 'NexusTablet' => 'Android.*Nexus[\s]+(7|9|10)',
  4372. 'SamsungTablet' => 'SAMSUNG.*Tablet|Galaxy.*Tab|SC-01C|GT-P1000|GT-P1003|GT-P1010|GT-P3105|GT-P6210|GT-P6800|GT-P6810|GT-P7100|GT-P7300|GT-P7310|GT-P7500|GT-P7510|SCH-I800|SCH-I815|SCH-I905|SGH-I957|SGH-I987|SGH-T849|SGH-T859|SGH-T869|SPH-P100|GT-P3100|GT-P3108|GT-P3110|GT-P5100|GT-P5110|GT-P6200|GT-P7320|GT-P7511|GT-N8000|GT-P8510|SGH-I497|SPH-P500|SGH-T779|SCH-I705|SCH-I915|GT-N8013|GT-P3113|GT-P5113|GT-P8110|GT-N8010|GT-N8005|GT-N8020|GT-P1013|GT-P6201|GT-P7501|GT-N5100|GT-N5105|GT-N5110|SHV-E140K|SHV-E140L|SHV-E140S|SHV-E150S|SHV-E230K|SHV-E230L|SHV-E230S|SHW-M180K|SHW-M180L|SHW-M180S|SHW-M180W|SHW-M300W|SHW-M305W|SHW-M380K|SHW-M380S|SHW-M380W|SHW-M430W|SHW-M480K|SHW-M480S|SHW-M480W|SHW-M485W|SHW-M486W|SHW-M500W|GT-I9228|SCH-P739|SCH-I925|GT-I9200|GT-P5200|GT-P5210|GT-P5210X|SM-T311|SM-T310|SM-T310X|SM-T210|SM-T210R|SM-T211|SM-P600|SM-P601|SM-P605|SM-P900|SM-P901|SM-T217|SM-T217A|SM-T217S|SM-P6000|SM-T3100|SGH-I467|XE500|SM-T110|GT-P5220|GT-I9200X|GT-N5110X|GT-N5120|SM-P905|SM-T111|SM-T2105|SM-T315|SM-T320|SM-T320X|SM-T321|SM-T520|SM-T525|SM-T530NU|SM-T230NU|SM-T330NU|SM-T900|XE500T1C|SM-P605V|SM-P905V|SM-T337V|SM-T537V|SM-T707V|SM-T807V|SM-P600X|SM-P900X|SM-T210X|SM-T230|SM-T230X|SM-T325|GT-P7503|SM-T531|SM-T330|SM-T530|SM-T705|SM-T705C|SM-T535|SM-T331|SM-T800|SM-T700|SM-T537|SM-T807|SM-P907A|SM-T337A|SM-T537A|SM-T707A|SM-T807A|SM-T237|SM-T807P|SM-P607T|SM-T217T|SM-T337T|SM-T807T|SM-T116NQ|SM-T116BU|SM-P550|SM-T350|SM-T550|SM-T9000|SM-P9000|SM-T705Y|SM-T805|GT-P3113|SM-T710|SM-T810|SM-T815|SM-T360|SM-T533|SM-T113|SM-T335|SM-T715|SM-T560|SM-T670|SM-T677|SM-T377|SM-T567|SM-T357T|SM-T555|SM-T561|SM-T713|SM-T719|SM-T813|SM-T819|SM-T580|SM-T355Y|SM-T280|SM-T817A|SM-T820|SM-W700|SM-P580|SM-T587|SM-P350|SM-P555M|SM-P355M|SM-T113NU|SM-T815Y', // SCH-P709|SCH-P729|SM-T2558|GT-I9205 - Samsung Mega - treat them like a regular phone.
  4373. // http://docs.aws.amazon.com/silk/latest/developerguide/user-agent.html
  4374. 'Kindle' => 'Kindle|Silk.*Accelerated|Android.*\b(KFOT|KFTT|KFJWI|KFJWA|KFOTE|KFSOWI|KFTHWI|KFTHWA|KFAPWI|KFAPWA|WFJWAE|KFSAWA|KFSAWI|KFASWI|KFARWI|KFFOWI|KFGIWI|KFMEWI)\b|Android.*Silk/[0-9.]+ like Chrome/[0-9.]+ (?!Mobile)',
  4375. // Only the Surface tablets with Windows RT are considered mobile.
  4376. // http://msdn.microsoft.com/en-us/library/ie/hh920767(v=vs.85).aspx
  4377. 'SurfaceTablet' => 'Windows NT [0-9.]+; ARM;.*(Tablet|ARMBJS)',
  4378. // http://shopping1.hp.com/is-bin/INTERSHOP.enfinity/WFS/WW-USSMBPublicStore-Site/en_US/-/USD/ViewStandardCatalog-Browse?CatalogCategoryID=JfIQ7EN5lqMAAAEyDcJUDwMT
  4379. 'HPTablet' => 'HP Slate (7|8|10)|HP ElitePad 900|hp-tablet|EliteBook.*Touch|HP 8|Slate 21|HP SlateBook 10',
  4380. // Watch out for PadFone, see #132.
  4381. // http://www.asus.com/de/Tablets_Mobile/Memo_Pad_Products/
  4382. 'AsusTablet' => '^.*PadFone((?!Mobile).)*$|Transformer|TF101|TF101G|TF300T|TF300TG|TF300TL|TF700T|TF700KL|TF701T|TF810C|ME171|ME301T|ME302C|ME371MG|ME370T|ME372MG|ME172V|ME173X|ME400C|Slider SL101|\bK00F\b|\bK00C\b|\bK00E\b|\bK00L\b|TX201LA|ME176C|ME102A|\bM80TA\b|ME372CL|ME560CG|ME372CG|ME302KL| K010 | K011 | K017 | K01E |ME572C|ME103K|ME170C|ME171C|\bME70C\b|ME581C|ME581CL|ME8510C|ME181C|P01Y|PO1MA|P01Z|\bP027\b',
  4383. 'BlackBerryTablet' => 'PlayBook|RIM Tablet',
  4384. 'HTCtablet' => 'HTC_Flyer_P512|HTC Flyer|HTC Jetstream|HTC-P715a|HTC EVO View 4G|PG41200|PG09410',
  4385. 'MotorolaTablet' => 'xoom|sholest|MZ615|MZ605|MZ505|MZ601|MZ602|MZ603|MZ604|MZ606|MZ607|MZ608|MZ609|MZ615|MZ616|MZ617',
  4386. 'NookTablet' => 'Android.*Nook|NookColor|nook browser|BNRV200|BNRV200A|BNTV250|BNTV250A|BNTV400|BNTV600|LogicPD Zoom2',
  4387. // http://www.acer.ro/ac/ro/RO/content/drivers
  4388. // http://www.packardbell.co.uk/pb/en/GB/content/download (Packard Bell is part of Acer)
  4389. // http://us.acer.com/ac/en/US/content/group/tablets
  4390. // http://www.acer.de/ac/de/DE/content/models/tablets/
  4391. // Can conflict with Micromax and Motorola phones codes.
  4392. 'AcerTablet' => 'Android.*; \b(A100|A101|A110|A200|A210|A211|A500|A501|A510|A511|A700|A701|W500|W500P|W501|W501P|W510|W511|W700|G100|G100W|B1-A71|B1-710|B1-711|A1-810|A1-811|A1-830)\b|W3-810|\bA3-A10\b|\bA3-A11\b|\bA3-A20\b|\bA3-A30',
  4393. // http://eu.computers.toshiba-europe.com/innovation/family/Tablets/1098744/banner_id/tablet_footerlink/
  4394. // http://us.toshiba.com/tablets/tablet-finder
  4395. // http://www.toshiba.co.jp/regza/tablet/
  4396. 'ToshibaTablet' => 'Android.*(AT100|AT105|AT200|AT205|AT270|AT275|AT300|AT305|AT1S5|AT500|AT570|AT700|AT830)|TOSHIBA.*FOLIO',
  4397. // http://www.nttdocomo.co.jp/english/service/developer/smart_phone/technical_info/spec/index.html
  4398. // http://www.lg.com/us/tablets
  4399. 'LGTablet' => '\bL-06C|LG-V909|LG-V900|LG-V700|LG-V510|LG-V500|LG-V410|LG-V400|LG-VK810\b',
  4400. 'FujitsuTablet' => 'Android.*\b(F-01D|F-02F|F-05E|F-10D|M532|Q572)\b',
  4401. // Prestigio Tablets http://www.prestigio.com/support
  4402. 'PrestigioTablet' => 'PMP3170B|PMP3270B|PMP3470B|PMP7170B|PMP3370B|PMP3570C|PMP5870C|PMP3670B|PMP5570C|PMP5770D|PMP3970B|PMP3870C|PMP5580C|PMP5880D|PMP5780D|PMP5588C|PMP7280C|PMP7280C3G|PMP7280|PMP7880D|PMP5597D|PMP5597|PMP7100D|PER3464|PER3274|PER3574|PER3884|PER5274|PER5474|PMP5097CPRO|PMP5097|PMP7380D|PMP5297C|PMP5297C_QUAD|PMP812E|PMP812E3G|PMP812F|PMP810E|PMP880TD|PMT3017|PMT3037|PMT3047|PMT3057|PMT7008|PMT5887|PMT5001|PMT5002',
  4403. // http://support.lenovo.com/en_GB/downloads/default.page?#
  4404. 'LenovoTablet' => 'Lenovo TAB|Idea(Tab|Pad)( A1|A10| K1|)|ThinkPad([ ]+)?Tablet|YT3-850M|YT3-X90L|YT3-X90F|YT3-X90X|Lenovo.*(S2109|S2110|S5000|S6000|K3011|A3000|A3500|A1000|A2107|A2109|A1107|A5500|A7600|B6000|B8000|B8080)(-|)(FL|F|HV|H|)',
  4405. // http://www.dell.com/support/home/us/en/04/Products/tab_mob/tablets
  4406. 'DellTablet' => 'Venue 11|Venue 8|Venue 7|Dell Streak 10|Dell Streak 7',
  4407. // http://www.yarvik.com/en/matrix/tablets/
  4408. 'YarvikTablet' => 'Android.*\b(TAB210|TAB211|TAB224|TAB250|TAB260|TAB264|TAB310|TAB360|TAB364|TAB410|TAB411|TAB420|TAB424|TAB450|TAB460|TAB461|TAB464|TAB465|TAB467|TAB468|TAB07-100|TAB07-101|TAB07-150|TAB07-151|TAB07-152|TAB07-200|TAB07-201-3G|TAB07-210|TAB07-211|TAB07-212|TAB07-214|TAB07-220|TAB07-400|TAB07-485|TAB08-150|TAB08-200|TAB08-201-3G|TAB08-201-30|TAB09-100|TAB09-211|TAB09-410|TAB10-150|TAB10-201|TAB10-211|TAB10-400|TAB10-410|TAB13-201|TAB274EUK|TAB275EUK|TAB374EUK|TAB462EUK|TAB474EUK|TAB9-200)\b',
  4409. 'MedionTablet' => 'Android.*\bOYO\b|LIFE.*(P9212|P9514|P9516|S9512)|LIFETAB',
  4410. 'ArnovaTablet' => '97G4|AN10G2|AN7bG3|AN7fG3|AN8G3|AN8cG3|AN7G3|AN9G3|AN7dG3|AN7dG3ST|AN7dG3ChildPad|AN10bG3|AN10bG3DT|AN9G2',
  4411. // http://www.intenso.de/kategorie_en.php?kategorie=33
  4412. // @todo: http://www.nbhkdz.com/read/b8e64202f92a2df129126bff.html - investigate
  4413. 'IntensoTablet' => 'INM8002KP|INM1010FP|INM805ND|Intenso Tab|TAB1004',
  4414. // IRU.ru Tablets http://www.iru.ru/catalog/soho/planetable/
  4415. 'IRUTablet' => 'M702pro',
  4416. 'MegafonTablet' => 'MegaFon V9|\bZTE V9\b|Android.*\bMT7A\b',
  4417. // http://www.e-boda.ro/tablete-pc.html
  4418. 'EbodaTablet' => 'E-Boda (Supreme|Impresspeed|Izzycomm|Essential)',
  4419. // http://www.allview.ro/produse/droseries/lista-tablete-pc/
  4420. 'AllViewTablet' => 'Allview.*(Viva|Alldro|City|Speed|All TV|Frenzy|Quasar|Shine|TX1|AX1|AX2)',
  4421. // http://wiki.archosfans.com/index.php?title=Main_Page
  4422. // @note Rewrite the regex format after we add more UAs.
  4423. 'ArchosTablet' => '\b(101G9|80G9|A101IT)\b|Qilive 97R|Archos5|\bARCHOS (70|79|80|90|97|101|FAMILYPAD|)(b|c|)(G10| Cobalt| TITANIUM(HD|)| Xenon| Neon|XSK| 2| XS 2| PLATINUM| CARBON|GAMEPAD)\b',
  4424. // http://www.ainol.com/plugin.php?identifier=ainol&module=product
  4425. 'AinolTablet' => 'NOVO7|NOVO8|NOVO10|Novo7Aurora|Novo7Basic|NOVO7PALADIN|novo9-Spark',
  4426. 'NokiaLumiaTablet' => 'Lumia 2520',
  4427. // @todo: inspect http://esupport.sony.com/US/p/select-system.pl?DIRECTOR=DRIVER
  4428. // Readers http://www.atsuhiro-me.net/ebook/sony-reader/sony-reader-web-browser
  4429. // http://www.sony.jp/support/tablet/
  4430. 'SonyTablet' => 'Sony.*Tablet|Xperia Tablet|Sony Tablet S|SO-03E|SGPT12|SGPT13|SGPT114|SGPT121|SGPT122|SGPT123|SGPT111|SGPT112|SGPT113|SGPT131|SGPT132|SGPT133|SGPT211|SGPT212|SGPT213|SGP311|SGP312|SGP321|EBRD1101|EBRD1102|EBRD1201|SGP351|SGP341|SGP511|SGP512|SGP521|SGP541|SGP551|SGP621|SGP612|SOT31',
  4431. // http://www.support.philips.com/support/catalog/worldproducts.jsp?userLanguage=en&userCountry=cn&categoryid=3G_LTE_TABLET_SU_CN_CARE&title=3G%20tablets%20/%20LTE%20range&_dyncharset=UTF-8
  4432. 'PhilipsTablet' => '\b(PI2010|PI3000|PI3100|PI3105|PI3110|PI3205|PI3210|PI3900|PI4010|PI7000|PI7100)\b',
  4433. // db + http://www.cube-tablet.com/buy-products.html
  4434. 'CubeTablet' => 'Android.*(K8GT|U9GT|U10GT|U16GT|U17GT|U18GT|U19GT|U20GT|U23GT|U30GT)|CUBE U8GT',
  4435. // http://www.cobyusa.com/?p=pcat&pcat_id=3001
  4436. 'CobyTablet' => 'MID1042|MID1045|MID1125|MID1126|MID7012|MID7014|MID7015|MID7034|MID7035|MID7036|MID7042|MID7048|MID7127|MID8042|MID8048|MID8127|MID9042|MID9740|MID9742|MID7022|MID7010',
  4437. // http://www.match.net.cn/products.asp
  4438. 'MIDTablet' => 'M9701|M9000|M9100|M806|M1052|M806|T703|MID701|MID713|MID710|MID727|MID760|MID830|MID728|MID933|MID125|MID810|MID732|MID120|MID930|MID800|MID731|MID900|MID100|MID820|MID735|MID980|MID130|MID833|MID737|MID960|MID135|MID860|MID736|MID140|MID930|MID835|MID733|MID4X10',
  4439. // http://www.msi.com/support
  4440. // @todo Research the Windows Tablets.
  4441. 'MSITablet' => 'MSI \b(Primo 73K|Primo 73L|Primo 81L|Primo 77|Primo 93|Primo 75|Primo 76|Primo 73|Primo 81|Primo 91|Primo 90|Enjoy 71|Enjoy 7|Enjoy 10)\b',
  4442. // @todo http://www.kyoceramobile.com/support/drivers/
  4443. // 'KyoceraTablet' => null,
  4444. // @todo http://intexuae.com/index.php/category/mobile-devices/tablets-products/
  4445. // 'IntextTablet' => null,
  4446. // http://pdadb.net/index.php?m=pdalist&list=SMiT (NoName Chinese Tablets)
  4447. // http://www.imp3.net/14/show.php?itemid=20454
  4448. 'SMiTTablet' => 'Android.*(\bMID\b|MID-560|MTV-T1200|MTV-PND531|MTV-P1101|MTV-PND530)',
  4449. // http://www.rock-chips.com/index.php?do=prod&pid=2
  4450. 'RockChipTablet' => 'Android.*(RK2818|RK2808A|RK2918|RK3066)|RK2738|RK2808A',
  4451. // http://www.fly-phone.com/devices/tablets/ ; http://www.fly-phone.com/service/
  4452. 'FlyTablet' => 'IQ310|Fly Vision',
  4453. // http://www.bqreaders.com/gb/tablets-prices-sale.html
  4454. 'bqTablet' => 'Android.*(bq)?.*(Elcano|Curie|Edison|Maxwell|Kepler|Pascal|Tesla|Hypatia|Platon|Newton|Livingstone|Cervantes|Avant|Aquaris [E|M]10)|Maxwell.*Lite|Maxwell.*Plus',
  4455. // http://www.huaweidevice.com/worldwide/productFamily.do?method=index&directoryId=5011&treeId=3290
  4456. // http://www.huaweidevice.com/worldwide/downloadCenter.do?method=index&directoryId=3372&treeId=0&tb=1&type=software (including legacy tablets)
  4457. 'HuaweiTablet' => 'MediaPad|MediaPad 7 Youth|IDEOS S7|S7-201c|S7-202u|S7-101|S7-103|S7-104|S7-105|S7-106|S7-201|S7-Slim',
  4458. // Nec or Medias Tab
  4459. 'NecTablet' => '\bN-06D|\bN-08D',
  4460. // Pantech Tablets: http://www.pantechusa.com/phones/
  4461. 'PantechTablet' => 'Pantech.*P4100',
  4462. // Broncho Tablets: http://www.broncho.cn/ (hard to find)
  4463. 'BronchoTablet' => 'Broncho.*(N701|N708|N802|a710)',
  4464. // http://versusuk.com/support.html
  4465. 'VersusTablet' => 'TOUCHPAD.*[78910]|\bTOUCHTAB\b',
  4466. // http://www.zync.in/index.php/our-products/tablet-phablets
  4467. 'ZyncTablet' => 'z1000|Z99 2G|z99|z930|z999|z990|z909|Z919|z900',
  4468. // http://www.positivoinformatica.com.br/www/pessoal/tablet-ypy/
  4469. 'PositivoTablet' => 'TB07STA|TB10STA|TB07FTA|TB10FTA',
  4470. // https://www.nabitablet.com/
  4471. 'NabiTablet' => 'Android.*\bNabi',
  4472. 'KoboTablet' => 'Kobo Touch|\bK080\b|\bVox\b Build|\bArc\b Build',
  4473. // French Danew Tablets http://www.danew.com/produits-tablette.php
  4474. 'DanewTablet' => 'DSlide.*\b(700|701R|702|703R|704|802|970|971|972|973|974|1010|1012)\b',
  4475. // Texet Tablets and Readers http://www.texet.ru/tablet/
  4476. 'TexetTablet' => 'NaviPad|TB-772A|TM-7045|TM-7055|TM-9750|TM-7016|TM-7024|TM-7026|TM-7041|TM-7043|TM-7047|TM-8041|TM-9741|TM-9747|TM-9748|TM-9751|TM-7022|TM-7021|TM-7020|TM-7011|TM-7010|TM-7023|TM-7025|TM-7037W|TM-7038W|TM-7027W|TM-9720|TM-9725|TM-9737W|TM-1020|TM-9738W|TM-9740|TM-9743W|TB-807A|TB-771A|TB-727A|TB-725A|TB-719A|TB-823A|TB-805A|TB-723A|TB-715A|TB-707A|TB-705A|TB-709A|TB-711A|TB-890HD|TB-880HD|TB-790HD|TB-780HD|TB-770HD|TB-721HD|TB-710HD|TB-434HD|TB-860HD|TB-840HD|TB-760HD|TB-750HD|TB-740HD|TB-730HD|TB-722HD|TB-720HD|TB-700HD|TB-500HD|TB-470HD|TB-431HD|TB-430HD|TB-506|TB-504|TB-446|TB-436|TB-416|TB-146SE|TB-126SE',
  4477. // Avoid detecting 'PLAYSTATION 3' as mobile.
  4478. 'PlaystationTablet' => 'Playstation.*(Portable|Vita)',
  4479. // http://www.trekstor.de/surftabs.html
  4480. 'TrekstorTablet' => 'ST10416-1|VT10416-1|ST70408-1|ST702xx-1|ST702xx-2|ST80208|ST97216|ST70104-2|VT10416-2|ST10216-2A|SurfTab',
  4481. // http://www.pyleaudio.com/Products.aspx?%2fproducts%2fPersonal-Electronics%2fTablets
  4482. 'PyleAudioTablet' => '\b(PTBL10CEU|PTBL10C|PTBL72BC|PTBL72BCEU|PTBL7CEU|PTBL7C|PTBL92BC|PTBL92BCEU|PTBL9CEU|PTBL9CUK|PTBL9C)\b',
  4483. // http://www.advandigital.com/index.php?link=content-product&jns=JP001
  4484. // because of the short codenames we have to include whitespaces to reduce the possible conflicts.
  4485. 'AdvanTablet' => 'Android.* \b(E3A|T3X|T5C|T5B|T3E|T3C|T3B|T1J|T1F|T2A|T1H|T1i|E1C|T1-E|T5-A|T4|E1-B|T2Ci|T1-B|T1-D|O1-A|E1-A|T1-A|T3A|T4i)\b ',
  4486. // http://www.danytech.com/category/tablet-pc
  4487. 'DanyTechTablet' => 'Genius Tab G3|Genius Tab S2|Genius Tab Q3|Genius Tab G4|Genius Tab Q4|Genius Tab G-II|Genius TAB GII|Genius TAB GIII|Genius Tab S1',
  4488. // http://www.galapad.net/product.html
  4489. 'GalapadTablet' => 'Android.*\bG1\b',
  4490. // http://www.micromaxinfo.com/tablet/funbook
  4491. 'MicromaxTablet' => 'Funbook|Micromax.*\b(P250|P560|P360|P362|P600|P300|P350|P500|P275)\b',
  4492. // http://www.karbonnmobiles.com/products_tablet.php
  4493. 'KarbonnTablet' => 'Android.*\b(A39|A37|A34|ST8|ST10|ST7|Smart Tab3|Smart Tab2)\b',
  4494. // http://www.myallfine.com/Products.asp
  4495. 'AllFineTablet' => 'Fine7 Genius|Fine7 Shine|Fine7 Air|Fine8 Style|Fine9 More|Fine10 Joy|Fine11 Wide',
  4496. // http://www.proscanvideo.com/products-search.asp?itemClass=TABLET&itemnmbr=
  4497. 'PROSCANTablet' => '\b(PEM63|PLT1023G|PLT1041|PLT1044|PLT1044G|PLT1091|PLT4311|PLT4311PL|PLT4315|PLT7030|PLT7033|PLT7033D|PLT7035|PLT7035D|PLT7044K|PLT7045K|PLT7045KB|PLT7071KG|PLT7072|PLT7223G|PLT7225G|PLT7777G|PLT7810K|PLT7849G|PLT7851G|PLT7852G|PLT8015|PLT8031|PLT8034|PLT8036|PLT8080K|PLT8082|PLT8088|PLT8223G|PLT8234G|PLT8235G|PLT8816K|PLT9011|PLT9045K|PLT9233G|PLT9735|PLT9760G|PLT9770G)\b',
  4498. // http://www.yonesnav.com/products/products.php
  4499. 'YONESTablet' => 'BQ1078|BC1003|BC1077|RK9702|BC9730|BC9001|IT9001|BC7008|BC7010|BC708|BC728|BC7012|BC7030|BC7027|BC7026',
  4500. // http://www.cjshowroom.com/eproducts.aspx?classcode=004001001
  4501. // China manufacturer makes tablets for different small brands (eg. http://www.zeepad.net/index.html)
  4502. 'ChangJiaTablet' => 'TPC7102|TPC7103|TPC7105|TPC7106|TPC7107|TPC7201|TPC7203|TPC7205|TPC7210|TPC7708|TPC7709|TPC7712|TPC7110|TPC8101|TPC8103|TPC8105|TPC8106|TPC8203|TPC8205|TPC8503|TPC9106|TPC9701|TPC97101|TPC97103|TPC97105|TPC97106|TPC97111|TPC97113|TPC97203|TPC97603|TPC97809|TPC97205|TPC10101|TPC10103|TPC10106|TPC10111|TPC10203|TPC10205|TPC10503',
  4503. // http://www.gloryunion.cn/products.asp
  4504. // http://www.allwinnertech.com/en/apply/mobile.html
  4505. // http://www.ptcl.com.pk/pd_content.php?pd_id=284 (EVOTAB)
  4506. // @todo: Softwiner tablets?
  4507. // aka. Cute or Cool tablets. Not sure yet, must research to avoid collisions.
  4508. 'GUTablet' => 'TX-A1301|TX-M9002|Q702|kf026', // A12R|D75A|D77|D79|R83|A95|A106C|R15|A75|A76|D71|D72|R71|R73|R77|D82|R85|D92|A97|D92|R91|A10F|A77F|W71F|A78F|W78F|W81F|A97F|W91F|W97F|R16G|C72|C73E|K72|K73|R96G
  4509. // http://www.pointofview-online.com/showroom.php?shop_mode=product_listing&category_id=118
  4510. 'PointOfViewTablet' => 'TAB-P506|TAB-navi-7-3G-M|TAB-P517|TAB-P-527|TAB-P701|TAB-P703|TAB-P721|TAB-P731N|TAB-P741|TAB-P825|TAB-P905|TAB-P925|TAB-PR945|TAB-PL1015|TAB-P1025|TAB-PI1045|TAB-P1325|TAB-PROTAB[0-9]+|TAB-PROTAB25|TAB-PROTAB26|TAB-PROTAB27|TAB-PROTAB26XL|TAB-PROTAB2-IPS9|TAB-PROTAB30-IPS9|TAB-PROTAB25XXL|TAB-PROTAB26-IPS10|TAB-PROTAB30-IPS10',
  4511. // http://www.overmax.pl/pl/katalog-produktow,p8/tablety,c14/
  4512. // @todo: add more tests.
  4513. 'OvermaxTablet' => 'OV-(SteelCore|NewBase|Basecore|Baseone|Exellen|Quattor|EduTab|Solution|ACTION|BasicTab|TeddyTab|MagicTab|Stream|TB-08|TB-09)',
  4514. // http://hclmetablet.com/India/index.php
  4515. 'HCLTablet' => 'HCL.*Tablet|Connect-3G-2.0|Connect-2G-2.0|ME Tablet U1|ME Tablet U2|ME Tablet G1|ME Tablet X1|ME Tablet Y2|ME Tablet Sync',
  4516. // http://www.edigital.hu/Tablet_es_e-book_olvaso/Tablet-c18385.html
  4517. 'DPSTablet' => 'DPS Dream 9|DPS Dual 7',
  4518. // http://www.visture.com/index.asp
  4519. 'VistureTablet' => 'V97 HD|i75 3G|Visture V4( HD)?|Visture V5( HD)?|Visture V10',
  4520. // http://www.mijncresta.nl/tablet
  4521. 'CrestaTablet' => 'CTP(-)?810|CTP(-)?818|CTP(-)?828|CTP(-)?838|CTP(-)?888|CTP(-)?978|CTP(-)?980|CTP(-)?987|CTP(-)?988|CTP(-)?989',
  4522. // MediaTek - http://www.mediatek.com/_en/01_products/02_proSys.php?cata_sn=1&cata1_sn=1&cata2_sn=309
  4523. 'MediatekTablet' => '\bMT8125|MT8389|MT8135|MT8377\b',
  4524. // Concorde tab
  4525. 'ConcordeTablet' => 'Concorde([ ]+)?Tab|ConCorde ReadMan',
  4526. // GoClever Tablets - http://www.goclever.com/uk/products,c1/tablet,c5/
  4527. 'GoCleverTablet' => 'GOCLEVER TAB|A7GOCLEVER|M1042|M7841|M742|R1042BK|R1041|TAB A975|TAB A7842|TAB A741|TAB A741L|TAB M723G|TAB M721|TAB A1021|TAB I921|TAB R721|TAB I720|TAB T76|TAB R70|TAB R76.2|TAB R106|TAB R83.2|TAB M813G|TAB I721|GCTA722|TAB I70|TAB I71|TAB S73|TAB R73|TAB R74|TAB R93|TAB R75|TAB R76.1|TAB A73|TAB A93|TAB A93.2|TAB T72|TAB R83|TAB R974|TAB R973|TAB A101|TAB A103|TAB A104|TAB A104.2|R105BK|M713G|A972BK|TAB A971|TAB R974.2|TAB R104|TAB R83.3|TAB A1042',
  4528. // Modecom Tablets - http://www.modecom.eu/tablets/portal/
  4529. 'ModecomTablet' => 'FreeTAB 9000|FreeTAB 7.4|FreeTAB 7004|FreeTAB 7800|FreeTAB 2096|FreeTAB 7.5|FreeTAB 1014|FreeTAB 1001 |FreeTAB 8001|FreeTAB 9706|FreeTAB 9702|FreeTAB 7003|FreeTAB 7002|FreeTAB 1002|FreeTAB 7801|FreeTAB 1331|FreeTAB 1004|FreeTAB 8002|FreeTAB 8014|FreeTAB 9704|FreeTAB 1003',
  4530. // Vonino Tablets - http://www.vonino.eu/tablets
  4531. 'VoninoTablet' => '\b(Argus[ _]?S|Diamond[ _]?79HD|Emerald[ _]?78E|Luna[ _]?70C|Onyx[ _]?S|Onyx[ _]?Z|Orin[ _]?HD|Orin[ _]?S|Otis[ _]?S|SpeedStar[ _]?S|Magnet[ _]?M9|Primus[ _]?94[ _]?3G|Primus[ _]?94HD|Primus[ _]?QS|Android.*\bQ8\b|Sirius[ _]?EVO[ _]?QS|Sirius[ _]?QS|Spirit[ _]?S)\b',
  4532. // ECS Tablets - http://www.ecs.com.tw/ECSWebSite/Product/Product_Tablet_List.aspx?CategoryID=14&MenuID=107&childid=M_107&LanID=0
  4533. 'ECSTablet' => 'V07OT2|TM105A|S10OT1|TR10CS1',
  4534. // Storex Tablets - http://storex.fr/espace_client/support.html
  4535. // @note: no need to add all the tablet codes since they are guided by the first regex.
  4536. 'StorexTablet' => 'eZee[_\']?(Tab|Go)[0-9]+|TabLC7|Looney Tunes Tab',
  4537. // Generic Vodafone tablets.
  4538. 'VodafoneTablet' => 'SmartTab([ ]+)?[0-9]+|SmartTabII10|SmartTabII7|VF-1497',
  4539. // French tablets - Essentiel B http://www.boulanger.fr/tablette_tactile_e-book/tablette_tactile_essentiel_b/cl_68908.htm?multiChoiceToDelete=brand&mc_brand=essentielb
  4540. // Aka: http://www.essentielb.fr/
  4541. 'EssentielBTablet' => 'Smart[ \']?TAB[ ]+?[0-9]+|Family[ \']?TAB2',
  4542. // Ross & Moor - http://ross-moor.ru/
  4543. 'RossMoorTablet' => 'RM-790|RM-997|RMD-878G|RMD-974R|RMT-705A|RMT-701|RME-601|RMT-501|RMT-711',
  4544. // i-mobile http://product.i-mobilephone.com/Mobile_Device
  4545. 'iMobileTablet' => 'i-mobile i-note',
  4546. // http://www.tolino.de/de/vergleichen/
  4547. 'TolinoTablet' => 'tolino tab [0-9.]+|tolino shine',
  4548. // AudioSonic - a Kmart brand
  4549. // http://www.kmart.com.au/webapp/wcs/stores/servlet/Search?langId=-1&storeId=10701&catalogId=10001&categoryId=193001&pageSize=72&currentPage=1&searchCategory=193001%2b4294965664&sortBy=p_MaxPrice%7c1
  4550. 'AudioSonicTablet' => '\bC-22Q|T7-QC|T-17B|T-17P\b',
  4551. // AMPE Tablets - http://www.ampe.com.my/product-category/tablets/
  4552. // @todo: add them gradually to avoid conflicts.
  4553. 'AMPETablet' => 'Android.* A78 ',
  4554. // Skk Mobile - http://skkmobile.com.ph/product_tablets.php
  4555. 'SkkTablet' => 'Android.* (SKYPAD|PHOENIX|CYCLOPS)',
  4556. // Tecno Mobile (only tablet) - http://www.tecno-mobile.com/index.php/product?filterby=smart&list_order=all&page=1
  4557. 'TecnoTablet' => 'TECNO P9',
  4558. // JXD (consoles & tablets) - http://jxd.hk/products.asp?selectclassid=009008&clsid=3
  4559. 'JXDTablet' => 'Android.* \b(F3000|A3300|JXD5000|JXD3000|JXD2000|JXD300B|JXD300|S5800|S7800|S602b|S5110b|S7300|S5300|S602|S603|S5100|S5110|S601|S7100a|P3000F|P3000s|P101|P200s|P1000m|P200m|P9100|P1000s|S6600b|S908|P1000|P300|S18|S6600|S9100)\b',
  4560. // i-Joy tablets - http://www.i-joy.es/en/cat/products/tablets/
  4561. 'iJoyTablet' => 'Tablet (Spirit 7|Essentia|Galatea|Fusion|Onix 7|Landa|Titan|Scooby|Deox|Stella|Themis|Argon|Unique 7|Sygnus|Hexen|Finity 7|Cream|Cream X2|Jade|Neon 7|Neron 7|Kandy|Scape|Saphyr 7|Rebel|Biox|Rebel|Rebel 8GB|Myst|Draco 7|Myst|Tab7-004|Myst|Tadeo Jones|Tablet Boing|Arrow|Draco Dual Cam|Aurix|Mint|Amity|Revolution|Finity 9|Neon 9|T9w|Amity 4GB Dual Cam|Stone 4GB|Stone 8GB|Andromeda|Silken|X2|Andromeda II|Halley|Flame|Saphyr 9,7|Touch 8|Planet|Triton|Unique 10|Hexen 10|Memphis 4GB|Memphis 8GB|Onix 10)',
  4562. // http://www.intracon.eu/tablet
  4563. 'FX2Tablet' => 'FX2 PAD7|FX2 PAD10',
  4564. // http://www.xoro.de/produkte/
  4565. // @note: Might be the same brand with 'Simply tablets'
  4566. 'XoroTablet' => 'KidsPAD 701|PAD[ ]?712|PAD[ ]?714|PAD[ ]?716|PAD[ ]?717|PAD[ ]?718|PAD[ ]?720|PAD[ ]?721|PAD[ ]?722|PAD[ ]?790|PAD[ ]?792|PAD[ ]?900|PAD[ ]?9715D|PAD[ ]?9716DR|PAD[ ]?9718DR|PAD[ ]?9719QR|PAD[ ]?9720QR|TelePAD1030|Telepad1032|TelePAD730|TelePAD731|TelePAD732|TelePAD735Q|TelePAD830|TelePAD9730|TelePAD795|MegaPAD 1331|MegaPAD 1851|MegaPAD 2151',
  4567. // http://www1.viewsonic.com/products/computing/tablets/
  4568. 'ViewsonicTablet' => 'ViewPad 10pi|ViewPad 10e|ViewPad 10s|ViewPad E72|ViewPad7|ViewPad E100|ViewPad 7e|ViewSonic VB733|VB100a',
  4569. // http://www.odys.de/web/internet-tablet_en.html
  4570. 'OdysTablet' => 'LOOX|XENO10|ODYS[ -](Space|EVO|Xpress|NOON)|\bXELIO\b|Xelio10Pro|XELIO7PHONETAB|XELIO10EXTREME|XELIOPT2|NEO_QUAD10',
  4571. // http://www.captiva-power.de/products.html#tablets-en
  4572. 'CaptivaTablet' => 'CAPTIVA PAD',
  4573. // IconBIT - http://www.iconbit.com/products/tablets/
  4574. 'IconbitTablet' => 'NetTAB|NT-3702|NT-3702S|NT-3702S|NT-3603P|NT-3603P|NT-0704S|NT-0704S|NT-3805C|NT-3805C|NT-0806C|NT-0806C|NT-0909T|NT-0909T|NT-0907S|NT-0907S|NT-0902S|NT-0902S',
  4575. // http://www.teclast.com/topic.php?channelID=70&topicID=140&pid=63
  4576. 'TeclastTablet' => 'T98 4G|\bP80\b|\bX90HD\b|X98 Air|X98 Air 3G|\bX89\b|P80 3G|\bX80h\b|P98 Air|\bX89HD\b|P98 3G|\bP90HD\b|P89 3G|X98 3G|\bP70h\b|P79HD 3G|G18d 3G|\bP79HD\b|\bP89s\b|\bA88\b|\bP10HD\b|\bP19HD\b|G18 3G|\bP78HD\b|\bA78\b|\bP75\b|G17s 3G|G17h 3G|\bP85t\b|\bP90\b|\bP11\b|\bP98t\b|\bP98HD\b|\bG18d\b|\bP85s\b|\bP11HD\b|\bP88s\b|\bA80HD\b|\bA80se\b|\bA10h\b|\bP89\b|\bP78s\b|\bG18\b|\bP85\b|\bA70h\b|\bA70\b|\bG17\b|\bP18\b|\bA80s\b|\bA11s\b|\bP88HD\b|\bA80h\b|\bP76s\b|\bP76h\b|\bP98\b|\bA10HD\b|\bP78\b|\bP88\b|\bA11\b|\bA10t\b|\bP76a\b|\bP76t\b|\bP76e\b|\bP85HD\b|\bP85a\b|\bP86\b|\bP75HD\b|\bP76v\b|\bA12\b|\bP75a\b|\bA15\b|\bP76Ti\b|\bP81HD\b|\bA10\b|\bT760VE\b|\bT720HD\b|\bP76\b|\bP73\b|\bP71\b|\bP72\b|\bT720SE\b|\bC520Ti\b|\bT760\b|\bT720VE\b|T720-3GE|T720-WiFi',
  4577. // Onda - http://www.onda-tablet.com/buy-android-onda.html?dir=desc&limit=all&order=price
  4578. 'OndaTablet' => '\b(V975i|Vi30|VX530|V701|Vi60|V701s|Vi50|V801s|V719|Vx610w|VX610W|V819i|Vi10|VX580W|Vi10|V711s|V813|V811|V820w|V820|Vi20|V711|VI30W|V712|V891w|V972|V819w|V820w|Vi60|V820w|V711|V813s|V801|V819|V975s|V801|V819|V819|V818|V811|V712|V975m|V101w|V961w|V812|V818|V971|V971s|V919|V989|V116w|V102w|V973|Vi40)\b[\s]+',
  4579. 'JaytechTablet' => 'TPC-PA762',
  4580. 'BlaupunktTablet' => 'Endeavour 800NG|Endeavour 1010',
  4581. // http://www.digma.ru/support/download/
  4582. // @todo: Ebooks also (if requested)
  4583. 'DigmaTablet' => '\b(iDx10|iDx9|iDx8|iDx7|iDxD7|iDxD8|iDsQ8|iDsQ7|iDsQ8|iDsD10|iDnD7|3TS804H|iDsQ11|iDj7|iDs10)\b',
  4584. // http://www.evolioshop.com/ro/tablete-pc.html
  4585. // http://www.evolio.ro/support/downloads_static.html?cat=2
  4586. // @todo: Research some more
  4587. 'EvolioTablet' => 'ARIA_Mini_wifi|Aria[ _]Mini|Evolio X10|Evolio X7|Evolio X8|\bEvotab\b|\bNeura\b',
  4588. // @todo http://www.lavamobiles.com/tablets-data-cards
  4589. 'LavaTablet' => 'QPAD E704|\bIvoryS\b|E-TAB IVORY|\bE-TAB\b',
  4590. // http://www.breezetablet.com/
  4591. 'AocTablet' => 'MW0811|MW0812|MW0922|MTK8382|MW1031|MW0831|MW0821|MW0931|MW0712',
  4592. // http://www.mpmaneurope.com/en/products/internet-tablets-14/android-tablets-14/
  4593. 'MpmanTablet' => 'MP11 OCTA|MP10 OCTA|MPQC1114|MPQC1004|MPQC994|MPQC974|MPQC973|MPQC804|MPQC784|MPQC780|\bMPG7\b|MPDCG75|MPDCG71|MPDC1006|MP101DC|MPDC9000|MPDC905|MPDC706HD|MPDC706|MPDC705|MPDC110|MPDC100|MPDC99|MPDC97|MPDC88|MPDC8|MPDC77|MP709|MID701|MID711|MID170|MPDC703|MPQC1010',
  4594. // https://www.celkonmobiles.com/?_a=categoryphones&sid=2
  4595. 'CelkonTablet' => 'CT695|CT888|CT[\s]?910|CT7 Tab|CT9 Tab|CT3 Tab|CT2 Tab|CT1 Tab|C820|C720|\bCT-1\b',
  4596. // http://www.wolderelectronics.com/productos/manuales-y-guias-rapidas/categoria-2-miTab
  4597. 'WolderTablet' => 'miTab \b(DIAMOND|SPACE|BROOKLYN|NEO|FLY|MANHATTAN|FUNK|EVOLUTION|SKY|GOCAR|IRON|GENIUS|POP|MINT|EPSILON|BROADWAY|JUMP|HOP|LEGEND|NEW AGE|LINE|ADVANCE|FEEL|FOLLOW|LIKE|LINK|LIVE|THINK|FREEDOM|CHICAGO|CLEVELAND|BALTIMORE-GH|IOWA|BOSTON|SEATTLE|PHOENIX|DALLAS|IN 101|MasterChef)\b',
  4598. // http://www.mi.com/en
  4599. 'MiTablet' => '\bMI PAD\b|\bHM NOTE 1W\b',
  4600. // http://www.nbru.cn/index.html
  4601. 'NibiruTablet' => 'Nibiru M1|Nibiru Jupiter One',
  4602. // http://navroad.com/products/produkty/tablety/
  4603. // http://navroad.com/products/produkty/tablety/
  4604. 'NexoTablet' => 'NEXO NOVA|NEXO 10|NEXO AVIO|NEXO FREE|NEXO GO|NEXO EVO|NEXO 3G|NEXO SMART|NEXO KIDDO|NEXO MOBI',
  4605. // http://leader-online.com/new_site/product-category/tablets/
  4606. // http://www.leader-online.net.au/List/Tablet
  4607. 'LeaderTablet' => 'TBLT10Q|TBLT10I|TBL-10WDKB|TBL-10WDKBO2013|TBL-W230V2|TBL-W450|TBL-W500|SV572|TBLT7I|TBA-AC7-8G|TBLT79|TBL-8W16|TBL-10W32|TBL-10WKB|TBL-W100',
  4608. // http://www.datawind.com/ubislate/
  4609. 'UbislateTablet' => 'UbiSlate[\s]?7C',
  4610. // http://www.pocketbook-int.com/ru/support
  4611. 'PocketBookTablet' => 'Pocketbook',
  4612. // http://www.kocaso.com/product_tablet.html
  4613. 'KocasoTablet' => '\b(TB-1207)\b',
  4614. // http://global.hisense.com/product/asia/tablet/Sero7/201412/t20141215_91832.htm
  4615. 'HisenseTablet' => '\b(F5281|E2371)\b',
  4616. // http://www.tesco.com/direct/hudl/
  4617. 'Hudl' => 'Hudl HT7S3|Hudl 2',
  4618. // http://www.telstra.com.au/home-phone/thub-2/
  4619. 'TelstraTablet' => 'T-Hub2',
  4620. 'GenericTablet' => 'Android.*\b97D\b|Tablet(?!.*PC)|BNTV250A|MID-WCDMA|LogicPD Zoom2|\bA7EB\b|CatNova8|A1_07|CT704|CT1002|\bM721\b|rk30sdk|\bEVOTAB\b|M758A|ET904|ALUMIUM10|Smartfren Tab|Endeavour 1010|Tablet-PC-4|Tagi Tab|\bM6pro\b|CT1020W|arc 10HD|\bTP750\b|\bQTAQZ3\b'
  4621. );
  4622. /**
  4623. * List of mobile Operating Systems.
  4624. *
  4625. * @var array
  4626. */
  4627. protected static $operatingSystems = array(
  4628. 'AndroidOS' => 'Android',
  4629. 'BlackBerryOS' => 'blackberry|\bBB10\b|rim tablet os',
  4630. 'PalmOS' => 'PalmOS|avantgo|blazer|elaine|hiptop|palm|plucker|xiino',
  4631. 'SymbianOS' => 'Symbian|SymbOS|Series60|Series40|SYB-[0-9]+|\bS60\b',
  4632. // @reference: http://en.wikipedia.org/wiki/Windows_Mobile
  4633. 'WindowsMobileOS' => 'Windows CE.*(PPC|Smartphone|Mobile|[0-9]{3}x[0-9]{3})|Window Mobile|Windows Phone [0-9.]+|WCE;',
  4634. // @reference: http://en.wikipedia.org/wiki/Windows_Phone
  4635. // http://wifeng.cn/?r=blog&a=view&id=106
  4636. // http://nicksnettravels.builttoroam.com/post/2011/01/10/Bogus-Windows-Phone-7-User-Agent-String.aspx
  4637. // http://msdn.microsoft.com/library/ms537503.aspx
  4638. // https://msdn.microsoft.com/en-us/library/hh869301(v=vs.85).aspx
  4639. 'WindowsPhoneOS' => 'Windows Phone 10.0|Windows Phone 8.1|Windows Phone 8.0|Windows Phone OS|XBLWP7|ZuneWP7|Windows NT 6.[23]; ARM;',
  4640. 'iOS' => '\biPhone.*Mobile|\biPod|\biPad',
  4641. // http://en.wikipedia.org/wiki/MeeGo
  4642. // @todo: research MeeGo in UAs
  4643. 'MeeGoOS' => 'MeeGo',
  4644. // http://en.wikipedia.org/wiki/Maemo
  4645. // @todo: research Maemo in UAs
  4646. 'MaemoOS' => 'Maemo',
  4647. 'JavaOS' => 'J2ME/|\bMIDP\b|\bCLDC\b', // '|Java/' produces bug #135
  4648. 'webOS' => 'webOS|hpwOS',
  4649. 'badaOS' => '\bBada\b',
  4650. 'BREWOS' => 'BREW',
  4651. );
  4652. /**
  4653. * List of mobile User Agents.
  4654. *
  4655. * IMPORTANT: This is a list of only mobile browsers.
  4656. * Mobile Detect 2.x supports only mobile browsers,
  4657. * it was never designed to detect all browsers.
  4658. * The change will come in 2017 in the 3.x release for PHP7.
  4659. *
  4660. * @var array
  4661. */
  4662. protected static $browsers = array(
  4663. //'Vivaldi' => 'Vivaldi',
  4664. // @reference: https://developers.google.com/chrome/mobile/docs/user-agent
  4665. 'Chrome' => '\bCrMo\b|CriOS|Android.*Chrome/[.0-9]* (Mobile)?',
  4666. 'Dolfin' => '\bDolfin\b',
  4667. 'Opera' => 'Opera.*Mini|Opera.*Mobi|Android.*Opera|Mobile.*OPR/[0-9.]+|Coast/[0-9.]+',
  4668. 'Skyfire' => 'Skyfire',
  4669. 'Edge' => 'Mobile Safari/[.0-9]* Edge',
  4670. 'IE' => 'IEMobile|MSIEMobile', // |Trident/[.0-9]+
  4671. 'Firefox' => 'fennec|firefox.*maemo|(Mobile|Tablet).*Firefox|Firefox.*Mobile|FxiOS',
  4672. 'Bolt' => 'bolt',
  4673. 'TeaShark' => 'teashark',
  4674. 'Blazer' => 'Blazer',
  4675. // @reference: http://developer.apple.com/library/safari/#documentation/AppleApplications/Reference/SafariWebContent/OptimizingforSafarioniPhone/OptimizingforSafarioniPhone.html#//apple_ref/doc/uid/TP40006517-SW3
  4676. 'Safari' => 'Version.*Mobile.*Safari|Safari.*Mobile|MobileSafari',
  4677. // http://en.wikipedia.org/wiki/Midori_(web_browser)
  4678. //'Midori' => 'midori',
  4679. //'Tizen' => 'Tizen',
  4680. 'UCBrowser' => 'UC.*Browser|UCWEB',
  4681. 'baiduboxapp' => 'baiduboxapp',
  4682. 'baidubrowser' => 'baidubrowser',
  4683. // https://github.com/serbanghita/Mobile-Detect/issues/7
  4684. 'DiigoBrowser' => 'DiigoBrowser',
  4685. // http://www.puffinbrowser.com/index.php
  4686. 'Puffin' => 'Puffin',
  4687. // http://mercury-browser.com/index.html
  4688. 'Mercury' => '\bMercury\b',
  4689. // http://en.wikipedia.org/wiki/Obigo_Browser
  4690. 'ObigoBrowser' => 'Obigo',
  4691. // http://en.wikipedia.org/wiki/NetFront
  4692. 'NetFront' => 'NF-Browser',
  4693. // @reference: http://en.wikipedia.org/wiki/Minimo
  4694. // http://en.wikipedia.org/wiki/Vision_Mobile_Browser
  4695. 'GenericBrowser' => 'NokiaBrowser|OviBrowser|OneBrowser|TwonkyBeamBrowser|SEMC.*Browser|FlyFlow|Minimo|NetFront|Novarra-Vision|MQQBrowser|MicroMessenger',
  4696. // @reference: https://en.wikipedia.org/wiki/Pale_Moon_(web_browser)
  4697. 'PaleMoon' => 'Android.*PaleMoon|Mobile.*PaleMoon',
  4698. );
  4699. /**
  4700. * Utilities.
  4701. *
  4702. * @var array
  4703. */
  4704. protected static $utilities = array(
  4705. // Experimental. When a mobile device wants to switch to 'Desktop Mode'.
  4706. // http://scottcate.com/technology/windows-phone-8-ie10-desktop-or-mobile/
  4707. // https://github.com/serbanghita/Mobile-Detect/issues/57#issuecomment-15024011
  4708. // https://developers.facebook.com/docs/sharing/best-practices
  4709. 'Bot' => 'Googlebot|facebookexternalhit|AdsBot-Google|Google Keyword Suggestion|Facebot|YandexBot|YandexMobileBot|bingbot|ia_archiver|AhrefsBot|Ezooms|GSLFbot|WBSearchBot|Twitterbot|TweetmemeBot|Twikle|PaperLiBot|Wotbox|UnwindFetchor|Exabot|MJ12bot|YandexImages|TurnitinBot|Pingdom',
  4710. 'MobileBot' => 'Googlebot-Mobile|AdsBot-Google-Mobile|YahooSeeker/M1A1-R2D2',
  4711. 'DesktopMode' => 'WPDesktop',
  4712. 'TV' => 'SonyDTV|HbbTV', // experimental
  4713. 'WebKit' => '(webkit)[ /]([\w.]+)',
  4714. // @todo: Include JXD consoles.
  4715. 'Console' => '\b(Nintendo|Nintendo WiiU|Nintendo 3DS|PLAYSTATION|Xbox)\b',
  4716. 'Watch' => 'SM-V700',
  4717. );
  4718. /**
  4719. * All possible HTTP headers that represent the
  4720. * User-Agent string.
  4721. *
  4722. * @var array
  4723. */
  4724. protected static $uaHttpHeaders = array(
  4725. // The default User-Agent string.
  4726. 'HTTP_USER_AGENT',
  4727. // Header can occur on devices using Opera Mini.
  4728. 'HTTP_X_OPERAMINI_PHONE_UA',
  4729. // Vodafone specific header: http://www.seoprinciple.com/mobile-web-community-still-angry-at-vodafone/24/
  4730. 'HTTP_X_DEVICE_USER_AGENT',
  4731. 'HTTP_X_ORIGINAL_USER_AGENT',
  4732. 'HTTP_X_SKYFIRE_PHONE',
  4733. 'HTTP_X_BOLT_PHONE_UA',
  4734. 'HTTP_DEVICE_STOCK_UA',
  4735. 'HTTP_X_UCBROWSER_DEVICE_UA'
  4736. );
  4737. /**
  4738. * The individual segments that could exist in a User-Agent string. VER refers to the regular
  4739. * expression defined in the constant self::VER.
  4740. *
  4741. * @var array
  4742. */
  4743. protected static $properties = array(
  4744. // Build
  4745. 'Mobile' => 'Mobile/[VER]',
  4746. 'Build' => 'Build/[VER]',
  4747. 'Version' => 'Version/[VER]',
  4748. 'VendorID' => 'VendorID/[VER]',
  4749. // Devices
  4750. 'iPad' => 'iPad.*CPU[a-z ]+[VER]',
  4751. 'iPhone' => 'iPhone.*CPU[a-z ]+[VER]',
  4752. 'iPod' => 'iPod.*CPU[a-z ]+[VER]',
  4753. //'BlackBerry' => array('BlackBerry[VER]', 'BlackBerry [VER];'),
  4754. 'Kindle' => 'Kindle/[VER]',
  4755. // Browser
  4756. 'Chrome' => array('Chrome/[VER]', 'CriOS/[VER]', 'CrMo/[VER]'),
  4757. 'Coast' => array('Coast/[VER]'),
  4758. 'Dolfin' => 'Dolfin/[VER]',
  4759. // @reference: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/User-Agent/Firefox
  4760. 'Firefox' => array('Firefox/[VER]', 'FxiOS/[VER]'),
  4761. 'Fennec' => 'Fennec/[VER]',
  4762. // http://msdn.microsoft.com/en-us/library/ms537503(v=vs.85).aspx
  4763. // https://msdn.microsoft.com/en-us/library/ie/hh869301(v=vs.85).aspx
  4764. 'Edge' => 'Edge/[VER]',
  4765. 'IE' => array('IEMobile/[VER];', 'IEMobile [VER]', 'MSIE [VER];', 'Trident/[0-9.]+;.*rv:[VER]'),
  4766. // http://en.wikipedia.org/wiki/NetFront
  4767. 'NetFront' => 'NetFront/[VER]',
  4768. 'NokiaBrowser' => 'NokiaBrowser/[VER]',
  4769. 'Opera' => array( ' OPR/[VER]', 'Opera Mini/[VER]', 'Version/[VER]' ),
  4770. 'Opera Mini' => 'Opera Mini/[VER]',
  4771. 'Opera Mobi' => 'Version/[VER]',
  4772. 'UC Browser' => 'UC Browser[VER]',
  4773. 'MQQBrowser' => 'MQQBrowser/[VER]',
  4774. 'MicroMessenger' => 'MicroMessenger/[VER]',
  4775. 'baiduboxapp' => 'baiduboxapp/[VER]',
  4776. 'baidubrowser' => 'baidubrowser/[VER]',
  4777. 'SamsungBrowser' => 'SamsungBrowser/[VER]',
  4778. 'Iron' => 'Iron/[VER]',
  4779. // @note: Safari 7534.48.3 is actually Version 5.1.
  4780. // @note: On BlackBerry the Version is overwriten by the OS.
  4781. 'Safari' => array( 'Version/[VER]', 'Safari/[VER]' ),
  4782. 'Skyfire' => 'Skyfire/[VER]',
  4783. 'Tizen' => 'Tizen/[VER]',
  4784. 'Webkit' => 'webkit[ /][VER]',
  4785. 'PaleMoon' => 'PaleMoon/[VER]',
  4786. // Engine
  4787. 'Gecko' => 'Gecko/[VER]',
  4788. 'Trident' => 'Trident/[VER]',
  4789. 'Presto' => 'Presto/[VER]',
  4790. 'Goanna' => 'Goanna/[VER]',
  4791. // OS
  4792. 'iOS' => ' \bi?OS\b [VER][ ;]{1}',
  4793. 'Android' => 'Android [VER]',
  4794. 'BlackBerry' => array('BlackBerry[\w]+/[VER]', 'BlackBerry.*Version/[VER]', 'Version/[VER]'),
  4795. 'BREW' => 'BREW [VER]',
  4796. 'Java' => 'Java/[VER]',
  4797. // @reference: http://windowsteamblog.com/windows_phone/b/wpdev/archive/2011/08/29/introducing-the-ie9-on-windows-phone-mango-user-agent-string.aspx
  4798. // @reference: http://en.wikipedia.org/wiki/Windows_NT#Releases
  4799. 'Windows Phone OS' => array( 'Windows Phone OS [VER]', 'Windows Phone [VER]'),
  4800. 'Windows Phone' => 'Windows Phone [VER]',
  4801. 'Windows CE' => 'Windows CE/[VER]',
  4802. // http://social.msdn.microsoft.com/Forums/en-US/windowsdeveloperpreviewgeneral/thread/6be392da-4d2f-41b4-8354-8dcee20c85cd
  4803. 'Windows NT' => 'Windows NT [VER]',
  4804. 'Symbian' => array('SymbianOS/[VER]', 'Symbian/[VER]'),
  4805. 'webOS' => array('webOS/[VER]', 'hpwOS/[VER];'),
  4806. );
  4807. /**
  4808. * Construct an instance of this class.
  4809. *
  4810. * @param array $headers Specify the headers as injection. Should be PHP _SERVER flavored.
  4811. * If left empty, will use the global _SERVER['HTTP_*'] vars instead.
  4812. * @param string $userAgent Inject the User-Agent header. If null, will use HTTP_USER_AGENT
  4813. * from the $headers array instead.
  4814. */
  4815. public function __construct(
  4816. array $headers = null,
  4817. $userAgent = null
  4818. ) {
  4819. $this->setHttpHeaders($headers);
  4820. $this->setUserAgent($userAgent);
  4821. }
  4822. /**
  4823. * Get the current script version.
  4824. * This is useful for the demo.php file,
  4825. * so people can check on what version they are testing
  4826. * for mobile devices.
  4827. *
  4828. * @return string The version number in semantic version format.
  4829. */
  4830. public static function getScriptVersion()
  4831. {
  4832. return self::VERSION;
  4833. }
  4834. /**
  4835. * Set the HTTP Headers. Must be PHP-flavored. This method will reset existing headers.
  4836. *
  4837. * @param array $httpHeaders The headers to set. If null, then using PHP's _SERVER to extract
  4838. * the headers. The default null is left for backwards compatibility.
  4839. */
  4840. public function setHttpHeaders($httpHeaders = null)
  4841. {
  4842. // use global _SERVER if $httpHeaders aren't defined
  4843. if (!is_array($httpHeaders) || !count($httpHeaders)) {
  4844. $httpHeaders = $_SERVER;
  4845. }
  4846. // clear existing headers
  4847. $this->httpHeaders = array();
  4848. // Only save HTTP headers. In PHP land, that means only _SERVER vars that
  4849. // start with HTTP_.
  4850. foreach ($httpHeaders as $key => $value) {
  4851. if (substr($key, 0, 5) === 'HTTP_') {
  4852. $this->httpHeaders[$key] = $value;
  4853. }
  4854. }
  4855. // In case we're dealing with CloudFront, we need to know.
  4856. $this->setCfHeaders($httpHeaders);
  4857. }
  4858. /**
  4859. * Retrieves the HTTP headers.
  4860. *
  4861. * @return array
  4862. */
  4863. public function getHttpHeaders()
  4864. {
  4865. return $this->httpHeaders;
  4866. }
  4867. /**
  4868. * Retrieves a particular header. If it doesn't exist, no exception/error is caused.
  4869. * Simply null is returned.
  4870. *
  4871. * @param string $header The name of the header to retrieve. Can be HTTP compliant such as
  4872. * "User-Agent" or "X-Device-User-Agent" or can be php-esque with the
  4873. * all-caps, HTTP_ prefixed, underscore seperated awesomeness.
  4874. *
  4875. * @return string|null The value of the header.
  4876. */
  4877. public function getHttpHeader($header)
  4878. {
  4879. // are we using PHP-flavored headers?
  4880. if (strpos($header, '_') === false) {
  4881. $header = str_replace('-', '_', $header);
  4882. $header = strtoupper($header);
  4883. }
  4884. // test the alternate, too
  4885. $altHeader = 'HTTP_' . $header;
  4886. //Test both the regular and the HTTP_ prefix
  4887. if (isset($this->httpHeaders[$header])) {
  4888. return $this->httpHeaders[$header];
  4889. } elseif (isset($this->httpHeaders[$altHeader])) {
  4890. return $this->httpHeaders[$altHeader];
  4891. }
  4892. return null;
  4893. }
  4894. public function getMobileHeaders()
  4895. {
  4896. return self::$mobileHeaders;
  4897. }
  4898. /**
  4899. * Get all possible HTTP headers that
  4900. * can contain the User-Agent string.
  4901. *
  4902. * @return array List of HTTP headers.
  4903. */
  4904. public function getUaHttpHeaders()
  4905. {
  4906. return self::$uaHttpHeaders;
  4907. }
  4908. /**
  4909. * Set CloudFront headers
  4910. * http://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/header-caching.html#header-caching-web-device
  4911. *
  4912. * @param array $cfHeaders List of HTTP headers
  4913. *
  4914. * @return boolean If there were CloudFront headers to be set
  4915. */
  4916. public function setCfHeaders($cfHeaders = null) {
  4917. // use global _SERVER if $cfHeaders aren't defined
  4918. if (!is_array($cfHeaders) || !count($cfHeaders)) {
  4919. $cfHeaders = $_SERVER;
  4920. }
  4921. // clear existing headers
  4922. $this->cloudfrontHeaders = array();
  4923. // Only save CLOUDFRONT headers. In PHP land, that means only _SERVER vars that
  4924. // start with cloudfront-.
  4925. $response = false;
  4926. foreach ($cfHeaders as $key => $value) {
  4927. if (substr(strtolower($key), 0, 16) === 'http_cloudfront_') {
  4928. $this->cloudfrontHeaders[strtoupper($key)] = $value;
  4929. $response = true;
  4930. }
  4931. }
  4932. return $response;
  4933. }
  4934. /**
  4935. * Retrieves the cloudfront headers.
  4936. *
  4937. * @return array
  4938. */
  4939. public function getCfHeaders()
  4940. {
  4941. return $this->cloudfrontHeaders;
  4942. }
  4943. /**
  4944. * Set the User-Agent to be used.
  4945. *
  4946. * @param string $userAgent The user agent string to set.
  4947. *
  4948. * @return string|null
  4949. */
  4950. public function setUserAgent($userAgent = null)
  4951. {
  4952. // Invalidate cache due to #375
  4953. $this->cache = array();
  4954. if (false === empty($userAgent)) {
  4955. return $this->userAgent = $userAgent;
  4956. } else {
  4957. $this->userAgent = null;
  4958. foreach ($this->getUaHttpHeaders() as $altHeader) {
  4959. if (false === empty($this->httpHeaders[$altHeader])) { // @todo: should use getHttpHeader(), but it would be slow. (Serban)
  4960. $this->userAgent .= $this->httpHeaders[$altHeader] . " ";
  4961. }
  4962. }
  4963. if (!empty($this->userAgent)) {
  4964. return $this->userAgent = trim($this->userAgent);
  4965. }
  4966. }
  4967. if (count($this->getCfHeaders()) > 0) {
  4968. return $this->userAgent = 'Amazon CloudFront';
  4969. }
  4970. return $this->userAgent = null;
  4971. }
  4972. /**
  4973. * Retrieve the User-Agent.
  4974. *
  4975. * @return string|null The user agent if it's set.
  4976. */
  4977. public function getUserAgent()
  4978. {
  4979. return $this->userAgent;
  4980. }
  4981. /**
  4982. * Set the detection type. Must be one of self::DETECTION_TYPE_MOBILE or
  4983. * self::DETECTION_TYPE_EXTENDED. Otherwise, nothing is set.
  4984. *
  4985. * @deprecated since version 2.6.9
  4986. *
  4987. * @param string $type The type. Must be a self::DETECTION_TYPE_* constant. The default
  4988. * parameter is null which will default to self::DETECTION_TYPE_MOBILE.
  4989. */
  4990. public function setDetectionType($type = null)
  4991. {
  4992. if ($type === null) {
  4993. $type = self::DETECTION_TYPE_MOBILE;
  4994. }
  4995. if ($type !== self::DETECTION_TYPE_MOBILE && $type !== self::DETECTION_TYPE_EXTENDED) {
  4996. return;
  4997. }
  4998. $this->detectionType = $type;
  4999. }
  5000. public function getMatchingRegex()
  5001. {
  5002. return $this->matchingRegex;
  5003. }
  5004. public function getMatchesArray()
  5005. {
  5006. return $this->matchesArray;
  5007. }
  5008. /**
  5009. * Retrieve the list of known phone devices.
  5010. *
  5011. * @return array List of phone devices.
  5012. */
  5013. public static function getPhoneDevices()
  5014. {
  5015. return self::$phoneDevices;
  5016. }
  5017. /**
  5018. * Retrieve the list of known tablet devices.
  5019. *
  5020. * @return array List of tablet devices.
  5021. */
  5022. public static function getTabletDevices()
  5023. {
  5024. return self::$tabletDevices;
  5025. }
  5026. /**
  5027. * Alias for getBrowsers() method.
  5028. *
  5029. * @return array List of user agents.
  5030. */
  5031. public static function getUserAgents()
  5032. {
  5033. return self::getBrowsers();
  5034. }
  5035. /**
  5036. * Retrieve the list of known browsers. Specifically, the user agents.
  5037. *
  5038. * @return array List of browsers / user agents.
  5039. */
  5040. public static function getBrowsers()
  5041. {
  5042. return self::$browsers;
  5043. }
  5044. /**
  5045. * Retrieve the list of known utilities.
  5046. *
  5047. * @return array List of utilities.
  5048. */
  5049. public static function getUtilities()
  5050. {
  5051. return self::$utilities;
  5052. }
  5053. /**
  5054. * Method gets the mobile detection rules. This method is used for the magic methods $detect->is*().
  5055. *
  5056. * @deprecated since version 2.6.9
  5057. *
  5058. * @return array All the rules (but not extended).
  5059. */
  5060. public static function getMobileDetectionRules()
  5061. {
  5062. static $rules;
  5063. if (!$rules) {
  5064. $rules = array_merge(
  5065. self::$phoneDevices,
  5066. self::$tabletDevices,
  5067. self::$operatingSystems,
  5068. self::$browsers
  5069. );
  5070. }
  5071. return $rules;
  5072. }
  5073. /**
  5074. * Method gets the mobile detection rules + utilities.
  5075. * The reason this is separate is because utilities rules
  5076. * don't necessary imply mobile. This method is used inside
  5077. * the new $detect->is('stuff') method.
  5078. *
  5079. * @deprecated since version 2.6.9
  5080. *
  5081. * @return array All the rules + extended.
  5082. */
  5083. public function getMobileDetectionRulesExtended()
  5084. {
  5085. static $rules;
  5086. if (!$rules) {
  5087. // Merge all rules together.
  5088. $rules = array_merge(
  5089. self::$phoneDevices,
  5090. self::$tabletDevices,
  5091. self::$operatingSystems,
  5092. self::$browsers,
  5093. self::$utilities
  5094. );
  5095. }
  5096. return $rules;
  5097. }
  5098. /**
  5099. * Retrieve the current set of rules.
  5100. *
  5101. * @deprecated since version 2.6.9
  5102. *
  5103. * @return array
  5104. */
  5105. public function getRules()
  5106. {
  5107. if ($this->detectionType == self::DETECTION_TYPE_EXTENDED) {
  5108. return self::getMobileDetectionRulesExtended();
  5109. } else {
  5110. return self::getMobileDetectionRules();
  5111. }
  5112. }
  5113. /**
  5114. * Retrieve the list of mobile operating systems.
  5115. *
  5116. * @return array The list of mobile operating systems.
  5117. */
  5118. public static function getOperatingSystems()
  5119. {
  5120. return self::$operatingSystems;
  5121. }
  5122. /**
  5123. * Check the HTTP headers for signs of mobile.
  5124. * This is the fastest mobile check possible; it's used
  5125. * inside isMobile() method.
  5126. *
  5127. * @return bool
  5128. */
  5129. public function checkHttpHeadersForMobile()
  5130. {
  5131. foreach ($this->getMobileHeaders() as $mobileHeader => $matchType) {
  5132. if (isset($this->httpHeaders[$mobileHeader])) {
  5133. if (is_array($matchType['matches'])) {
  5134. foreach ($matchType['matches'] as $_match) {
  5135. if (strpos($this->httpHeaders[$mobileHeader], $_match) !== false) {
  5136. return true;
  5137. }
  5138. }
  5139. return false;
  5140. } else {
  5141. return true;
  5142. }
  5143. }
  5144. }
  5145. return false;
  5146. }
  5147. /**
  5148. * Magic overloading method.
  5149. *
  5150. * @method boolean is[...]()
  5151. * @param string $name
  5152. * @param array $arguments
  5153. * @return mixed
  5154. * @throws BadMethodCallException when the method doesn't exist and doesn't start with 'is'
  5155. */
  5156. public function __call($name, $arguments)
  5157. {
  5158. // make sure the name starts with 'is', otherwise
  5159. if (substr($name, 0, 2) !== 'is') {
  5160. throw new BadMethodCallException("No such method exists: $name");
  5161. }
  5162. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5163. $key = substr($name, 2);
  5164. return $this->matchUAAgainstKey($key);
  5165. }
  5166. /**
  5167. * Find a detection rule that matches the current User-agent.
  5168. *
  5169. * @param null $userAgent deprecated
  5170. * @return boolean
  5171. */
  5172. protected function matchDetectionRulesAgainstUA($userAgent = null)
  5173. {
  5174. // Begin general search.
  5175. foreach ($this->getRules() as $_regex) {
  5176. if (empty($_regex)) {
  5177. continue;
  5178. }
  5179. if ($this->match($_regex, $userAgent)) {
  5180. return true;
  5181. }
  5182. }
  5183. return false;
  5184. }
  5185. /**
  5186. * Search for a certain key in the rules array.
  5187. * If the key is found then try to match the corresponding
  5188. * regex against the User-Agent.
  5189. *
  5190. * @param string $key
  5191. *
  5192. * @return boolean
  5193. */
  5194. protected function matchUAAgainstKey($key)
  5195. {
  5196. // Make the keys lowercase so we can match: isIphone(), isiPhone(), isiphone(), etc.
  5197. $key = strtolower($key);
  5198. if (false === isset($this->cache[$key])) {
  5199. // change the keys to lower case
  5200. $_rules = array_change_key_case($this->getRules());
  5201. if (false === empty($_rules[$key])) {
  5202. $this->cache[$key] = $this->match($_rules[$key]);
  5203. }
  5204. if (false === isset($this->cache[$key])) {
  5205. $this->cache[$key] = false;
  5206. }
  5207. }
  5208. return $this->cache[$key];
  5209. }
  5210. /**
  5211. * Check if the device is mobile.
  5212. * Returns true if any type of mobile device detected, including special ones
  5213. * @param null $userAgent deprecated
  5214. * @param null $httpHeaders deprecated
  5215. * @return bool
  5216. */
  5217. public function isMobile($userAgent = null, $httpHeaders = null)
  5218. {
  5219. if ($httpHeaders) {
  5220. $this->setHttpHeaders($httpHeaders);
  5221. }
  5222. if ($userAgent) {
  5223. $this->setUserAgent($userAgent);
  5224. }
  5225. // Check specifically for cloudfront headers if the useragent === 'Amazon CloudFront'
  5226. if ($this->getUserAgent() === 'Amazon CloudFront') {
  5227. $cfHeaders = $this->getCfHeaders();
  5228. if(array_key_exists('HTTP_CLOUDFRONT_IS_MOBILE_VIEWER', $cfHeaders) && $cfHeaders['HTTP_CLOUDFRONT_IS_MOBILE_VIEWER'] === 'true') {
  5229. return true;
  5230. }
  5231. }
  5232. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5233. if ($this->checkHttpHeadersForMobile()) {
  5234. return true;
  5235. } else {
  5236. return $this->matchDetectionRulesAgainstUA();
  5237. }
  5238. }
  5239. /**
  5240. * Check if the device is a tablet.
  5241. * Return true if any type of tablet device is detected.
  5242. *
  5243. * @param string $userAgent deprecated
  5244. * @param array $httpHeaders deprecated
  5245. * @return bool
  5246. */
  5247. public function isTablet($userAgent = null, $httpHeaders = null)
  5248. {
  5249. // Check specifically for cloudfront headers if the useragent === 'Amazon CloudFront'
  5250. if ($this->getUserAgent() === 'Amazon CloudFront') {
  5251. $cfHeaders = $this->getCfHeaders();
  5252. if(array_key_exists('HTTP_CLOUDFRONT_IS_TABLET_VIEWER', $cfHeaders) && $cfHeaders['HTTP_CLOUDFRONT_IS_TABLET_VIEWER'] === 'true') {
  5253. return true;
  5254. }
  5255. }
  5256. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5257. foreach (self::$tabletDevices as $_regex) {
  5258. if ($this->match($_regex, $userAgent)) {
  5259. return true;
  5260. }
  5261. }
  5262. return false;
  5263. }
  5264. /**
  5265. * This method checks for a certain property in the
  5266. * userAgent.
  5267. * @todo: The httpHeaders part is not yet used.
  5268. *
  5269. * @param string $key
  5270. * @param string $userAgent deprecated
  5271. * @param string $httpHeaders deprecated
  5272. * @return bool|int|null
  5273. */
  5274. public function is($key, $userAgent = null, $httpHeaders = null)
  5275. {
  5276. // Set the UA and HTTP headers only if needed (eg. batch mode).
  5277. if ($httpHeaders) {
  5278. $this->setHttpHeaders($httpHeaders);
  5279. }
  5280. if ($userAgent) {
  5281. $this->setUserAgent($userAgent);
  5282. }
  5283. $this->setDetectionType(self::DETECTION_TYPE_EXTENDED);
  5284. return $this->matchUAAgainstKey($key);
  5285. }
  5286. /**
  5287. * Some detection rules are relative (not standard),
  5288. * because of the diversity of devices, vendors and
  5289. * their conventions in representing the User-Agent or
  5290. * the HTTP headers.
  5291. *
  5292. * This method will be used to check custom regexes against
  5293. * the User-Agent string.
  5294. *
  5295. * @param $regex
  5296. * @param string $userAgent
  5297. * @return bool
  5298. *
  5299. * @todo: search in the HTTP headers too.
  5300. */
  5301. public function match($regex, $userAgent = null)
  5302. {
  5303. $match = (bool) preg_match(sprintf('#%s#is', $regex), (false === empty($userAgent) ? $userAgent : $this->userAgent), $matches);
  5304. // If positive match is found, store the results for debug.
  5305. if ($match) {
  5306. $this->matchingRegex = $regex;
  5307. $this->matchesArray = $matches;
  5308. }
  5309. return $match;
  5310. }
  5311. /**
  5312. * Get the properties array.
  5313. *
  5314. * @return array
  5315. */
  5316. public static function getProperties()
  5317. {
  5318. return self::$properties;
  5319. }
  5320. /**
  5321. * Prepare the version number.
  5322. *
  5323. * @todo Remove the error supression from str_replace() call.
  5324. *
  5325. * @param string $ver The string version, like "2.6.21.2152";
  5326. *
  5327. * @return float
  5328. */
  5329. public function prepareVersionNo($ver)
  5330. {
  5331. $ver = str_replace(array('_', ' ', '/'), '.', $ver);
  5332. $arrVer = explode('.', $ver, 2);
  5333. if (isset($arrVer[1])) {
  5334. $arrVer[1] = @str_replace('.', '', $arrVer[1]); // @todo: treat strings versions.
  5335. }
  5336. return (float) implode('.', $arrVer);
  5337. }
  5338. /**
  5339. * Check the version of the given property in the User-Agent.
  5340. * Will return a float number. (eg. 2_0 will return 2.0, 4.3.1 will return 4.31)
  5341. *
  5342. * @param string $propertyName The name of the property. See self::getProperties() array
  5343. * keys for all possible properties.
  5344. * @param string $type Either self::VERSION_TYPE_STRING to get a string value or
  5345. * self::VERSION_TYPE_FLOAT indicating a float value. This parameter
  5346. * is optional and defaults to self::VERSION_TYPE_STRING. Passing an
  5347. * invalid parameter will default to the this type as well.
  5348. *
  5349. * @return string|float The version of the property we are trying to extract.
  5350. */
  5351. public function version($propertyName, $type = self::VERSION_TYPE_STRING)
  5352. {
  5353. if (empty($propertyName)) {
  5354. return false;
  5355. }
  5356. // set the $type to the default if we don't recognize the type
  5357. if ($type !== self::VERSION_TYPE_STRING && $type !== self::VERSION_TYPE_FLOAT) {
  5358. $type = self::VERSION_TYPE_STRING;
  5359. }
  5360. $properties = self::getProperties();
  5361. // Check if the property exists in the properties array.
  5362. if (true === isset($properties[$propertyName])) {
  5363. // Prepare the pattern to be matched.
  5364. // Make sure we always deal with an array (string is converted).
  5365. $properties[$propertyName] = (array) $properties[$propertyName];
  5366. foreach ($properties[$propertyName] as $propertyMatchString) {
  5367. $propertyPattern = str_replace('[VER]', self::VER, $propertyMatchString);
  5368. // Identify and extract the version.
  5369. preg_match(sprintf('#%s#is', $propertyPattern), $this->userAgent, $match);
  5370. if (false === empty($match[1])) {
  5371. $version = ($type == self::VERSION_TYPE_FLOAT ? $this->prepareVersionNo($match[1]) : $match[1]);
  5372. return $version;
  5373. }
  5374. }
  5375. }
  5376. return false;
  5377. }
  5378. /**
  5379. * Retrieve the mobile grading, using self::MOBILE_GRADE_* constants.
  5380. *
  5381. * @return string One of the self::MOBILE_GRADE_* constants.
  5382. */
  5383. public function mobileGrade()
  5384. {
  5385. $isMobile = $this->isMobile();
  5386. if (
  5387. // Apple iOS 4-7.0 – Tested on the original iPad (4.3 / 5.0), iPad 2 (4.3 / 5.1 / 6.1), iPad 3 (5.1 / 6.0), iPad Mini (6.1), iPad Retina (7.0), iPhone 3GS (4.3), iPhone 4 (4.3 / 5.1), iPhone 4S (5.1 / 6.0), iPhone 5 (6.0), and iPhone 5S (7.0)
  5388. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5389. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5390. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5391. // Android 2.1-2.3 - Tested on the HTC Incredible (2.2), original Droid (2.2), HTC Aria (2.1), Google Nexus S (2.3). Functional on 1.5 & 1.6 but performance may be sluggish, tested on Google G1 (1.5)
  5392. // Android 3.1 (Honeycomb) - Tested on the Samsung Galaxy Tab 10.1 and Motorola XOOM
  5393. // Android 4.0 (ICS) - Tested on a Galaxy Nexus. Note: transition performance can be poor on upgraded devices
  5394. // Android 4.1 (Jelly Bean) - Tested on a Galaxy Nexus and Galaxy 7
  5395. ( $this->version('Android', self::VERSION_TYPE_FLOAT)>2.1 && $this->is('Webkit') ) ||
  5396. // Windows Phone 7.5-8 - Tested on the HTC Surround (7.5), HTC Trophy (7.5), LG-E900 (7.5), Nokia 800 (7.8), HTC Mazaa (7.8), Nokia Lumia 520 (8), Nokia Lumia 920 (8), HTC 8x (8)
  5397. $this->version('Windows Phone OS', self::VERSION_TYPE_FLOAT) >= 7.5 ||
  5398. // Tested on the Torch 9800 (6) and Style 9670 (6), BlackBerry® Torch 9810 (7), BlackBerry Z10 (10)
  5399. $this->is('BlackBerry') && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) >= 6.0 ||
  5400. // Blackberry Playbook (1.0-2.0) - Tested on PlayBook
  5401. $this->match('Playbook.*Tablet') ||
  5402. // Palm WebOS (1.4-3.0) - Tested on the Palm Pixi (1.4), Pre (1.4), Pre 2 (2.0), HP TouchPad (3.0)
  5403. ( $this->version('webOS', self::VERSION_TYPE_FLOAT) >= 1.4 && $this->match('Palm|Pre|Pixi') ) ||
  5404. // Palm WebOS 3.0 - Tested on HP TouchPad
  5405. $this->match('hp.*TouchPad') ||
  5406. // Firefox Mobile 18 - Tested on Android 2.3 and 4.1 devices
  5407. ( $this->is('Firefox') && $this->version('Firefox', self::VERSION_TYPE_FLOAT) >= 18 ) ||
  5408. // Chrome for Android - Tested on Android 4.0, 4.1 device
  5409. ( $this->is('Chrome') && $this->is('AndroidOS') && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 4.0 ) ||
  5410. // Skyfire 4.1 - Tested on Android 2.3 device
  5411. ( $this->is('Skyfire') && $this->version('Skyfire', self::VERSION_TYPE_FLOAT) >= 4.1 && $this->is('AndroidOS') && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 ) ||
  5412. // Opera Mobile 11.5-12: Tested on Android 2.3
  5413. ( $this->is('Opera') && $this->version('Opera Mobi', self::VERSION_TYPE_FLOAT) >= 11.5 && $this->is('AndroidOS') ) ||
  5414. // Meego 1.2 - Tested on Nokia 950 and N9
  5415. $this->is('MeeGoOS') ||
  5416. // Tizen (pre-release) - Tested on early hardware
  5417. $this->is('Tizen') ||
  5418. // Samsung Bada 2.0 - Tested on a Samsung Wave 3, Dolphin browser
  5419. // @todo: more tests here!
  5420. $this->is('Dolfin') && $this->version('Bada', self::VERSION_TYPE_FLOAT) >= 2.0 ||
  5421. // UC Browser - Tested on Android 2.3 device
  5422. ( ($this->is('UC Browser') || $this->is('Dolfin')) && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 ) ||
  5423. // Kindle 3 and Fire - Tested on the built-in WebKit browser for each
  5424. ( $this->match('Kindle Fire') ||
  5425. $this->is('Kindle') && $this->version('Kindle', self::VERSION_TYPE_FLOAT) >= 3.0 ) ||
  5426. // Nook Color 1.4.1 - Tested on original Nook Color, not Nook Tablet
  5427. $this->is('AndroidOS') && $this->is('NookTablet') ||
  5428. // Chrome Desktop 16-24 - Tested on OS X 10.7 and Windows 7
  5429. $this->version('Chrome', self::VERSION_TYPE_FLOAT) >= 16 && !$isMobile ||
  5430. // Safari Desktop 5-6 - Tested on OS X 10.7 and Windows 7
  5431. $this->version('Safari', self::VERSION_TYPE_FLOAT) >= 5.0 && !$isMobile ||
  5432. // Firefox Desktop 10-18 - Tested on OS X 10.7 and Windows 7
  5433. $this->version('Firefox', self::VERSION_TYPE_FLOAT) >= 10.0 && !$isMobile ||
  5434. // Internet Explorer 7-9 - Tested on Windows XP, Vista and 7
  5435. $this->version('IE', self::VERSION_TYPE_FLOAT) >= 7.0 && !$isMobile ||
  5436. // Opera Desktop 10-12 - Tested on OS X 10.7 and Windows 7
  5437. $this->version('Opera', self::VERSION_TYPE_FLOAT) >= 10 && !$isMobile
  5438. ){
  5439. return self::MOBILE_GRADE_A;
  5440. }
  5441. if (
  5442. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT)<4.3 ||
  5443. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT)<4.3 ||
  5444. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT)<4.3 ||
  5445. // Blackberry 5.0: Tested on the Storm 2 9550, Bold 9770
  5446. $this->is('Blackberry') && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) >= 5 && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT)<6 ||
  5447. //Opera Mini (5.0-6.5) - Tested on iOS 3.2/4.3 and Android 2.3
  5448. ($this->version('Opera Mini', self::VERSION_TYPE_FLOAT) >= 5.0 && $this->version('Opera Mini', self::VERSION_TYPE_FLOAT) <= 7.0 &&
  5449. ($this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 || $this->is('iOS')) ) ||
  5450. // Nokia Symbian^3 - Tested on Nokia N8 (Symbian^3), C7 (Symbian^3), also works on N97 (Symbian^1)
  5451. $this->match('NokiaN8|NokiaC7|N97.*Series60|Symbian/3') ||
  5452. // @todo: report this (tested on Nokia N71)
  5453. $this->version('Opera Mobi', self::VERSION_TYPE_FLOAT) >= 11 && $this->is('SymbianOS')
  5454. ){
  5455. return self::MOBILE_GRADE_B;
  5456. }
  5457. if (
  5458. // Blackberry 4.x - Tested on the Curve 8330
  5459. $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) <= 5.0 ||
  5460. // Windows Mobile - Tested on the HTC Leo (WinMo 5.2)
  5461. $this->match('MSIEMobile|Windows CE.*Mobile') || $this->version('Windows Mobile', self::VERSION_TYPE_FLOAT) <= 5.2 ||
  5462. // Tested on original iPhone (3.1), iPhone 3 (3.2)
  5463. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5464. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5465. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5466. // Internet Explorer 7 and older - Tested on Windows XP
  5467. $this->version('IE', self::VERSION_TYPE_FLOAT) <= 7.0 && !$isMobile
  5468. ){
  5469. return self::MOBILE_GRADE_C;
  5470. }
  5471. // All older smartphone platforms and featurephones - Any device that doesn't support media queries
  5472. // will receive the basic, C grade experience.
  5473. return self::MOBILE_GRADE_C;
  5474. }
  5475. }
  5476. $mobileDetect = new Mobile_Detect;
  5477. $userDevice = ($mobileDetect->isMobile() ? ($mobileDetect->isTablet() ? 'tablet' : 'phone') : 'computer');
  5478. function orgEmail($header = "Message From Admin", $title = "Important Message", $user = "Organizr User", $mainMessage = "", $button = null, $buttonURL = null, $subTitle = "", $subMessage = ""){
  5479. $path = getServerPath();
  5480. return '
  5481. <!DOCTYPE HTML PUBLIC "-//W3C//DTD XHTML 1.0 Transitional //EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  5482. <html xmlns="http://www.w3.org/1999/xhtml" xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office">
  5483. <head>
  5484. <!--[if gte mso 9]><xml>
  5485. <o:OfficeDocumentSettings>
  5486. <o:AllowPNG/>
  5487. <o:PixelsPerInch>96</o:PixelsPerInch>
  5488. </o:OfficeDocumentSettings>
  5489. </xml><![endif]-->
  5490. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  5491. <meta name="viewport" content="width=device-width">
  5492. <!--[if !mso]><!-->
  5493. <meta http-equiv="X-UA-Compatible" content="IE=edge">
  5494. <!--<![endif]-->
  5495. <title></title>
  5496. <!--[if !mso]><!-- -->
  5497. <link href="https://fonts.googleapis.com/css?family=Ubuntu" rel="stylesheet" type="text/css">
  5498. <link href="https://fonts.googleapis.com/css?family=Lato" rel="stylesheet" type="text/css">
  5499. <!--<![endif]-->
  5500. <style type="text/css" id="media-query">
  5501. body {
  5502. margin: 0;
  5503. padding: 0;
  5504. }
  5505. table,
  5506. tr,
  5507. td {
  5508. vertical-align: top;
  5509. border-collapse: collapse;
  5510. }
  5511. .ie-browser table,
  5512. .mso-container table {
  5513. table-layout: fixed;
  5514. }
  5515. * {
  5516. line-height: inherit;
  5517. }
  5518. a[x-apple-data-detectors=true] {
  5519. color: inherit !important;
  5520. text-decoration: none !important;
  5521. }
  5522. [owa] .img-container div,
  5523. [owa] .img-container button {
  5524. display: block !important;
  5525. }
  5526. [owa] .fullwidth button {
  5527. width: 100% !important;
  5528. }
  5529. [owa] .block-grid .col {
  5530. display: table-cell;
  5531. float: none !important;
  5532. vertical-align: top;
  5533. }
  5534. .ie-browser .num12,
  5535. .ie-browser .block-grid,
  5536. [owa] .num12,
  5537. [owa] .block-grid {
  5538. width: 615px !important;
  5539. }
  5540. .ExternalClass,
  5541. .ExternalClass p,
  5542. .ExternalClass span,
  5543. .ExternalClass font,
  5544. .ExternalClass td,
  5545. .ExternalClass div {
  5546. line-height: 100%;
  5547. }
  5548. .ie-browser .mixed-two-up .num4,
  5549. [owa] .mixed-two-up .num4 {
  5550. width: 204px !important;
  5551. }
  5552. .ie-browser .mixed-two-up .num8,
  5553. [owa] .mixed-two-up .num8 {
  5554. width: 408px !important;
  5555. }
  5556. .ie-browser .block-grid.two-up .col,
  5557. [owa] .block-grid.two-up .col {
  5558. width: 307px !important;
  5559. }
  5560. .ie-browser .block-grid.three-up .col,
  5561. [owa] .block-grid.three-up .col {
  5562. width: 205px !important;
  5563. }
  5564. .ie-browser .block-grid.four-up .col,
  5565. [owa] .block-grid.four-up .col {
  5566. width: 153px !important;
  5567. }
  5568. .ie-browser .block-grid.five-up .col,
  5569. [owa] .block-grid.five-up .col {
  5570. width: 123px !important;
  5571. }
  5572. .ie-browser .block-grid.six-up .col,
  5573. [owa] .block-grid.six-up .col {
  5574. width: 102px !important;
  5575. }
  5576. .ie-browser .block-grid.seven-up .col,
  5577. [owa] .block-grid.seven-up .col {
  5578. width: 87px !important;
  5579. }
  5580. .ie-browser .block-grid.eight-up .col,
  5581. [owa] .block-grid.eight-up .col {
  5582. width: 76px !important;
  5583. }
  5584. .ie-browser .block-grid.nine-up .col,
  5585. [owa] .block-grid.nine-up .col {
  5586. width: 68px !important;
  5587. }
  5588. .ie-browser .block-grid.ten-up .col,
  5589. [owa] .block-grid.ten-up .col {
  5590. width: 61px !important;
  5591. }
  5592. .ie-browser .block-grid.eleven-up .col,
  5593. [owa] .block-grid.eleven-up .col {
  5594. width: 55px !important;
  5595. }
  5596. .ie-browser .block-grid.twelve-up .col,
  5597. [owa] .block-grid.twelve-up .col {
  5598. width: 51px !important;
  5599. }
  5600. @media only screen and (min-width: 635px) {
  5601. .block-grid {
  5602. width: 615px !important;
  5603. }
  5604. .block-grid .col {
  5605. display: table-cell;
  5606. Float: none !important;
  5607. vertical-align: top;
  5608. }
  5609. .block-grid .col.num12 {
  5610. width: 615px !important;
  5611. }
  5612. .block-grid.mixed-two-up .col.num4 {
  5613. width: 204px !important;
  5614. }
  5615. .block-grid.mixed-two-up .col.num8 {
  5616. width: 408px !important;
  5617. }
  5618. .block-grid.two-up .col {
  5619. width: 307px !important;
  5620. }
  5621. .block-grid.three-up .col {
  5622. width: 205px !important;
  5623. }
  5624. .block-grid.four-up .col {
  5625. width: 153px !important;
  5626. }
  5627. .block-grid.five-up .col {
  5628. width: 123px !important;
  5629. }
  5630. .block-grid.six-up .col {
  5631. width: 102px !important;
  5632. }
  5633. .block-grid.seven-up .col {
  5634. width: 87px !important;
  5635. }
  5636. .block-grid.eight-up .col {
  5637. width: 76px !important;
  5638. }
  5639. .block-grid.nine-up .col {
  5640. width: 68px !important;
  5641. }
  5642. .block-grid.ten-up .col {
  5643. width: 61px !important;
  5644. }
  5645. .block-grid.eleven-up .col {
  5646. width: 55px !important;
  5647. }
  5648. .block-grid.twelve-up .col {
  5649. width: 51px !important;
  5650. }
  5651. }
  5652. @media (max-width: 635px) {
  5653. .block-grid,
  5654. .col {
  5655. min-width: 320px !important;
  5656. max-width: 100% !important;
  5657. }
  5658. .block-grid {
  5659. width: calc(100% - 40px) !important;
  5660. }
  5661. .col {
  5662. width: 100% !important;
  5663. }
  5664. .col>div {
  5665. margin: 0 auto;
  5666. }
  5667. img.fullwidth {
  5668. max-width: 100% !important;
  5669. }
  5670. }
  5671. </style>
  5672. </head>
  5673. <body class="clean-body" style="margin: 0;padding: 0;-webkit-text-size-adjust: 100%;background-color: #FFFFFF">
  5674. <!--[if IE]><div class="ie-browser"><![endif]-->
  5675. <!--[if mso]><div class="mso-container"><![endif]-->
  5676. <div class="nl-container" style="min-width: 320px;Margin: 0 auto;background-color: #FFFFFF">
  5677. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td align="center" style="background-color: #FFFFFF;"><![endif]-->
  5678. <div style="background-color:#333333;">
  5679. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5680. class="block-grid ">
  5681. <div style="border-collapse: collapse;display: table;width: 100%;">
  5682. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5683. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:0px; padding-bottom:0px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5684. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5685. <div style="background-color: transparent; width: 100% !important;">
  5686. <!--[if (!mso)&(!IE)]><!-->
  5687. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:0px; padding-bottom:0px; padding-right: 0px; padding-left: 0px;">
  5688. <!--<![endif]-->
  5689. <div align="left" class="img-container left fullwidth" style="padding-right: 30px; padding-left: 30px;">
  5690. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px;" align="left"><![endif]-->
  5691. <img class="left fullwidth" align="left" border="0" src="'.$path.'images/organizr-logo-h.png" alt="Image" title="Image"
  5692. style="outline: none;text-decoration: none;-ms-interpolation-mode: bicubic;clear: both;display: block !important;border: 0;height: auto;float: none;width: 100%;max-width: 555px"
  5693. width="555">
  5694. <!--[if mso]></td></tr></table><![endif]-->
  5695. </div>
  5696. <!--[if (!mso)&(!IE)]><!-->
  5697. </div>
  5698. <!--<![endif]-->
  5699. </div>
  5700. </div>
  5701. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5702. </div>
  5703. </div>
  5704. </div>
  5705. <div style="background-color:#333333;">
  5706. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5707. class="block-grid ">
  5708. <div style="border-collapse: collapse;display: table;width: 100%;">
  5709. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5710. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:0px; padding-bottom:0px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5711. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5712. <div style="background-color: transparent; width: 100% !important;">
  5713. <!--[if (!mso)&(!IE)]><!-->
  5714. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:0px; padding-bottom:0px; padding-right: 0px; padding-left: 0px;">
  5715. <!--<![endif]-->
  5716. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 0px; padding-left: 0px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5717. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 0px; padding-left: 0px; padding-top: 0px; padding-bottom: 0px;">
  5718. <div style="font-size:12px;line-height:14px;color:#FFFFFF;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5719. <p style="margin: 0;font-size: 12px;line-height: 14px;text-align: center"><span style="font-size: 16px; line-height: 19px;"><strong><span style="line-height: 19px; font-size: 16px;">'.$header.'</span></strong>
  5720. </span>
  5721. </p>
  5722. </div>
  5723. </div>
  5724. <!--[if mso]></td></tr></table><![endif]-->
  5725. <!--[if (!mso)&(!IE)]><!-->
  5726. </div>
  5727. <!--<![endif]-->
  5728. </div>
  5729. </div>
  5730. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5731. </div>
  5732. </div>
  5733. </div>
  5734. <div style="background-color:#393939;">
  5735. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5736. class="block-grid ">
  5737. <div style="border-collapse: collapse;display: table;width: 100%;">
  5738. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#393939;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5739. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:5px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5740. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5741. <div style="background-color: transparent; width: 100% !important;">
  5742. <!--[if (!mso)&(!IE)]><!-->
  5743. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:5px; padding-right: 0px; padding-left: 0px;">
  5744. <!--<![endif]-->
  5745. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5746. <div style="font-family:\'Ubuntu\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 30px; padding-left: 30px; padding-top: 0px; padding-bottom: 0px;">
  5747. <div style="font-family:Ubuntu, Tahoma, Verdana, Segoe, sans-serif;font-size:12px;line-height:14px;color:#FFFFFF;text-align:left;">
  5748. <p style="margin: 0;font-size: 12px;line-height: 14px;text-align: center"><span style="font-size: 16px; line-height: 19px;"><strong>'.$title.'</strong></span></p>
  5749. </div>
  5750. </div>
  5751. <!--[if mso]></td></tr></table><![endif]-->
  5752. <div style="padding-right: 5px; padding-left: 5px; padding-top: 5px; padding-bottom: 5px;">
  5753. <!--[if (mso)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 5px;padding-left: 5px; padding-top: 5px; padding-bottom: 5px;"><table width="55%" align="center" cellpadding="0" cellspacing="0" border="0"><tr><td><![endif]-->
  5754. <div align="center">
  5755. <div style="border-top: 2px solid #66D9EF; width:55%; line-height:2px; height:2px; font-size:2px;">&#160;</div>
  5756. </div>
  5757. <!--[if (mso)]></td></tr></table></td></tr></table><![endif]-->
  5758. </div>
  5759. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top: 15px; padding-bottom: 10px;"><![endif]-->
  5760. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 30px; padding-left: 30px; padding-top: 15px; padding-bottom: 10px;">
  5761. <div style="font-family:\'Lato\',Tahoma,Verdana,Segoe,sans-serif;font-size:12px;line-height:14px;color:#FFFFFF;text-align:left;">
  5762. <p style="margin: 0;font-size: 12px;line-height: 14px"><span style="font-size: 28px; line-height: 33px;">Hey '.$user.',</span></p>
  5763. </div>
  5764. </div>
  5765. <!--[if mso]></td></tr></table><![endif]-->
  5766. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 15px; padding-left: 30px; padding-top: 10px; padding-bottom: 25px;"><![endif]-->
  5767. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:180%;color:#FFFFFF; padding-right: 15px; padding-left: 30px; padding-top: 10px; padding-bottom: 25px;">
  5768. <div style="font-size:12px;line-height:22px;font-family:\'Lato\',Tahoma,Verdana,Segoe,sans-serif;color:#FFFFFF;text-align:left;">
  5769. <p style="margin: 0;font-size: 14px;line-height: 25px"><span style="font-size: 18px; line-height: 32px;"><em><span style="line-height: 32px; font-size: 18px;">'.$mainMessage.'</span></em>
  5770. </span>
  5771. </p>
  5772. </div>
  5773. </div>
  5774. <!--[if mso]></td></tr></table><![endif]-->
  5775. <div align="center" class="button-container center" style="padding-right: 30px; padding-left: 30px; padding-top:15px; padding-bottom:15px;">
  5776. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0" style="border-spacing: 0; border-collapse: collapse; mso-table-lspace:0pt; mso-table-rspace:0pt;"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top:15px; padding-bottom:15px;" align="center"><v:roundrect xmlns:v="urn:schemas-microsoft-com:vml" xmlns:w="urn:schemas-microsoft-com:office:word" href="'.$path.'" style="height:48px; v-text-anchor:middle; width:194px;" arcsize="53%" strokecolor="" fillcolor="#66D9EF"><w:anchorlock/><center style="color:#000; font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif; font-size:18px;"><![endif]-->
  5777. <a href="'.$buttonURL.'" target="_blank" style="display: inline-block;text-decoration: none;-webkit-text-size-adjust: none;text-align: center;color: #000; background-color: #66D9EF; border-radius: 25px; -webkit-border-radius: 25px; -moz-border-radius: 25px; max-width: 180px; width: 114px; width: auto; border-top: 3px solid transparent; border-right: 3px solid transparent; border-bottom: 3px solid transparent; border-left: 3px solid transparent; padding-top: 5px; padding-right: 30px; padding-bottom: 5px; padding-left: 30px; font-family: \'Lato\', Tahoma, Verdana, Segoe, sans-serif;mso-border-alt: none">
  5778. <span style="font-size:12px;line-height:21px;"><span style="font-size: 18px; line-height: 32px;" data-mce-style="font-size: 18px; line-height: 44px;">'.$button.'</span></span></a>
  5779. <!--[if mso]></center></v:roundrect></td></tr></table><![endif]-->
  5780. </div>
  5781. <!--[if mso]></center></v:roundrect></td></tr></table><![endif]-->
  5782. </div>
  5783. <!--[if (!mso)&(!IE)]><!-->
  5784. </div>
  5785. <!--<![endif]-->
  5786. </div>
  5787. </div>
  5788. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5789. </div>
  5790. </div>
  5791. </div>
  5792. <div style="background-color:#ffffff;">
  5793. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5794. class="block-grid ">
  5795. <div style="border-collapse: collapse;display: table;width: 100%;">
  5796. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#ffffff;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5797. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:30px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5798. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5799. <div style="background-color: transparent; width: 100% !important;">
  5800. <!--[if (!mso)&(!IE)]><!-->
  5801. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:30px; padding-right: 0px; padding-left: 0px;">
  5802. <!--<![endif]-->
  5803. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 10px;"><![endif]-->
  5804. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#555555; padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 10px;">
  5805. <div style="font-size:12px;line-height:14px;color:#555555;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5806. <p style="margin: 0;font-size: 14px;line-height: 17px;text-align: center"><strong><span style="font-size: 26px; line-height: 31px;">'.$subTitle.'<br></span></strong></p>
  5807. </div>
  5808. </div>
  5809. <!--[if mso]></td></tr></table><![endif]-->
  5810. <div style="padding-right: 20px; padding-left: 20px; padding-top: 15px; padding-bottom: 20px;">
  5811. <!--[if (mso)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 20px;padding-left: 20px; padding-top: 15px; padding-bottom: 20px;"><table width="40%" align="center" cellpadding="0" cellspacing="0" border="0"><tr><td><![endif]-->
  5812. <div align="center">
  5813. <div style="border-top: 3px solid #66D9EF; width:40%; line-height:3px; height:3px; font-size:3px;">&#160;</div>
  5814. </div>
  5815. <!--[if (mso)]></td></tr></table></td></tr></table><![endif]-->
  5816. </div>
  5817. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5818. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:180%;color:#7E7D7D; padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 0px;">
  5819. <div style="font-size:12px;line-height:22px;color:#7E7D7D;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5820. <p style="margin: 0;font-size: 14px;line-height: 25px;text-align: center"><em><span style="font-size: 18px; line-height: 32px;">'.$subMessage.'</span></em></p>
  5821. </div>
  5822. </div>
  5823. <!--[if mso]></td></tr></table><![endif]-->
  5824. <!--[if (!mso)&(!IE)]><!-->
  5825. </div>
  5826. <!--<![endif]-->
  5827. </div>
  5828. </div>
  5829. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5830. </div>
  5831. </div>
  5832. </div>
  5833. <div style="background-color:#333333;">
  5834. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5835. class="block-grid ">
  5836. <div style="border-collapse: collapse;display: table;width: 100%;">
  5837. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5838. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:5px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5839. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5840. <div style="background-color: transparent; width: 100% !important;">
  5841. <!--[if (!mso)&(!IE)]><!-->
  5842. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:5px; padding-right: 0px; padding-left: 0px;">
  5843. <!--<![endif]-->
  5844. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 10px; padding-bottom: 10px;"><![endif]-->
  5845. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#959595; padding-right: 10px; padding-left: 10px; padding-top: 10px; padding-bottom: 10px;">
  5846. <div style="font-size:12px;line-height:14px;color:#959595;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5847. <p style="margin: 0;font-size: 14px;line-height: 17px;text-align: center">This&#160;email was sent by <a style="color:#AD80FD;text-decoration: underline;" title="Organizr"
  5848. href="https://github.com/causefx/Organizr" target="_blank" rel="noopener noreferrer">Organizr</a><strong><br></strong></p>
  5849. </div>
  5850. </div>
  5851. <!--[if mso]></td></tr></table><![endif]-->
  5852. <!--[if (!mso)&(!IE)]><!-->
  5853. </div>
  5854. <!--<![endif]-->
  5855. </div>
  5856. </div>
  5857. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5858. </div>
  5859. </div>
  5860. </div>
  5861. <!--[if (mso)|(IE)]></td></tr></table><![endif]-->
  5862. </div>
  5863. <!--[if (mso)|(IE)]></div><![endif]-->
  5864. </body>
  5865. </html>
  5866. ';
  5867. }
  5868. function mimeTypes(){
  5869. return array(
  5870. '123' => 'application/vnd.lotus-1-2-3',
  5871. '3dml' => 'text/vnd.in3d.3dml',
  5872. '3ds' => 'image/x-3ds',
  5873. '3g2' => 'video/3gpp2',
  5874. '3gp' => 'video/3gpp',
  5875. '7z' => 'application/x-7z-compressed',
  5876. 'aab' => 'application/x-authorware-bin',
  5877. 'aac' => 'audio/x-aac',
  5878. 'aam' => 'application/x-authorware-map',
  5879. 'aas' => 'application/x-authorware-seg',
  5880. 'abw' => 'application/x-abiword',
  5881. 'ac' => 'application/pkix-attr-cert',
  5882. 'acc' => 'application/vnd.americandynamics.acc',
  5883. 'ace' => 'application/x-ace-compressed',
  5884. 'acu' => 'application/vnd.acucobol',
  5885. 'acutc' => 'application/vnd.acucorp',
  5886. 'adp' => 'audio/adpcm',
  5887. 'aep' => 'application/vnd.audiograph',
  5888. 'afm' => 'application/x-font-type1',
  5889. 'afp' => 'application/vnd.ibm.modcap',
  5890. 'ahead' => 'application/vnd.ahead.space',
  5891. 'ai' => 'application/postscript',
  5892. 'aif' => 'audio/x-aiff',
  5893. 'aifc' => 'audio/x-aiff',
  5894. 'aiff' => 'audio/x-aiff',
  5895. 'air' => 'application/vnd.adobe.air-application-installer-package+zip',
  5896. 'ait' => 'application/vnd.dvb.ait',
  5897. 'ami' => 'application/vnd.amiga.ami',
  5898. 'apk' => 'application/vnd.android.package-archive',
  5899. 'appcache' => 'text/cache-manifest',
  5900. 'application' => 'application/x-ms-application',
  5901. 'apr' => 'application/vnd.lotus-approach',
  5902. 'arc' => 'application/x-freearc',
  5903. 'asc' => 'application/pgp-signature',
  5904. 'asf' => 'video/x-ms-asf',
  5905. 'asm' => 'text/x-asm',
  5906. 'aso' => 'application/vnd.accpac.simply.aso',
  5907. 'asx' => 'video/x-ms-asf',
  5908. 'atc' => 'application/vnd.acucorp',
  5909. 'atom' => 'application/atom+xml',
  5910. 'atomcat' => 'application/atomcat+xml',
  5911. 'atomsvc' => 'application/atomsvc+xml',
  5912. 'atx' => 'application/vnd.antix.game-component',
  5913. 'au' => 'audio/basic',
  5914. 'avi' => 'video/x-msvideo',
  5915. 'aw' => 'application/applixware',
  5916. 'azf' => 'application/vnd.airzip.filesecure.azf',
  5917. 'azs' => 'application/vnd.airzip.filesecure.azs',
  5918. 'azw' => 'application/vnd.amazon.ebook',
  5919. 'bat' => 'application/x-msdownload',
  5920. 'bcpio' => 'application/x-bcpio',
  5921. 'bdf' => 'application/x-font-bdf',
  5922. 'bdm' => 'application/vnd.syncml.dm+wbxml',
  5923. 'bed' => 'application/vnd.realvnc.bed',
  5924. 'bh2' => 'application/vnd.fujitsu.oasysprs',
  5925. 'bin' => 'application/octet-stream',
  5926. 'blb' => 'application/x-blorb',
  5927. 'blorb' => 'application/x-blorb',
  5928. 'bmi' => 'application/vnd.bmi',
  5929. 'bmp' => 'image/bmp',
  5930. 'book' => 'application/vnd.framemaker',
  5931. 'box' => 'application/vnd.previewsystems.box',
  5932. 'boz' => 'application/x-bzip2',
  5933. 'bpk' => 'application/octet-stream',
  5934. 'btif' => 'image/prs.btif',
  5935. 'bz' => 'application/x-bzip',
  5936. 'bz2' => 'application/x-bzip2',
  5937. 'c' => 'text/x-c',
  5938. 'c11amc' => 'application/vnd.cluetrust.cartomobile-config',
  5939. 'c11amz' => 'application/vnd.cluetrust.cartomobile-config-pkg',
  5940. 'c4d' => 'application/vnd.clonk.c4group',
  5941. 'c4f' => 'application/vnd.clonk.c4group',
  5942. 'c4g' => 'application/vnd.clonk.c4group',
  5943. 'c4p' => 'application/vnd.clonk.c4group',
  5944. 'c4u' => 'application/vnd.clonk.c4group',
  5945. 'cab' => 'application/vnd.ms-cab-compressed',
  5946. 'caf' => 'audio/x-caf',
  5947. 'cap' => 'application/vnd.tcpdump.pcap',
  5948. 'car' => 'application/vnd.curl.car',
  5949. 'cat' => 'application/vnd.ms-pki.seccat',
  5950. 'cb7' => 'application/x-cbr',
  5951. 'cba' => 'application/x-cbr',
  5952. 'cbr' => 'application/x-cbr',
  5953. 'cbt' => 'application/x-cbr',
  5954. 'cbz' => 'application/x-cbr',
  5955. 'cc' => 'text/x-c',
  5956. 'cct' => 'application/x-director',
  5957. 'ccxml' => 'application/ccxml+xml',
  5958. 'cdbcmsg' => 'application/vnd.contact.cmsg',
  5959. 'cdf' => 'application/x-netcdf',
  5960. 'cdkey' => 'application/vnd.mediastation.cdkey',
  5961. 'cdmia' => 'application/cdmi-capability',
  5962. 'cdmic' => 'application/cdmi-container',
  5963. 'cdmid' => 'application/cdmi-domain',
  5964. 'cdmio' => 'application/cdmi-object',
  5965. 'cdmiq' => 'application/cdmi-queue',
  5966. 'cdx' => 'chemical/x-cdx',
  5967. 'cdxml' => 'application/vnd.chemdraw+xml',
  5968. 'cdy' => 'application/vnd.cinderella',
  5969. 'cer' => 'application/pkix-cert',
  5970. 'cfs' => 'application/x-cfs-compressed',
  5971. 'cgm' => 'image/cgm',
  5972. 'chat' => 'application/x-chat',
  5973. 'chm' => 'application/vnd.ms-htmlhelp',
  5974. 'chrt' => 'application/vnd.kde.kchart',
  5975. 'cif' => 'chemical/x-cif',
  5976. 'cii' => 'application/vnd.anser-web-certificate-issue-initiation',
  5977. 'cil' => 'application/vnd.ms-artgalry',
  5978. 'cla' => 'application/vnd.claymore',
  5979. 'class' => 'application/java-vm',
  5980. 'clkk' => 'application/vnd.crick.clicker.keyboard',
  5981. 'clkp' => 'application/vnd.crick.clicker.palette',
  5982. 'clkt' => 'application/vnd.crick.clicker.template',
  5983. 'clkw' => 'application/vnd.crick.clicker.wordbank',
  5984. 'clkx' => 'application/vnd.crick.clicker',
  5985. 'clp' => 'application/x-msclip',
  5986. 'cmc' => 'application/vnd.cosmocaller',
  5987. 'cmdf' => 'chemical/x-cmdf',
  5988. 'cml' => 'chemical/x-cml',
  5989. 'cmp' => 'application/vnd.yellowriver-custom-menu',
  5990. 'cmx' => 'image/x-cmx',
  5991. 'cod' => 'application/vnd.rim.cod',
  5992. 'com' => 'application/x-msdownload',
  5993. 'conf' => 'text/plain',
  5994. 'cpio' => 'application/x-cpio',
  5995. 'cpp' => 'text/x-c',
  5996. 'cpt' => 'application/mac-compactpro',
  5997. 'crd' => 'application/x-mscardfile',
  5998. 'crl' => 'application/pkix-crl',
  5999. 'crt' => 'application/x-x509-ca-cert',
  6000. 'cryptonote' => 'application/vnd.rig.cryptonote',
  6001. 'csh' => 'application/x-csh',
  6002. 'csml' => 'chemical/x-csml',
  6003. 'csp' => 'application/vnd.commonspace',
  6004. 'css' => 'text/css',
  6005. 'cst' => 'application/x-director',
  6006. 'csv' => 'text/csv',
  6007. 'cu' => 'application/cu-seeme',
  6008. 'curl' => 'text/vnd.curl',
  6009. 'cww' => 'application/prs.cww',
  6010. 'cxt' => 'application/x-director',
  6011. 'cxx' => 'text/x-c',
  6012. 'dae' => 'model/vnd.collada+xml',
  6013. 'daf' => 'application/vnd.mobius.daf',
  6014. 'dart' => 'application/vnd.dart',
  6015. 'dataless' => 'application/vnd.fdsn.seed',
  6016. 'davmount' => 'application/davmount+xml',
  6017. 'dbk' => 'application/docbook+xml',
  6018. 'dcr' => 'application/x-director',
  6019. 'dcurl' => 'text/vnd.curl.dcurl',
  6020. 'dd2' => 'application/vnd.oma.dd2+xml',
  6021. 'ddd' => 'application/vnd.fujixerox.ddd',
  6022. 'deb' => 'application/x-debian-package',
  6023. 'def' => 'text/plain',
  6024. 'deploy' => 'application/octet-stream',
  6025. 'der' => 'application/x-x509-ca-cert',
  6026. 'dfac' => 'application/vnd.dreamfactory',
  6027. 'dgc' => 'application/x-dgc-compressed',
  6028. 'dic' => 'text/x-c',
  6029. 'dir' => 'application/x-director',
  6030. 'dis' => 'application/vnd.mobius.dis',
  6031. 'dist' => 'application/octet-stream',
  6032. 'distz' => 'application/octet-stream',
  6033. 'djv' => 'image/vnd.djvu',
  6034. 'djvu' => 'image/vnd.djvu',
  6035. 'dll' => 'application/x-msdownload',
  6036. 'dmg' => 'application/x-apple-diskimage',
  6037. 'dmp' => 'application/vnd.tcpdump.pcap',
  6038. 'dms' => 'application/octet-stream',
  6039. 'dna' => 'application/vnd.dna',
  6040. 'doc' => 'application/msword',
  6041. 'docm' => 'application/vnd.ms-word.document.macroenabled.12',
  6042. 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
  6043. 'dot' => 'application/msword',
  6044. 'dotm' => 'application/vnd.ms-word.template.macroenabled.12',
  6045. 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template',
  6046. 'dp' => 'application/vnd.osgi.dp',
  6047. 'dpg' => 'application/vnd.dpgraph',
  6048. 'dra' => 'audio/vnd.dra',
  6049. 'dsc' => 'text/prs.lines.tag',
  6050. 'dssc' => 'application/dssc+der',
  6051. 'dtb' => 'application/x-dtbook+xml',
  6052. 'dtd' => 'application/xml-dtd',
  6053. 'dts' => 'audio/vnd.dts',
  6054. 'dtshd' => 'audio/vnd.dts.hd',
  6055. 'dump' => 'application/octet-stream',
  6056. 'dvb' => 'video/vnd.dvb.file',
  6057. 'dvi' => 'application/x-dvi',
  6058. 'dwf' => 'model/vnd.dwf',
  6059. 'dwg' => 'image/vnd.dwg',
  6060. 'dxf' => 'image/vnd.dxf',
  6061. 'dxp' => 'application/vnd.spotfire.dxp',
  6062. 'dxr' => 'application/x-director',
  6063. 'ecelp4800' => 'audio/vnd.nuera.ecelp4800',
  6064. 'ecelp7470' => 'audio/vnd.nuera.ecelp7470',
  6065. 'ecelp9600' => 'audio/vnd.nuera.ecelp9600',
  6066. 'ecma' => 'application/ecmascript',
  6067. 'edm' => 'application/vnd.novadigm.edm',
  6068. 'edx' => 'application/vnd.novadigm.edx',
  6069. 'efif' => 'application/vnd.picsel',
  6070. 'ei6' => 'application/vnd.pg.osasli',
  6071. 'elc' => 'application/octet-stream',
  6072. 'emf' => 'application/x-msmetafile',
  6073. 'eml' => 'message/rfc822',
  6074. 'emma' => 'application/emma+xml',
  6075. 'emz' => 'application/x-msmetafile',
  6076. 'eol' => 'audio/vnd.digital-winds',
  6077. 'eot' => 'application/vnd.ms-fontobject',
  6078. 'eps' => 'application/postscript',
  6079. 'epub' => 'application/epub+zip',
  6080. 'es3' => 'application/vnd.eszigno3+xml',
  6081. 'esa' => 'application/vnd.osgi.subsystem',
  6082. 'esf' => 'application/vnd.epson.esf',
  6083. 'et3' => 'application/vnd.eszigno3+xml',
  6084. 'etx' => 'text/x-setext',
  6085. 'eva' => 'application/x-eva',
  6086. 'evy' => 'application/x-envoy',
  6087. 'exe' => 'application/x-msdownload',
  6088. 'exi' => 'application/exi',
  6089. 'ext' => 'application/vnd.novadigm.ext',
  6090. 'ez' => 'application/andrew-inset',
  6091. 'ez2' => 'application/vnd.ezpix-album',
  6092. 'ez3' => 'application/vnd.ezpix-package',
  6093. 'f' => 'text/x-fortran',
  6094. 'f4v' => 'video/x-f4v',
  6095. 'f77' => 'text/x-fortran',
  6096. 'f90' => 'text/x-fortran',
  6097. 'fbs' => 'image/vnd.fastbidsheet',
  6098. 'fcdt' => 'application/vnd.adobe.formscentral.fcdt',
  6099. 'fcs' => 'application/vnd.isac.fcs',
  6100. 'fdf' => 'application/vnd.fdf',
  6101. 'fe_launch' => 'application/vnd.denovo.fcselayout-link',
  6102. 'fg5' => 'application/vnd.fujitsu.oasysgp',
  6103. 'fgd' => 'application/x-director',
  6104. 'fh' => 'image/x-freehand',
  6105. 'fh4' => 'image/x-freehand',
  6106. 'fh5' => 'image/x-freehand',
  6107. 'fh7' => 'image/x-freehand',
  6108. 'fhc' => 'image/x-freehand',
  6109. 'fig' => 'application/x-xfig',
  6110. 'flac' => 'audio/x-flac',
  6111. 'fli' => 'video/x-fli',
  6112. 'flo' => 'application/vnd.micrografx.flo',
  6113. 'flv' => 'video/x-flv',
  6114. 'flw' => 'application/vnd.kde.kivio',
  6115. 'flx' => 'text/vnd.fmi.flexstor',
  6116. 'fly' => 'text/vnd.fly',
  6117. 'fm' => 'application/vnd.framemaker',
  6118. 'fnc' => 'application/vnd.frogans.fnc',
  6119. 'for' => 'text/x-fortran',
  6120. 'fpx' => 'image/vnd.fpx',
  6121. 'frame' => 'application/vnd.framemaker',
  6122. 'fsc' => 'application/vnd.fsc.weblaunch',
  6123. 'fst' => 'image/vnd.fst',
  6124. 'ftc' => 'application/vnd.fluxtime.clip',
  6125. 'fti' => 'application/vnd.anser-web-funds-transfer-initiation',
  6126. 'fvt' => 'video/vnd.fvt',
  6127. 'fxp' => 'application/vnd.adobe.fxp',
  6128. 'fxpl' => 'application/vnd.adobe.fxp',
  6129. 'fzs' => 'application/vnd.fuzzysheet',
  6130. 'g2w' => 'application/vnd.geoplan',
  6131. 'g3' => 'image/g3fax',
  6132. 'g3w' => 'application/vnd.geospace',
  6133. 'gac' => 'application/vnd.groove-account',
  6134. 'gam' => 'application/x-tads',
  6135. 'gbr' => 'application/rpki-ghostbusters',
  6136. 'gca' => 'application/x-gca-compressed',
  6137. 'gdl' => 'model/vnd.gdl',
  6138. 'geo' => 'application/vnd.dynageo',
  6139. 'gex' => 'application/vnd.geometry-explorer',
  6140. 'ggb' => 'application/vnd.geogebra.file',
  6141. 'ggt' => 'application/vnd.geogebra.tool',
  6142. 'ghf' => 'application/vnd.groove-help',
  6143. 'gif' => 'image/gif',
  6144. 'gim' => 'application/vnd.groove-identity-message',
  6145. 'gml' => 'application/gml+xml',
  6146. 'gmx' => 'application/vnd.gmx',
  6147. 'gnumeric' => 'application/x-gnumeric',
  6148. 'gph' => 'application/vnd.flographit',
  6149. 'gpx' => 'application/gpx+xml',
  6150. 'gqf' => 'application/vnd.grafeq',
  6151. 'gqs' => 'application/vnd.grafeq',
  6152. 'gram' => 'application/srgs',
  6153. 'gramps' => 'application/x-gramps-xml',
  6154. 'gre' => 'application/vnd.geometry-explorer',
  6155. 'grv' => 'application/vnd.groove-injector',
  6156. 'grxml' => 'application/srgs+xml',
  6157. 'gsf' => 'application/x-font-ghostscript',
  6158. 'gtar' => 'application/x-gtar',
  6159. 'gtm' => 'application/vnd.groove-tool-message',
  6160. 'gtw' => 'model/vnd.gtw',
  6161. 'gv' => 'text/vnd.graphviz',
  6162. 'gxf' => 'application/gxf',
  6163. 'gxt' => 'application/vnd.geonext',
  6164. 'h' => 'text/x-c',
  6165. 'h261' => 'video/h261',
  6166. 'h263' => 'video/h263',
  6167. 'h264' => 'video/h264',
  6168. 'hal' => 'application/vnd.hal+xml',
  6169. 'hbci' => 'application/vnd.hbci',
  6170. 'hdf' => 'application/x-hdf',
  6171. 'hh' => 'text/x-c',
  6172. 'hlp' => 'application/winhlp',
  6173. 'hpgl' => 'application/vnd.hp-hpgl',
  6174. 'hpid' => 'application/vnd.hp-hpid',
  6175. 'hps' => 'application/vnd.hp-hps',
  6176. 'hqx' => 'application/mac-binhex40',
  6177. 'htke' => 'application/vnd.kenameaapp',
  6178. 'htm' => 'text/html',
  6179. 'html' => 'text/html',
  6180. 'hvd' => 'application/vnd.yamaha.hv-dic',
  6181. 'hvp' => 'application/vnd.yamaha.hv-voice',
  6182. 'hvs' => 'application/vnd.yamaha.hv-script',
  6183. 'i2g' => 'application/vnd.intergeo',
  6184. 'icc' => 'application/vnd.iccprofile',
  6185. 'ice' => 'x-conference/x-cooltalk',
  6186. 'icm' => 'application/vnd.iccprofile',
  6187. 'ico' => 'image/x-icon',
  6188. 'ics' => 'text/calendar',
  6189. 'ief' => 'image/ief',
  6190. 'ifb' => 'text/calendar',
  6191. 'ifm' => 'application/vnd.shana.informed.formdata',
  6192. 'iges' => 'model/iges',
  6193. 'igl' => 'application/vnd.igloader',
  6194. 'igm' => 'application/vnd.insors.igm',
  6195. 'igs' => 'model/iges',
  6196. 'igx' => 'application/vnd.micrografx.igx',
  6197. 'iif' => 'application/vnd.shana.informed.interchange',
  6198. 'imp' => 'application/vnd.accpac.simply.imp',
  6199. 'ims' => 'application/vnd.ms-ims',
  6200. 'in' => 'text/plain',
  6201. 'ink' => 'application/inkml+xml',
  6202. 'inkml' => 'application/inkml+xml',
  6203. 'install' => 'application/x-install-instructions',
  6204. 'iota' => 'application/vnd.astraea-software.iota',
  6205. 'ipfix' => 'application/ipfix',
  6206. 'ipk' => 'application/vnd.shana.informed.package',
  6207. 'irm' => 'application/vnd.ibm.rights-management',
  6208. 'irp' => 'application/vnd.irepository.package+xml',
  6209. 'iso' => 'application/x-iso9660-image',
  6210. 'itp' => 'application/vnd.shana.informed.formtemplate',
  6211. 'ivp' => 'application/vnd.immervision-ivp',
  6212. 'ivu' => 'application/vnd.immervision-ivu',
  6213. 'jad' => 'text/vnd.sun.j2me.app-descriptor',
  6214. 'jam' => 'application/vnd.jam',
  6215. 'jar' => 'application/java-archive',
  6216. 'java' => 'text/x-java-source',
  6217. 'jisp' => 'application/vnd.jisp',
  6218. 'jlt' => 'application/vnd.hp-jlyt',
  6219. 'jnlp' => 'application/x-java-jnlp-file',
  6220. 'joda' => 'application/vnd.joost.joda-archive',
  6221. 'jpe' => 'image/jpeg',
  6222. 'jpeg' => 'image/jpeg',
  6223. 'jpg' => 'image/jpeg',
  6224. 'jpgm' => 'video/jpm',
  6225. 'jpgv' => 'video/jpeg',
  6226. 'jpm' => 'video/jpm',
  6227. 'js' => 'application/javascript',
  6228. 'json' => 'application/json',
  6229. 'jsonml' => 'application/jsonml+json',
  6230. 'kar' => 'audio/midi',
  6231. 'karbon' => 'application/vnd.kde.karbon',
  6232. 'kfo' => 'application/vnd.kde.kformula',
  6233. 'kia' => 'application/vnd.kidspiration',
  6234. 'kml' => 'application/vnd.google-earth.kml+xml',
  6235. 'kmz' => 'application/vnd.google-earth.kmz',
  6236. 'kne' => 'application/vnd.kinar',
  6237. 'knp' => 'application/vnd.kinar',
  6238. 'kon' => 'application/vnd.kde.kontour',
  6239. 'kpr' => 'application/vnd.kde.kpresenter',
  6240. 'kpt' => 'application/vnd.kde.kpresenter',
  6241. 'kpxx' => 'application/vnd.ds-keypoint',
  6242. 'ksp' => 'application/vnd.kde.kspread',
  6243. 'ktr' => 'application/vnd.kahootz',
  6244. 'ktx' => 'image/ktx',
  6245. 'ktz' => 'application/vnd.kahootz',
  6246. 'kwd' => 'application/vnd.kde.kword',
  6247. 'kwt' => 'application/vnd.kde.kword',
  6248. 'lasxml' => 'application/vnd.las.las+xml',
  6249. 'latex' => 'application/x-latex',
  6250. 'lbd' => 'application/vnd.llamagraphics.life-balance.desktop',
  6251. 'lbe' => 'application/vnd.llamagraphics.life-balance.exchange+xml',
  6252. 'les' => 'application/vnd.hhe.lesson-player',
  6253. 'lha' => 'application/x-lzh-compressed',
  6254. 'link66' => 'application/vnd.route66.link66+xml',
  6255. 'list' => 'text/plain',
  6256. 'list3820' => 'application/vnd.ibm.modcap',
  6257. 'listafp' => 'application/vnd.ibm.modcap',
  6258. 'lnk' => 'application/x-ms-shortcut',
  6259. 'log' => 'text/plain',
  6260. 'lostxml' => 'application/lost+xml',
  6261. 'lrf' => 'application/octet-stream',
  6262. 'lrm' => 'application/vnd.ms-lrm',
  6263. 'ltf' => 'application/vnd.frogans.ltf',
  6264. 'lvp' => 'audio/vnd.lucent.voice',
  6265. 'lwp' => 'application/vnd.lotus-wordpro',
  6266. 'lzh' => 'application/x-lzh-compressed',
  6267. 'm13' => 'application/x-msmediaview',
  6268. 'm14' => 'application/x-msmediaview',
  6269. 'm1v' => 'video/mpeg',
  6270. 'm21' => 'application/mp21',
  6271. 'm2a' => 'audio/mpeg',
  6272. 'm2v' => 'video/mpeg',
  6273. 'm3a' => 'audio/mpeg',
  6274. 'm3u' => 'audio/x-mpegurl',
  6275. 'm3u8' => 'application/vnd.apple.mpegurl',
  6276. 'm4a' => 'audio/mp4',
  6277. 'm4u' => 'video/vnd.mpegurl',
  6278. 'm4v' => 'video/x-m4v',
  6279. 'ma' => 'application/mathematica',
  6280. 'mads' => 'application/mads+xml',
  6281. 'mag' => 'application/vnd.ecowin.chart',
  6282. 'maker' => 'application/vnd.framemaker',
  6283. 'man' => 'text/troff',
  6284. 'mar' => 'application/octet-stream',
  6285. 'mathml' => 'application/mathml+xml',
  6286. 'mb' => 'application/mathematica',
  6287. 'mbk' => 'application/vnd.mobius.mbk',
  6288. 'mbox' => 'application/mbox',
  6289. 'mc1' => 'application/vnd.medcalcdata',
  6290. 'mcd' => 'application/vnd.mcd',
  6291. 'mcurl' => 'text/vnd.curl.mcurl',
  6292. 'mdb' => 'application/x-msaccess',
  6293. 'mdi' => 'image/vnd.ms-modi',
  6294. 'me' => 'text/troff',
  6295. 'mesh' => 'model/mesh',
  6296. 'meta4' => 'application/metalink4+xml',
  6297. 'metalink' => 'application/metalink+xml',
  6298. 'mets' => 'application/mets+xml',
  6299. 'mfm' => 'application/vnd.mfmp',
  6300. 'mft' => 'application/rpki-manifest',
  6301. 'mgp' => 'application/vnd.osgeo.mapguide.package',
  6302. 'mgz' => 'application/vnd.proteus.magazine',
  6303. 'mid' => 'audio/midi',
  6304. 'midi' => 'audio/midi',
  6305. 'mie' => 'application/x-mie',
  6306. 'mif' => 'application/vnd.mif',
  6307. 'mime' => 'message/rfc822',
  6308. 'mj2' => 'video/mj2',
  6309. 'mjp2' => 'video/mj2',
  6310. 'mk3d' => 'video/x-matroska',
  6311. 'mka' => 'audio/x-matroska',
  6312. 'mks' => 'video/x-matroska',
  6313. 'mkv' => 'video/x-matroska',
  6314. 'mlp' => 'application/vnd.dolby.mlp',
  6315. 'mmd' => 'application/vnd.chipnuts.karaoke-mmd',
  6316. 'mmf' => 'application/vnd.smaf',
  6317. 'mmr' => 'image/vnd.fujixerox.edmics-mmr',
  6318. 'mng' => 'video/x-mng',
  6319. 'mny' => 'application/x-msmoney',
  6320. 'mobi' => 'application/x-mobipocket-ebook',
  6321. 'mods' => 'application/mods+xml',
  6322. 'mov' => 'video/quicktime',
  6323. 'movie' => 'video/x-sgi-movie',
  6324. 'mp2' => 'audio/mpeg',
  6325. 'mp21' => 'application/mp21',
  6326. 'mp2a' => 'audio/mpeg',
  6327. 'mp3' => 'audio/mpeg',
  6328. 'mp4' => 'video/mp4',
  6329. 'mp4a' => 'audio/mp4',
  6330. 'mp4s' => 'application/mp4',
  6331. 'mp4v' => 'video/mp4',
  6332. 'mpc' => 'application/vnd.mophun.certificate',
  6333. 'mpe' => 'video/mpeg',
  6334. 'mpeg' => 'video/mpeg',
  6335. 'mpg' => 'video/mpeg',
  6336. 'mpg4' => 'video/mp4',
  6337. 'mpga' => 'audio/mpeg',
  6338. 'mpkg' => 'application/vnd.apple.installer+xml',
  6339. 'mpm' => 'application/vnd.blueice.multipass',
  6340. 'mpn' => 'application/vnd.mophun.application',
  6341. 'mpp' => 'application/vnd.ms-project',
  6342. 'mpt' => 'application/vnd.ms-project',
  6343. 'mpy' => 'application/vnd.ibm.minipay',
  6344. 'mqy' => 'application/vnd.mobius.mqy',
  6345. 'mrc' => 'application/marc',
  6346. 'mrcx' => 'application/marcxml+xml',
  6347. 'ms' => 'text/troff',
  6348. 'mscml' => 'application/mediaservercontrol+xml',
  6349. 'mseed' => 'application/vnd.fdsn.mseed',
  6350. 'mseq' => 'application/vnd.mseq',
  6351. 'msf' => 'application/vnd.epson.msf',
  6352. 'msh' => 'model/mesh',
  6353. 'msi' => 'application/x-msdownload',
  6354. 'msl' => 'application/vnd.mobius.msl',
  6355. 'msty' => 'application/vnd.muvee.style',
  6356. 'mts' => 'model/vnd.mts',
  6357. 'mus' => 'application/vnd.musician',
  6358. 'musicxml' => 'application/vnd.recordare.musicxml+xml',
  6359. 'mvb' => 'application/x-msmediaview',
  6360. 'mwf' => 'application/vnd.mfer',
  6361. 'mxf' => 'application/mxf',
  6362. 'mxl' => 'application/vnd.recordare.musicxml',
  6363. 'mxml' => 'application/xv+xml',
  6364. 'mxs' => 'application/vnd.triscape.mxs',
  6365. 'mxu' => 'video/vnd.mpegurl',
  6366. 'n-gage' => 'application/vnd.nokia.n-gage.symbian.install',
  6367. 'n3' => 'text/n3',
  6368. 'nb' => 'application/mathematica',
  6369. 'nbp' => 'application/vnd.wolfram.player',
  6370. 'nc' => 'application/x-netcdf',
  6371. 'ncx' => 'application/x-dtbncx+xml',
  6372. 'nfo' => 'text/x-nfo',
  6373. 'ngdat' => 'application/vnd.nokia.n-gage.data',
  6374. 'nitf' => 'application/vnd.nitf',
  6375. 'nlu' => 'application/vnd.neurolanguage.nlu',
  6376. 'nml' => 'application/vnd.enliven',
  6377. 'nnd' => 'application/vnd.noblenet-directory',
  6378. 'nns' => 'application/vnd.noblenet-sealer',
  6379. 'nnw' => 'application/vnd.noblenet-web',
  6380. 'npx' => 'image/vnd.net-fpx',
  6381. 'nsc' => 'application/x-conference',
  6382. 'nsf' => 'application/vnd.lotus-notes',
  6383. 'ntf' => 'application/vnd.nitf',
  6384. 'nzb' => 'application/x-nzb',
  6385. 'oa2' => 'application/vnd.fujitsu.oasys2',
  6386. 'oa3' => 'application/vnd.fujitsu.oasys3',
  6387. 'oas' => 'application/vnd.fujitsu.oasys',
  6388. 'obd' => 'application/x-msbinder',
  6389. 'obj' => 'application/x-tgif',
  6390. 'oda' => 'application/oda',
  6391. 'odb' => 'application/vnd.oasis.opendocument.database',
  6392. 'odc' => 'application/vnd.oasis.opendocument.chart',
  6393. 'odf' => 'application/vnd.oasis.opendocument.formula',
  6394. 'odft' => 'application/vnd.oasis.opendocument.formula-template',
  6395. 'odg' => 'application/vnd.oasis.opendocument.graphics',
  6396. 'odi' => 'application/vnd.oasis.opendocument.image',
  6397. 'odm' => 'application/vnd.oasis.opendocument.text-master',
  6398. 'odp' => 'application/vnd.oasis.opendocument.presentation',
  6399. 'ods' => 'application/vnd.oasis.opendocument.spreadsheet',
  6400. 'odt' => 'application/vnd.oasis.opendocument.text',
  6401. 'oga' => 'audio/ogg',
  6402. 'ogg' => 'audio/ogg',
  6403. 'ogv' => 'video/ogg',
  6404. 'ogx' => 'application/ogg',
  6405. 'omdoc' => 'application/omdoc+xml',
  6406. 'onepkg' => 'application/onenote',
  6407. 'onetmp' => 'application/onenote',
  6408. 'onetoc' => 'application/onenote',
  6409. 'onetoc2' => 'application/onenote',
  6410. 'opf' => 'application/oebps-package+xml',
  6411. 'opml' => 'text/x-opml',
  6412. 'oprc' => 'application/vnd.palm',
  6413. 'org' => 'application/vnd.lotus-organizer',
  6414. 'osf' => 'application/vnd.yamaha.openscoreformat',
  6415. 'osfpvg' => 'application/vnd.yamaha.openscoreformat.osfpvg+xml',
  6416. 'otc' => 'application/vnd.oasis.opendocument.chart-template',
  6417. 'otf' => 'application/x-font-otf',
  6418. 'otg' => 'application/vnd.oasis.opendocument.graphics-template',
  6419. 'oth' => 'application/vnd.oasis.opendocument.text-web',
  6420. 'oti' => 'application/vnd.oasis.opendocument.image-template',
  6421. 'otp' => 'application/vnd.oasis.opendocument.presentation-template',
  6422. 'ots' => 'application/vnd.oasis.opendocument.spreadsheet-template',
  6423. 'ott' => 'application/vnd.oasis.opendocument.text-template',
  6424. 'oxps' => 'application/oxps',
  6425. 'oxt' => 'application/vnd.openofficeorg.extension',
  6426. 'p' => 'text/x-pascal',
  6427. 'p10' => 'application/pkcs10',
  6428. 'p12' => 'application/x-pkcs12',
  6429. 'p7b' => 'application/x-pkcs7-certificates',
  6430. 'p7c' => 'application/pkcs7-mime',
  6431. 'p7m' => 'application/pkcs7-mime',
  6432. 'p7r' => 'application/x-pkcs7-certreqresp',
  6433. 'p7s' => 'application/pkcs7-signature',
  6434. 'p8' => 'application/pkcs8',
  6435. 'pas' => 'text/x-pascal',
  6436. 'paw' => 'application/vnd.pawaafile',
  6437. 'pbd' => 'application/vnd.powerbuilder6',
  6438. 'pbm' => 'image/x-portable-bitmap',
  6439. 'pcap' => 'application/vnd.tcpdump.pcap',
  6440. 'pcf' => 'application/x-font-pcf',
  6441. 'pcl' => 'application/vnd.hp-pcl',
  6442. 'pclxl' => 'application/vnd.hp-pclxl',
  6443. 'pct' => 'image/x-pict',
  6444. 'pcurl' => 'application/vnd.curl.pcurl',
  6445. 'pcx' => 'image/x-pcx',
  6446. 'pdb' => 'application/vnd.palm',
  6447. 'pdf' => 'application/pdf',
  6448. 'pfa' => 'application/x-font-type1',
  6449. 'pfb' => 'application/x-font-type1',
  6450. 'pfm' => 'application/x-font-type1',
  6451. 'pfr' => 'application/font-tdpfr',
  6452. 'pfx' => 'application/x-pkcs12',
  6453. 'pgm' => 'image/x-portable-graymap',
  6454. 'pgn' => 'application/x-chess-pgn',
  6455. 'pgp' => 'application/pgp-encrypted',
  6456. 'pic' => 'image/x-pict',
  6457. 'pkg' => 'application/octet-stream',
  6458. 'pki' => 'application/pkixcmp',
  6459. 'pkipath' => 'application/pkix-pkipath',
  6460. 'plb' => 'application/vnd.3gpp.pic-bw-large',
  6461. 'plc' => 'application/vnd.mobius.plc',
  6462. 'plf' => 'application/vnd.pocketlearn',
  6463. 'pls' => 'application/pls+xml',
  6464. 'pml' => 'application/vnd.ctc-posml',
  6465. 'png' => 'image/png',
  6466. 'pnm' => 'image/x-portable-anymap',
  6467. 'portpkg' => 'application/vnd.macports.portpkg',
  6468. 'pot' => 'application/vnd.ms-powerpoint',
  6469. 'potm' => 'application/vnd.ms-powerpoint.template.macroenabled.12',
  6470. 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template',
  6471. 'ppam' => 'application/vnd.ms-powerpoint.addin.macroenabled.12',
  6472. 'ppd' => 'application/vnd.cups-ppd',
  6473. 'ppm' => 'image/x-portable-pixmap',
  6474. 'pps' => 'application/vnd.ms-powerpoint',
  6475. 'ppsm' => 'application/vnd.ms-powerpoint.slideshow.macroenabled.12',
  6476. 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow',
  6477. 'ppt' => 'application/vnd.ms-powerpoint',
  6478. 'pptm' => 'application/vnd.ms-powerpoint.presentation.macroenabled.12',
  6479. 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
  6480. 'pqa' => 'application/vnd.palm',
  6481. 'prc' => 'application/x-mobipocket-ebook',
  6482. 'pre' => 'application/vnd.lotus-freelance',
  6483. 'prf' => 'application/pics-rules',
  6484. 'ps' => 'application/postscript',
  6485. 'psb' => 'application/vnd.3gpp.pic-bw-small',
  6486. 'psd' => 'image/vnd.adobe.photoshop',
  6487. 'psf' => 'application/x-font-linux-psf',
  6488. 'pskcxml' => 'application/pskc+xml',
  6489. 'ptid' => 'application/vnd.pvi.ptid1',
  6490. 'pub' => 'application/x-mspublisher',
  6491. 'pvb' => 'application/vnd.3gpp.pic-bw-var',
  6492. 'pwn' => 'application/vnd.3m.post-it-notes',
  6493. 'pya' => 'audio/vnd.ms-playready.media.pya',
  6494. 'pyv' => 'video/vnd.ms-playready.media.pyv',
  6495. 'qam' => 'application/vnd.epson.quickanime',
  6496. 'qbo' => 'application/vnd.intu.qbo',
  6497. 'qfx' => 'application/vnd.intu.qfx',
  6498. 'qps' => 'application/vnd.publishare-delta-tree',
  6499. 'qt' => 'video/quicktime',
  6500. 'qwd' => 'application/vnd.quark.quarkxpress',
  6501. 'qwt' => 'application/vnd.quark.quarkxpress',
  6502. 'qxb' => 'application/vnd.quark.quarkxpress',
  6503. 'qxd' => 'application/vnd.quark.quarkxpress',
  6504. 'qxl' => 'application/vnd.quark.quarkxpress',
  6505. 'qxt' => 'application/vnd.quark.quarkxpress',
  6506. 'ra' => 'audio/x-pn-realaudio',
  6507. 'ram' => 'audio/x-pn-realaudio',
  6508. 'rar' => 'application/x-rar-compressed',
  6509. 'ras' => 'image/x-cmu-raster',
  6510. 'rcprofile' => 'application/vnd.ipunplugged.rcprofile',
  6511. 'rdf' => 'application/rdf+xml',
  6512. 'rdz' => 'application/vnd.data-vision.rdz',
  6513. 'rep' => 'application/vnd.businessobjects',
  6514. 'res' => 'application/x-dtbresource+xml',
  6515. 'rgb' => 'image/x-rgb',
  6516. 'rif' => 'application/reginfo+xml',
  6517. 'rip' => 'audio/vnd.rip',
  6518. 'ris' => 'application/x-research-info-systems',
  6519. 'rl' => 'application/resource-lists+xml',
  6520. 'rlc' => 'image/vnd.fujixerox.edmics-rlc',
  6521. 'rld' => 'application/resource-lists-diff+xml',
  6522. 'rm' => 'application/vnd.rn-realmedia',
  6523. 'rmi' => 'audio/midi',
  6524. 'rmp' => 'audio/x-pn-realaudio-plugin',
  6525. 'rms' => 'application/vnd.jcp.javame.midlet-rms',
  6526. 'rmvb' => 'application/vnd.rn-realmedia-vbr',
  6527. 'rnc' => 'application/relax-ng-compact-syntax',
  6528. 'roa' => 'application/rpki-roa',
  6529. 'roff' => 'text/troff',
  6530. 'rp9' => 'application/vnd.cloanto.rp9',
  6531. 'rpss' => 'application/vnd.nokia.radio-presets',
  6532. 'rpst' => 'application/vnd.nokia.radio-preset',
  6533. 'rq' => 'application/sparql-query',
  6534. 'rs' => 'application/rls-services+xml',
  6535. 'rsd' => 'application/rsd+xml',
  6536. 'rss' => 'application/rss+xml',
  6537. 'rtf' => 'application/rtf',
  6538. 'rtx' => 'text/richtext',
  6539. 's' => 'text/x-asm',
  6540. 's3m' => 'audio/s3m',
  6541. 'saf' => 'application/vnd.yamaha.smaf-audio',
  6542. 'sbml' => 'application/sbml+xml',
  6543. 'sc' => 'application/vnd.ibm.secure-container',
  6544. 'scd' => 'application/x-msschedule',
  6545. 'scm' => 'application/vnd.lotus-screencam',
  6546. 'scq' => 'application/scvp-cv-request',
  6547. 'scs' => 'application/scvp-cv-response',
  6548. 'scurl' => 'text/vnd.curl.scurl',
  6549. 'sda' => 'application/vnd.stardivision.draw',
  6550. 'sdc' => 'application/vnd.stardivision.calc',
  6551. 'sdd' => 'application/vnd.stardivision.impress',
  6552. 'sdkd' => 'application/vnd.solent.sdkm+xml',
  6553. 'sdkm' => 'application/vnd.solent.sdkm+xml',
  6554. 'sdp' => 'application/sdp',
  6555. 'sdw' => 'application/vnd.stardivision.writer',
  6556. 'see' => 'application/vnd.seemail',
  6557. 'seed' => 'application/vnd.fdsn.seed',
  6558. 'sema' => 'application/vnd.sema',
  6559. 'semd' => 'application/vnd.semd',
  6560. 'semf' => 'application/vnd.semf',
  6561. 'ser' => 'application/java-serialized-object',
  6562. 'setpay' => 'application/set-payment-initiation',
  6563. 'setreg' => 'application/set-registration-initiation',
  6564. 'sfd-hdstx' => 'application/vnd.hydrostatix.sof-data',
  6565. 'sfs' => 'application/vnd.spotfire.sfs',
  6566. 'sfv' => 'text/x-sfv',
  6567. 'sgi' => 'image/sgi',
  6568. 'sgl' => 'application/vnd.stardivision.writer-global',
  6569. 'sgm' => 'text/sgml',
  6570. 'sgml' => 'text/sgml',
  6571. 'sh' => 'application/x-sh',
  6572. 'shar' => 'application/x-shar',
  6573. 'shf' => 'application/shf+xml',
  6574. 'sid' => 'image/x-mrsid-image',
  6575. 'sig' => 'application/pgp-signature',
  6576. 'sil' => 'audio/silk',
  6577. 'silo' => 'model/mesh',
  6578. 'sis' => 'application/vnd.symbian.install',
  6579. 'sisx' => 'application/vnd.symbian.install',
  6580. 'sit' => 'application/x-stuffit',
  6581. 'sitx' => 'application/x-stuffitx',
  6582. 'skd' => 'application/vnd.koan',
  6583. 'skm' => 'application/vnd.koan',
  6584. 'skp' => 'application/vnd.koan',
  6585. 'skt' => 'application/vnd.koan',
  6586. 'sldm' => 'application/vnd.ms-powerpoint.slide.macroenabled.12',
  6587. 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide',
  6588. 'slt' => 'application/vnd.epson.salt',
  6589. 'sm' => 'application/vnd.stepmania.stepchart',
  6590. 'smf' => 'application/vnd.stardivision.math',
  6591. 'smi' => 'application/smil+xml',
  6592. 'smil' => 'application/smil+xml',
  6593. 'smv' => 'video/x-smv',
  6594. 'smzip' => 'application/vnd.stepmania.package',
  6595. 'snd' => 'audio/basic',
  6596. 'snf' => 'application/x-font-snf',
  6597. 'so' => 'application/octet-stream',
  6598. 'spc' => 'application/x-pkcs7-certificates',
  6599. 'spf' => 'application/vnd.yamaha.smaf-phrase',
  6600. 'spl' => 'application/x-futuresplash',
  6601. 'spot' => 'text/vnd.in3d.spot',
  6602. 'spp' => 'application/scvp-vp-response',
  6603. 'spq' => 'application/scvp-vp-request',
  6604. 'spx' => 'audio/ogg',
  6605. 'sql' => 'application/x-sql',
  6606. 'src' => 'application/x-wais-source',
  6607. 'srt' => 'application/x-subrip',
  6608. 'sru' => 'application/sru+xml',
  6609. 'srx' => 'application/sparql-results+xml',
  6610. 'ssdl' => 'application/ssdl+xml',
  6611. 'sse' => 'application/vnd.kodak-descriptor',
  6612. 'ssf' => 'application/vnd.epson.ssf',
  6613. 'ssml' => 'application/ssml+xml',
  6614. 'st' => 'application/vnd.sailingtracker.track',
  6615. 'stc' => 'application/vnd.sun.xml.calc.template',
  6616. 'std' => 'application/vnd.sun.xml.draw.template',
  6617. 'stf' => 'application/vnd.wt.stf',
  6618. 'sti' => 'application/vnd.sun.xml.impress.template',
  6619. 'stk' => 'application/hyperstudio',
  6620. 'stl' => 'application/vnd.ms-pki.stl',
  6621. 'str' => 'application/vnd.pg.format',
  6622. 'stw' => 'application/vnd.sun.xml.writer.template',
  6623. 'sub' => 'image/vnd.dvb.subtitle',
  6624. 'sub' => 'text/vnd.dvb.subtitle',
  6625. 'sus' => 'application/vnd.sus-calendar',
  6626. 'susp' => 'application/vnd.sus-calendar',
  6627. 'sv4cpio' => 'application/x-sv4cpio',
  6628. 'sv4crc' => 'application/x-sv4crc',
  6629. 'svc' => 'application/vnd.dvb.service',
  6630. 'svd' => 'application/vnd.svd',
  6631. 'svg' => 'image/svg+xml',
  6632. 'svgz' => 'image/svg+xml',
  6633. 'swa' => 'application/x-director',
  6634. 'swf' => 'application/x-shockwave-flash',
  6635. 'swi' => 'application/vnd.aristanetworks.swi',
  6636. 'sxc' => 'application/vnd.sun.xml.calc',
  6637. 'sxd' => 'application/vnd.sun.xml.draw',
  6638. 'sxg' => 'application/vnd.sun.xml.writer.global',
  6639. 'sxi' => 'application/vnd.sun.xml.impress',
  6640. 'sxm' => 'application/vnd.sun.xml.math',
  6641. 'sxw' => 'application/vnd.sun.xml.writer',
  6642. 't' => 'text/troff',
  6643. 't3' => 'application/x-t3vm-image',
  6644. 'taglet' => 'application/vnd.mynfc',
  6645. 'tao' => 'application/vnd.tao.intent-module-archive',
  6646. 'tar' => 'application/x-tar',
  6647. 'tcap' => 'application/vnd.3gpp2.tcap',
  6648. 'tcl' => 'application/x-tcl',
  6649. 'teacher' => 'application/vnd.smart.teacher',
  6650. 'tei' => 'application/tei+xml',
  6651. 'teicorpus' => 'application/tei+xml',
  6652. 'tex' => 'application/x-tex',
  6653. 'texi' => 'application/x-texinfo',
  6654. 'texinfo' => 'application/x-texinfo',
  6655. 'text' => 'text/plain',
  6656. 'tfi' => 'application/thraud+xml',
  6657. 'tfm' => 'application/x-tex-tfm',
  6658. 'tga' => 'image/x-tga',
  6659. 'thmx' => 'application/vnd.ms-officetheme',
  6660. 'tif' => 'image/tiff',
  6661. 'tiff' => 'image/tiff',
  6662. 'tmo' => 'application/vnd.tmobile-livetv',
  6663. 'torrent' => 'application/x-bittorrent',
  6664. 'tpl' => 'application/vnd.groove-tool-template',
  6665. 'tpt' => 'application/vnd.trid.tpt',
  6666. 'tr' => 'text/troff',
  6667. 'tra' => 'application/vnd.trueapp',
  6668. 'trm' => 'application/x-msterminal',
  6669. 'tsd' => 'application/timestamped-data',
  6670. 'tsv' => 'text/tab-separated-values',
  6671. 'ttc' => 'application/x-font-ttf',
  6672. 'ttf' => 'application/x-font-ttf',
  6673. 'ttl' => 'text/turtle',
  6674. 'twd' => 'application/vnd.simtech-mindmapper',
  6675. 'twds' => 'application/vnd.simtech-mindmapper',
  6676. 'txd' => 'application/vnd.genomatix.tuxedo',
  6677. 'txf' => 'application/vnd.mobius.txf',
  6678. 'txt' => 'text/plain',
  6679. 'u32' => 'application/x-authorware-bin',
  6680. 'udeb' => 'application/x-debian-package',
  6681. 'ufd' => 'application/vnd.ufdl',
  6682. 'ufdl' => 'application/vnd.ufdl',
  6683. 'ulx' => 'application/x-glulx',
  6684. 'umj' => 'application/vnd.umajin',
  6685. 'unityweb' => 'application/vnd.unity',
  6686. 'uoml' => 'application/vnd.uoml+xml',
  6687. 'uri' => 'text/uri-list',
  6688. 'uris' => 'text/uri-list',
  6689. 'urls' => 'text/uri-list',
  6690. 'ustar' => 'application/x-ustar',
  6691. 'utz' => 'application/vnd.uiq.theme',
  6692. 'uu' => 'text/x-uuencode',
  6693. 'uva' => 'audio/vnd.dece.audio',
  6694. 'uvd' => 'application/vnd.dece.data',
  6695. 'uvf' => 'application/vnd.dece.data',
  6696. 'uvg' => 'image/vnd.dece.graphic',
  6697. 'uvh' => 'video/vnd.dece.hd',
  6698. 'uvi' => 'image/vnd.dece.graphic',
  6699. 'uvm' => 'video/vnd.dece.mobile',
  6700. 'uvp' => 'video/vnd.dece.pd',
  6701. 'uvs' => 'video/vnd.dece.sd',
  6702. 'uvt' => 'application/vnd.dece.ttml+xml',
  6703. 'uvu' => 'video/vnd.uvvu.mp4',
  6704. 'uvv' => 'video/vnd.dece.video',
  6705. 'uvva' => 'audio/vnd.dece.audio',
  6706. 'uvvd' => 'application/vnd.dece.data',
  6707. 'uvvf' => 'application/vnd.dece.data',
  6708. 'uvvg' => 'image/vnd.dece.graphic',
  6709. 'uvvh' => 'video/vnd.dece.hd',
  6710. 'uvvi' => 'image/vnd.dece.graphic',
  6711. 'uvvm' => 'video/vnd.dece.mobile',
  6712. 'uvvp' => 'video/vnd.dece.pd',
  6713. 'uvvs' => 'video/vnd.dece.sd',
  6714. 'uvvt' => 'application/vnd.dece.ttml+xml',
  6715. 'uvvu' => 'video/vnd.uvvu.mp4',
  6716. 'uvvv' => 'video/vnd.dece.video',
  6717. 'uvvx' => 'application/vnd.dece.unspecified',
  6718. 'uvvz' => 'application/vnd.dece.zip',
  6719. 'uvx' => 'application/vnd.dece.unspecified',
  6720. 'uvz' => 'application/vnd.dece.zip',
  6721. 'vcard' => 'text/vcard',
  6722. 'vcd' => 'application/x-cdlink',
  6723. 'vcf' => 'text/x-vcard',
  6724. 'vcg' => 'application/vnd.groove-vcard',
  6725. 'vcs' => 'text/x-vcalendar',
  6726. 'vcx' => 'application/vnd.vcx',
  6727. 'vis' => 'application/vnd.visionary',
  6728. 'viv' => 'video/vnd.vivo',
  6729. 'vob' => 'video/x-ms-vob',
  6730. 'vor' => 'application/vnd.stardivision.writer',
  6731. 'vox' => 'application/x-authorware-bin',
  6732. 'vrml' => 'model/vrml',
  6733. 'vsd' => 'application/vnd.visio',
  6734. 'vsf' => 'application/vnd.vsf',
  6735. 'vss' => 'application/vnd.visio',
  6736. 'vst' => 'application/vnd.visio',
  6737. 'vsw' => 'application/vnd.visio',
  6738. 'vtu' => 'model/vnd.vtu',
  6739. 'vxml' => 'application/voicexml+xml',
  6740. 'w3d' => 'application/x-director',
  6741. 'wad' => 'application/x-doom',
  6742. 'wav' => 'audio/x-wav',
  6743. 'wax' => 'audio/x-ms-wax',
  6744. 'wbmp' => 'image/vnd.wap.wbmp',
  6745. 'wbs' => 'application/vnd.criticaltools.wbs+xml',
  6746. 'wbxml' => 'application/vnd.wap.wbxml',
  6747. 'wcm' => 'application/vnd.ms-works',
  6748. 'wdb' => 'application/vnd.ms-works',
  6749. 'wdp' => 'image/vnd.ms-photo',
  6750. 'weba' => 'audio/webm',
  6751. 'webm' => 'video/webm',
  6752. 'webp' => 'image/webp',
  6753. 'wg' => 'application/vnd.pmi.widget',
  6754. 'wgt' => 'application/widget',
  6755. 'wks' => 'application/vnd.ms-works',
  6756. 'wm' => 'video/x-ms-wm',
  6757. 'wma' => 'audio/x-ms-wma',
  6758. 'wmd' => 'application/x-ms-wmd',
  6759. 'wmf' => 'application/x-msmetafile',
  6760. 'wml' => 'text/vnd.wap.wml',
  6761. 'wmlc' => 'application/vnd.wap.wmlc',
  6762. 'wmls' => 'text/vnd.wap.wmlscript',
  6763. 'wmlsc' => 'application/vnd.wap.wmlscriptc',
  6764. 'wmv' => 'video/x-ms-wmv',
  6765. 'wmx' => 'video/x-ms-wmx',
  6766. 'wmz' => 'application/x-ms-wmz',
  6767. 'wmz' => 'application/x-msmetafile',
  6768. 'woff' => 'application/font-woff',
  6769. 'wpd' => 'application/vnd.wordperfect',
  6770. 'wpl' => 'application/vnd.ms-wpl',
  6771. 'wps' => 'application/vnd.ms-works',
  6772. 'wqd' => 'application/vnd.wqd',
  6773. 'wri' => 'application/x-mswrite',
  6774. 'wrl' => 'model/vrml',
  6775. 'wsdl' => 'application/wsdl+xml',
  6776. 'wspolicy' => 'application/wspolicy+xml',
  6777. 'wtb' => 'application/vnd.webturbo',
  6778. 'wvx' => 'video/x-ms-wvx',
  6779. 'x32' => 'application/x-authorware-bin',
  6780. 'x3d' => 'model/x3d+xml',
  6781. 'x3db' => 'model/x3d+binary',
  6782. 'x3dbz' => 'model/x3d+binary',
  6783. 'x3dv' => 'model/x3d+vrml',
  6784. 'x3dvz' => 'model/x3d+vrml',
  6785. 'x3dz' => 'model/x3d+xml',
  6786. 'xaml' => 'application/xaml+xml',
  6787. 'xap' => 'application/x-silverlight-app',
  6788. 'xar' => 'application/vnd.xara',
  6789. 'xbap' => 'application/x-ms-xbap',
  6790. 'xbd' => 'application/vnd.fujixerox.docuworks.binder',
  6791. 'xbm' => 'image/x-xbitmap',
  6792. 'xdf' => 'application/xcap-diff+xml',
  6793. 'xdm' => 'application/vnd.syncml.dm+xml',
  6794. 'xdp' => 'application/vnd.adobe.xdp+xml',
  6795. 'xdssc' => 'application/dssc+xml',
  6796. 'xdw' => 'application/vnd.fujixerox.docuworks',
  6797. 'xenc' => 'application/xenc+xml',
  6798. 'xer' => 'application/patch-ops-error+xml',
  6799. 'xfdf' => 'application/vnd.adobe.xfdf',
  6800. 'xfdl' => 'application/vnd.xfdl',
  6801. 'xht' => 'application/xhtml+xml',
  6802. 'xhtml' => 'application/xhtml+xml',
  6803. 'xhvml' => 'application/xv+xml',
  6804. 'xif' => 'image/vnd.xiff',
  6805. 'xla' => 'application/vnd.ms-excel',
  6806. 'xlam' => 'application/vnd.ms-excel.addin.macroenabled.12',
  6807. 'xlc' => 'application/vnd.ms-excel',
  6808. 'xlf' => 'application/x-xliff+xml',
  6809. 'xlm' => 'application/vnd.ms-excel',
  6810. 'xls' => 'application/vnd.ms-excel',
  6811. 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroenabled.12',
  6812. 'xlsm' => 'application/vnd.ms-excel.sheet.macroenabled.12',
  6813. 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
  6814. 'xlt' => 'application/vnd.ms-excel',
  6815. 'xltm' => 'application/vnd.ms-excel.template.macroenabled.12',
  6816. 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template',
  6817. 'xlw' => 'application/vnd.ms-excel',
  6818. 'xm' => 'audio/xm',
  6819. 'xml' => 'application/xml',
  6820. 'xo' => 'application/vnd.olpc-sugar',
  6821. 'xop' => 'application/xop+xml',
  6822. 'xpi' => 'application/x-xpinstall',
  6823. 'xpl' => 'application/xproc+xml',
  6824. 'xpm' => 'image/x-xpixmap',
  6825. 'xpr' => 'application/vnd.is-xpr',
  6826. 'xps' => 'application/vnd.ms-xpsdocument',
  6827. 'xpw' => 'application/vnd.intercon.formnet',
  6828. 'xpx' => 'application/vnd.intercon.formnet',
  6829. 'xsl' => 'application/xml',
  6830. 'xslt' => 'application/xslt+xml',
  6831. 'xsm' => 'application/vnd.syncml+xml',
  6832. 'xspf' => 'application/xspf+xml',
  6833. 'xul' => 'application/vnd.mozilla.xul+xml',
  6834. 'xvm' => 'application/xv+xml',
  6835. 'xvml' => 'application/xv+xml',
  6836. 'xwd' => 'image/x-xwindowdump',
  6837. 'xyz' => 'chemical/x-xyz',
  6838. 'xz' => 'application/x-xz',
  6839. 'yang' => 'application/yang',
  6840. 'yin' => 'application/yin+xml',
  6841. 'z1' => 'application/x-zmachine',
  6842. 'z2' => 'application/x-zmachine',
  6843. 'z3' => 'application/x-zmachine',
  6844. 'z4' => 'application/x-zmachine',
  6845. 'z5' => 'application/x-zmachine',
  6846. 'z6' => 'application/x-zmachine',
  6847. 'z7' => 'application/x-zmachine',
  6848. 'z8' => 'application/x-zmachine',
  6849. 'zaz' => 'application/vnd.zzazz.deck+xml',
  6850. 'zip' => 'application/zip',
  6851. 'zir' => 'application/vnd.zul',
  6852. 'zirz' => 'application/vnd.zul',
  6853. 'zmm' => 'application/vnd.handheld-entertainment+xml'
  6854. );
  6855. }
  6856. // Always run this
  6857. dependCheck();