Logo ROOT  
Reference Guide
 
Loading...
Searching...
No Matches
TWebCanvas.cxx
Go to the documentation of this file.
1// Author: Sergey Linev, GSI 7/12/2016
2
3/*************************************************************************
4 * Copyright (C) 1995-2023, Rene Brun and Fons Rademakers. *
5 * All rights reserved. *
6 * *
7 * For the licensing terms see $ROOTSYS/LICENSE. *
8 * For the list of contributors see $ROOTSYS/README/CREDITS. *
9 *************************************************************************/
10
11#include "TWebCanvas.h"
12
13#include "TWebSnapshot.h"
14#include "TWebPadPainter.h"
15#include "TWebPS.h"
16#include "TWebMenuItem.h"
18#include "THttpServer.h"
19
20#include "TSystem.h"
21#include "TStyle.h"
22#include "TCanvas.h"
23#include "TButton.h"
24#include "TSlider.h"
25#include "TFrame.h"
26#include "TPaveText.h"
27#include "TPaveStats.h"
28#include "TText.h"
29#include "TROOT.h"
30#include "TClass.h"
31#include "TColor.h"
32#include "TObjArray.h"
33#include "TArrayI.h"
34#include "TList.h"
35#include "TF1.h"
36#include "TF2.h"
37#include "TF3.h"
38#include "TH1.h"
39#include "TH2.h"
40#include "THStack.h"
41#include "TMultiGraph.h"
42#include "TEnv.h"
43#include "TError.h"
44#include "TGraph.h"
45#include "TGraphPolar.h"
46#include "TGraphPolargram.h"
47#include "TGraph2D.h"
48#include "TGaxis.h"
49#include "TScatter.h"
50#include "TCutG.h"
51#include "TBufferJSON.h"
52#include "TBase64.h"
53#include "TAtt3D.h"
54#include "TView.h"
55#include "TExec.h"
56#include "TVirtualX.h"
57#include "TMath.h"
58#include "TTimer.h"
59#include "TThread.h"
60
61#include <cstdio>
62#include <cstring>
63#include <fstream>
64#include <iostream>
65#include <memory>
66#include <sstream>
67#include <vector>
68
69
70class TWebCanvasTimer : public TTimer {
75public:
77
78 Bool_t IsSlow() const { return fSlow; }
80 {
81 fSlow = slow;
82 fSlowCnt = 0;
83 SetTime(slow ? 50 : 10);
84 }
85
86 /// used to send control messages to clients
87 void Timeout() override
88 {
90 return;
94 if (res) {
95 fSlowCnt = 0;
96 } else if (++fSlowCnt > 100 && !IsSlow()) {
98 }
99 }
100};
101
102
103/** \class TWebCanvas
104\ingroup webgui6
105\ingroup webwidgets
106
107Basic TCanvasImp ABI implementation for Web-based Graphics
108Provides painting of main ROOT classes in web browsers using [JSROOT](https://root.cern/js/)
109
110Following settings parameters can be useful for TWebCanvas:
111
112 WebGui.FullCanvas: 1 read-only mode (0), full-functional canvas (1) (default - 1)
113 WebGui.StyleDelivery: 1 provide gStyle object to JSROOT client (default - 1)
114 WebGui.PaletteDelivery: 1 provide color palette to JSROOT client (default - 1)
115 WebGui.TF1UseSave: 1 used saved values for function drawing: 0 - off, 1 - if client fail to evaluate function, 2 - always (default - 1)
116
117TWebCanvas is used by default in interactive ROOT session. To use web-based canvas in batch mode for image
118generation, one should explicitly specify `--web` option when starting ROOT:
119
120 [shell] root -b --web tutorials/hsimple.root -e 'hpxpy->Draw("colz"); c1->SaveAs("image.png");'
121
122If for any reasons TWebCanvas does not provide required functionality, one always can disable it.
123Either by specifying `root --web=off` when starting ROOT or by setting `Canvas.Name: TRootCanvas` in rootrc file.
124
125*/
126
127using namespace std::string_literals;
128
129static const std::string sid_pad_histogram = "__pad_histogram__";
130
131
140
141static std::vector<WebFont_t> gWebFonts;
142
143std::string TWebCanvas::gCustomScripts = {};
144std::vector<std::string> TWebCanvas::gCustomClasses = {};
145
148std::vector<std::string> TWebCanvas::gBatchFiles;
149std::vector<std::string> TWebCanvas::gBatchJsons;
150std::vector<int> TWebCanvas::gBatchWidths;
151std::vector<int> TWebCanvas::gBatchHeights;
152
153//////////////////////////////////////////////////////////////////////////////////////////////////
154/// Configure batch image mode for web graphics.
155/// Allows to process many images with single headless browser invocation and increase performance of image production.
156/// When many canvases are stored as image in difference places, they first collected in batch and then processed when at least `n`
157/// images are prepared. Only then headless browser invoked and create all these images at once.
158/// This allows to significantly increase performance of image production in web mode
159
166
167//////////////////////////////////////////////////////////////////////////////////////////////////
168/// Flush batch images
169
171{
172 bool res = true;
173
174 if (gBatchJsons.size() > 0)
176
177 gBatchFiles.clear();
178 gBatchJsons.clear();
179 gBatchWidths.clear();
180 gBatchHeights.clear();
181
182 return res;
183}
184
185////////////////////////////////////////////////////////////////////////////////
186/// Constructor
187
189 : TCanvasImp(c, name, x, y, width, height)
190{
191 // Workaround for multi-threaded environment
192 // Ensure main thread id picked when canvas implementation is created -
193 // otherwise it may be assigned in other thread and screw-up gPad access.
194 // Workaround may not work if main thread id was wrongly initialized before
195 // This resolves issue https://github.com/root-project/root/issues/15498
197
198 fTimer = new TWebCanvasTimer(*this);
199
201 fStyleDelivery = gEnv->GetValue("WebGui.StyleDelivery", 1);
202 fPaletteDelivery = gEnv->GetValue("WebGui.PaletteDelivery", 1);
203 fPrimitivesMerge = gEnv->GetValue("WebGui.PrimitivesMerge", 100);
204 fTF1UseSave = gEnv->GetValue("WebGui.TF1UseSave", (Int_t) 1);
206
207 fWebConn.emplace_back(0); // add special connection which only used to perform updates
208
209 fTimer->TurnOn();
210
211 // fAsyncMode = kTRUE;
212}
213
214
215////////////////////////////////////////////////////////////////////////////////
216/// Destructor
217
219{
220 if(fWindow)
221 fWindow->Reset();
222
223 delete fTimer;
224}
225
226//////////////////////////////////////////////////////////////////////////////////////////////////
227/// Add font to static list of fonts supported by the canvas
228/// Name specifies name of the font, second is font file with .ttf or .woff2 extension
229/// Only True Type Fonts (ttf) are supported by PDF
230/// Returns font index which can be used in
231/// auto font_indx = TWebCanvas::AddFont("test", "test.ttf", 2);
232/// gStyle->SetStatFont(font_indx);
233
234Font_t TWebCanvas::AddFont(const char *name, const char *fontfile, Int_t precision)
235{
236 Font_t maxindx = 22;
237 for (auto &entry : gWebFonts) {
238 if (entry.fName == name)
239 return precision > 0 ? entry.fIndx*10 + precision : entry.fIndx;
240 if (entry.fIndx > maxindx)
241 maxindx = entry.fIndx;
242 }
243
244 TString fullname = fontfile, fmt = "ttf";
245 auto pos = fullname.Last('.');
246 if (pos != kNPOS) {
247 fmt = fullname(pos+1, fullname.Length() - pos);
248 fmt.ToLower();
249 if ((fmt != "ttf") && (fmt != "woff2")) {
250 ::Error("TWebCanvas::AddFont", "Unsupported font file extension %s", fmt.Data());
251 return (Font_t) -1;
252 }
253 }
254
255 gSystem->ExpandPathName(fullname);
256
257 if (gSystem->AccessPathName(fullname.Data(), kReadPermission)) {
258 ::Error("TWebCanvas::AddFont", "Not possible to read font file %s", fullname.Data());
259 return (Font_t) -1;
260 }
261
262 std::ifstream is(fullname.Data(), std::ios::in | std::ios::binary);
263 std::string res;
264 if (is) {
265 is.seekg(0, std::ios::end);
266 res.resize(is.tellg());
267 is.seekg(0, std::ios::beg);
268 is.read((char *)res.data(), res.length());
269 if (!is)
270 res.clear();
271 }
272
273 if (res.empty()) {
274 ::Error("TWebCanvas::AddFont", "Fail to read font file %s", fullname.Data());
275 return (Font_t) -1;
276 }
277
278 TString base64 = TBase64::Encode(res.c_str(), res.length());
279
280 maxindx++;
281
282 gWebFonts.emplace_back(maxindx, name, fmt, base64);
283
284 return precision > 0 ? maxindx*10 + precision : maxindx;
285}
286
287////////////////////////////////////////////////////////////////////////////////
288/// Initialize window for the web canvas
289/// At this place canvas is not yet register to the list of canvases - one cannot call RWebWindow::Show()
290
292{
293 return 111222333; // should not be used at all
294}
295
296////////////////////////////////////////////////////////////////////////////////
297/// Creates web-based pad painter
298
303
304////////////////////////////////////////////////////////////////////////////////
305/// Returns kTRUE when object is fully supported on JSROOT side
306/// In ROOT7 Paint function will just return appropriate flag that object can be displayed on JSROOT side
307
309{
310 if (!obj)
311 return kTRUE;
312
313 static const struct {
314 const char *name{nullptr};
315 bool with_derived{false};
316 bool reduse_by_many{false};
317 } supported_classes[] = {{"ROOT::Experimental::RTreeMapPainter"},
318 {"TH1", true},
319 {"TF1", true},
320 {"TGraph", true},
321 {"TScatter"},
322 {"TFrame"},
323 {"THStack"},
324 {"TMultiGraph"},
325 {"TGraphPolargram", true},
326 {"TPave", true},
327 {"TGaxis"},
328 {"TEfficiency"},
329 {"TPave", true},
330 {"TButton", true},
331 {"TSlider", true},
332 {"TArrow"},
333 {"TBox", false, true}, // can be handled via TWebPainter, disable for large number of primitives (like in greyscale.C)
334 {"TWbox"}, // some extra calls which cannot be handled via TWebPainter
335 {"TLine", false, true}, // can be handler via TWebPainter, disable for large number of primitives (like in greyscale.C)
336 {"TEllipse", true, true}, // can be handled via TWebPainter, disable for large number of primitives (like in greyscale.C)
337 {"TPie"},
338 {"TText"},
339 {"TLatex"},
340 {"TLink"},
341 {"TAnnotation"},
342 {"TMathText"},
343 {"TMarker"},
344 {"TPolyMarker"},
345 {"TPolyLine", true, true}, // can be handled via TWebPainter, simplify colors handling
346 {"TPolyMarker3D"},
347 {"TPolyLine3D"},
348 {"TGraphTime"},
349 {"TGraph2D"},
350 {"TGraph2DErrors"},
351 {"TGraphTime"},
352 {"TASImage"},
353 {"TRatioPlot"},
354 {"TSpline"},
355 {"TSpline3"},
356 {"TSpline5"},
357 {"TGeoManager"},
358 {"TGeoVolume"},
359 {}};
360
361 // fast check of class name
362 for (int i = 0; supported_classes[i].name != nullptr; ++i)
364 return kTRUE;
365
366 // now check inheritance only for configured classes
367 for (int i = 0; supported_classes[i].name != nullptr; ++i)
369 if (obj->InheritsFrom(supported_classes[i].name))
370 return kTRUE;
371
372 return IsCustomClass(obj->IsA());
373}
374
375//////////////////////////////////////////////////////////////////////////////////////////////////
376/// Configures custom script for canvas.
377/// If started with "modules:" prefix, module(s) will be imported with `loadModules` function of JSROOT.
378/// If custom path was configured in RWebWindowsManager::AddServerLocation, it can be used in module paths.
379/// If started with "load:" prefix, code will be loaded with `loadScript` function of JSROOT (old, deprecated way)
380/// Script also can be a plain JavaScript code which imports JSROOT and provides draw function for custom classes
381/// See tutorials/visualisation/webgui/custom/custom.mjs demonstrating such example
382
383void TWebCanvas::SetCustomScripts(const std::string &src)
384{
386}
387
388//////////////////////////////////////////////////////////////////////////////////////////////////
389/// Returns configured custom script
390
392{
393 return gCustomScripts;
394}
395
396//////////////////////////////////////////////////////////////////////////////////////////////////
397/// For batch mode special handling of scripts are required
398/// Headless browser not able to load modules from the file system
399/// Therefore custom web-canvas modules and scripts has to be loaded in advance and processed
400
402{
403 if (!batch || gCustomScripts.empty() || (gCustomScripts.find("modules:") != 0))
404 return gCustomScripts;
405
407
408 std::string content;
409
410 std::string modules_names = gCustomScripts.substr(8);
411
412 std::map<std::string, bool> mapped_funcs;
413
414 while (!modules_names.empty()) {
415 std::string modname;
416 auto p = modules_names.find(";");
417 if (p == std::string::npos) {
419 modules_names.clear();
420 } else {
421 modname = modules_names.substr(0, p);
422 modules_names = modules_names.substr(p+1);
423 }
424
425 p = modname.find("/");
426 if ((p == std::string::npos) || modname.empty())
427 continue;
428
429 std::string pathname = modname.substr(0, p+1);
430 std::string filename = modname.substr(p+1);
431
432 auto fpath = loc[pathname];
433
434 if (fpath.empty())
435 continue;
436
438 if (cont.empty())
439 continue;
440
441 // check that special mark is in the script
442 auto pmark = cont.find("$$jsroot_batch_conform$$");
443 if (pmark == std::string::npos)
444 continue;
445
446 // process line like this
447 // import { ObjectPainter, addMoveHandler, addDrawFunc, ensureTCanvas } from 'jsroot';
448
449 static const std::string str1 = "import {";
450 static const std::string str2 = "} from 'jsroot';";
451
452 auto p1 = cont.find(str1);
453 auto p2 = cont.find(str2, p1);
454 if ((p1 == std::string::npos) || (p2 == std::string::npos) || (p2 > pmark))
455 continue;
456
457 TString globs;
458
459 TString funcs = cont.substr(p1 + 8, p2 - p1 - 8).c_str();
460 auto arr = funcs.Tokenize(",");
461
462 TIter next(arr);
463 while (auto obj = next()) {
464 TString name = obj->GetName();
465 name = name.Strip(TString::kBoth);
466 if (!mapped_funcs[name.Data()]) {
467 globs.Append(TString::Format("globalThis.%s = JSROOT.%s;\n", name.Data(), name.Data()));
468 mapped_funcs[name.Data()] = true;
469 }
470 }
471 delete arr;
472
473 cont.erase(p1, p2 + str2.length() - p1);
474
475 cont.insert(p1, globs.Data());
476
477 content.append(cont);
478 }
479
480 return content;
481}
482
483
484//////////////////////////////////////////////////////////////////////////////////////////////////
485/// Assign custom class
486
487void TWebCanvas::AddCustomClass(const std::string &clname, bool with_derived)
488{
489 if (with_derived)
490 gCustomClasses.emplace_back("+"s + clname);
491 else
492 gCustomClasses.emplace_back(clname);
493}
494
495//////////////////////////////////////////////////////////////////////////////////////////////////
496/// Checks if class belongs to custom
497
499{
500 for (auto &name : gCustomClasses) {
501 if (name[0] == '+') {
502 if (cl->InheritsFrom(name.substr(1).c_str()))
503 return true;
504 } else if (name.compare(cl->GetName()) == 0) {
505 return true;
506 }
507 }
508 return false;
509}
510
511//////////////////////////////////////////////////////////////////////////////////////////////////
512/// Creates representation of the object for painting in web browser
513
515{
516 if (IsJSSupportedClass(obj, masterps != nullptr)) {
517 master.NewPrimitive(obj, opt).SetSnapshot(TWebSnapshot::kObject, obj);
518 return;
519 }
520
521 // painter is not necessary for batch canvas, but keep configuring it for a while
522 auto *painter = dynamic_cast<TWebPadPainter *>(Canvas()->GetCanvasPainter());
523
524 TView *view = nullptr;
525
527
528 gPad = pad;
529
530 if (obj->InheritsFrom(TAtt3D::Class()) && !pad->GetView()) {
531 pad->GetViewer3D("pad");
532 view = TView::CreateView(1, 0, 0); // Cartesian view by default
533 pad->SetView(view);
534
535 // Set view to perform first auto-range (scaling) pass
536 view->SetAutoRange(kTRUE);
537 }
538
540
541 TWebPS ps;
542 ps.GetPainting()->SetClassName(obj->ClassName());
543 ps.GetPainting()->SetObjectName(obj->GetName());
544 gVirtualPS = masterps ? masterps : &ps;
545 if (painter)
546 painter->SetPainting(ps.GetPainting());
547
548 // calling Paint function for the object
549 obj->Paint(opt);
550
551 if (view) {
552 view->SetAutoRange(kFALSE);
553 // call 3D paint once again to make real drawing
554 obj->Paint(opt);
555 pad->SetView(nullptr);
556 }
557
558 if (painter)
559 painter->SetPainting(nullptr);
560
562
563 fPadsStatus[pad]._has_specials = true;
564
565 // if there are master PS, do not create separate entries
566 if (!masterps && !ps.IsEmptyPainting())
567 master.NewPrimitive(obj, opt).SetSnapshot(TWebSnapshot::kSVG, ps.TakePainting(), kTRUE);
568}
569
570//////////////////////////////////////////////////////////////////////////////////////////////////
571/// Calculate hash function for all colors and palette
572
574{
575 UInt_t hash = 0;
576
577 TObjArray *colors = (TObjArray *)gROOT->GetListOfColors();
578
579 if (colors) {
580 for (Int_t n = 0; n <= colors->GetLast(); ++n)
581 if (colors->At(n))
582 hash += TString::Hash(colors->At(n), TColor::Class()->Size());
583 }
584
586
587 hash += TString::Hash(pal.GetArray(), pal.GetSize() * sizeof(Int_t));
588
589 return hash;
590}
591
592
593//////////////////////////////////////////////////////////////////////////////////////////////////
594/// Add special canvas objects with list of colors and color palette
595
597{
598 TObjArray *colors = (TObjArray *)gROOT->GetListOfColors();
599
600 if (!colors)
601 return;
602
603 //Int_t cnt = 0;
604 //for (Int_t n = 0; n <= colors->GetLast(); ++n)
605 // if (colors->At(n))
606 // cnt++;
607 //if (cnt <= 598)
608 // return; // normally there are 598 colors defined
609
611
612 auto listofcols = new TWebPainting;
613 for (Int_t n = 0; n <= colors->GetLast(); ++n)
614 listofcols->AddColor(n, (TColor *)colors->At(n));
615
616 // store palette in the buffer
617 auto *tgt = listofcols->Reserve(pal.GetSize());
618 for (Int_t i = 0; i < pal.GetSize(); i++)
619 tgt[i] = pal[i];
620 listofcols->FixSize();
621
622 master.NewSpecials().SetSnapshot(TWebSnapshot::kColors, listofcols, kTRUE);
623}
624
625//////////////////////////////////////////////////////////////////////////////////////////////////
626/// Add special canvas objects with custom fonts
627
629{
630 for (auto &entry : gWebFonts) {
631 TString code = TString::Format("%d:%s:%s:%s", entry.fIndx, entry.fName.Data(), entry.fFormat.Data(), entry.fData.Data());
632 auto custom_font = new TWebPainting;
633 custom_font->AddOper(code.Data());
634 master.NewSpecials().SetSnapshot(TWebSnapshot::kFont, custom_font, kTRUE);
635 }
636}
637
638//////////////////////////////////////////////////////////////////////////////////////////////////
639/// Create snapshot for pad and all primitives
640/// Callback function is used to create JSON in the middle of data processing -
641/// when all misc objects removed from canvas list of primitives or histogram list of functions
642/// After that objects are moved back to their places
643
645{
646 auto &pad_status = fPadsStatus[pad];
647
648 // send primitives if version 0 or actual pad version grater than already send version
649 bool process_primitives = (version == 0) || (pad_status.fVersion > version);
650
651 if (paddata.IsSetObjectIds()) {
652 paddata.SetActive(pad == gPad);
653 paddata.SetObjectIDAsPtr(pad);
654 }
655 paddata.SetSnapshot(TWebSnapshot::kSubPad, pad); // add ref to the pad
656 paddata.SetWithoutPrimitives(!process_primitives);
657 paddata.SetHasExecs(pad->GetListOfExecs()); // if pad execs are there provide more events from client
658
659 // check style changes every time when creating canvas snapshot
660 if (resfunc && (GetStyleDelivery() > 0)) {
661
663 auto hash = TString::Hash(gStyle, TStyle::Class()->Size());
664 if ((hash != fStyleHash) || (fStyleVersion == 0)) {
667 }
668 }
669
671 paddata.NewPrimitive().SetSnapshot(TWebSnapshot::kStyle, gStyle);
672 }
673
674 // for the first time add custom fonts to the canvas snapshot
675 if (resfunc && (version == 0))
677
678 fAllPads.emplace_back(pad);
679
680 TList *primitives = pad->GetListOfPrimitives();
681
683 bool usemaster = primitives ? (primitives->GetSize() > fPrimitivesMerge) : false;
684
685 TIter iter(primitives);
686 TObject *obj = nullptr;
687 TFrame *frame = nullptr;
688 TPaveText *title = nullptr;
689 TGraphPolar *first_polar = nullptr;
690 TGraphPolargram *polargram = nullptr;
692 bool need_frame = false, has_histo = false, need_palette = false;
693 std::string need_title;
694
695 auto checkNeedPalette = [](TH1* hist, const TString &opt) {
696 auto check = [&opt](const TString &arg) {
697 return opt.Contains(arg + "Z") || opt.Contains(arg + "HZ");
698 };
699
700 return ((hist->GetDimension() == 2) && (check("COL") || check("LEGO") || check("LEGO4") || check("SURF2"))) ||
701 ((hist->GetDimension() == 3) && (check("BOX2") || check("BOX3")));
702 };
703
704 while (process_primitives && ((obj = iter()) != nullptr)) {
705 TString opt = iter.GetOption();
706 opt.ToUpper();
707
708 if (obj->InheritsFrom(THStack::Class())) {
709 // workaround for THStack, create extra components before sending to client
710 if (!opt.Contains("PADS") && !opt.Contains("SAME")) {
712
713 auto hs = static_cast<THStack *>(obj);
714
715 if (!opt.Contains("NOSTACK") && !opt.Contains("CANDLE") && !opt.Contains("VIOLIN") && !IsReadOnly() && !fUsedObjs[hs]) {
717 fUsedObjs[hs] = true;
718 }
719
720 if (strlen(obj->GetTitle()) > 0)
721 need_title = obj->GetTitle();
723 hs->BuildPrimitives(iter.GetOption(), do_rebuild_stack);
724 has_histo = true;
725 need_frame = true;
726 }
727 } else if (obj->InheritsFrom(TMultiGraph::Class())) {
728 // workaround for TMultiGraph
729 if (opt.Contains("A")) {
730 auto mg = static_cast<TMultiGraph *>(obj);
732 mg->GetHistogram(); // force creation of histogram without any drawings
733 has_histo = true;
734 if (strlen(obj->GetTitle()) > 0)
735 need_title = obj->GetTitle();
736 need_frame = true;
737 }
738 } else if (obj->InheritsFrom(TFrame::Class())) {
739 if (!frame)
740 frame = static_cast<TFrame *>(obj);
741 } else if (obj->InheritsFrom(TH1::Class())) {
742 need_frame = true;
743 has_histo = true;
744 if (!obj->TestBit(TH1::kNoTitle) && !opt.Contains("SAME") && !opt.Contains("AXIS") && !opt.Contains("AXIG") && (strlen(obj->GetTitle()) > 0))
745 need_title = obj->GetTitle();
746 if (checkNeedPalette(static_cast<TH1*>(obj), opt))
747 need_palette = true;
748 } else if (obj->InheritsFrom(TGraphPolar::Class())) {
749 auto polar = static_cast<TGraphPolar *> (obj);
750 if (!first_polar) {
752 need_title = first_polar->GetTitle();
753 polargram = first_polar->GetPolargram();
754 if (!polargram) {
755 polargram = first_polar->CreatePolargram(opt);
756 polargram_drawopt = opt.Contains("N") ? "N" : "";
757 if (opt.Contains("O")) polargram_drawopt.Append("O");
758 }
759 }
760 polar->SetPolargram(polargram);
761 } else if (obj->InheritsFrom(TGraph::Class())) {
762 if (opt.Contains("A")) {
763 need_frame = true;
764 if (!has_histo && (strlen(obj->GetTitle()) > 0) && !obj->TestBit(TH1::kNoTitle))
765 need_title = obj->GetTitle();
766 }
767 } else if (obj->InheritsFrom(TGraph2D::Class())) {
768 if (!has_histo && (strlen(obj->GetTitle()) > 0))
769 need_title = obj->GetTitle();
770 } else if (obj->InheritsFrom(TScatter::Class())) {
771 need_frame = need_palette = true;
772 if (strlen(obj->GetTitle()) > 0)
773 need_title = obj->GetTitle();
774 } else if (obj->InheritsFrom(TF1::Class())) {
775 if (!opt.Contains("SAME")) {
777 if (!has_histo && (strlen(obj->GetTitle()) > 0))
778 need_title = obj->GetTitle();
779 }
780 } else if (obj->InheritsFrom(TPaveText::Class())) {
781 if (strcmp(obj->GetName(), "title") == 0)
782 title = static_cast<TPaveText *>(obj);
783 } else if (obj->InheritsFrom(TButton::Class())) {
784 auto btn = (TButton *) obj;
785 auto text = dynamic_cast<TText *> (btn->GetListOfPrimitives()->First());
786 if (text) {
787 text->SetTitle(btn->GetTitle());
788 text->SetTextSize(btn->GetTextSize());
789 text->SetTextFont(btn->GetTextFont());
790 text->SetTextAlign(btn->GetTextAlign());
791 text->SetTextColor(btn->GetTextColor());
792 text->SetTextAngle(btn->GetTextAngle());
793 }
794 }
795 }
796
797 if (need_frame && !frame && primitives && CanCreateObject("TFrame")) {
798 if (!IsReadOnly() && need_palette && (pad->GetRightMargin() < 0.12) && (pad->GetRightMargin() == gStyle->GetPadRightMargin()))
799 pad->SetRightMargin(0.12);
800
801 frame = pad->GetFrame();
802 if(frame)
803 primitives->AddFirst(frame, "");
804 }
805
806 if (!need_title.empty() && gStyle->GetOptTitle()) {
807 if (title) {
808 auto line0 = title->GetLine(0);
809 if (line0 && !IsReadOnly()) line0->SetTitle(need_title.c_str());
810 } else if (primitives && CanCreateObject("TPaveText")) {
811 title = new TPaveText(0, 0, 0, 0, "blNDC");
814 title->SetName("title");
817 title->SetTextFont(gStyle->GetTitleFont(""));
818 if (gStyle->GetTitleFont("") % 10 > 2)
820 title->AddText(need_title.c_str());
821 title->SetBit(kCanDelete);
822 primitives->Add(title, title->GetOption());
823 }
824 }
825
826 if (polargram && (polargram_drawopt != "-"))
827 primitives->Add(polargram, polargram_drawopt);
828
829 auto flush_master = [&]() {
830 if (!usemaster || masterps.IsEmptyPainting()) return;
831
832 paddata.NewPrimitive(pad).SetSnapshot(TWebSnapshot::kSVG, masterps.TakePainting(), kTRUE);
833 masterps.CreatePainting(); // create for next operations
834 };
835
836 auto check_cutg_in_options = [&](const TString &opt) {
837 auto p1 = opt.Index("["), p2 = opt.Index("]");
838 if ((p1 != kNPOS) && (p2 != kNPOS) && p2 > p1 + 1) {
839 TString cutname = opt(p1 + 1, p2 - p1 - 1);
840 TObject *cutg = primitives->FindObject(cutname.Data());
841 if (!cutg || (cutg->IsA() != TCutG::Class())) {
842 cutg = gROOT->GetListOfSpecials()->FindObject(cutname.Data());
843 if (cutg && cutg->IsA() == TCutG::Class())
844 paddata.NewPrimitive(cutg, "__ignore_drawing__").SetSnapshot(TWebSnapshot::kObject, cutg);
845 }
846 }
847 };
848
849 auto check_save_tf1 = [&](TObject *fobj, bool ignore_nodraw = false) {
850 if (!paddata.IsBatchMode() && (fTF1UseSave <= 0))
851 return;
852 if (!ignore_nodraw && fobj->TestBit(TF1::kNotDraw))
853 return;
854
855 auto f1 = static_cast<TF1 *>(fobj);
856 // check if TF1 can be used
857 if (!f1->IsValid())
858 return;
859
860 // in default case save buffer used as is
861 if ((fTF1UseSave == 1) && f1->HasSave())
862 return;
863
864 auto f3 = dynamic_cast<TF3 *>(f1);
865 auto f2 = dynamic_cast<TF2 *>(f1);
866 if (f3)
867 f3->Save(f3->GetXmin(), f3->GetXmax(), f3->GetYmin(), f3->GetYmax(), f3->GetZmin(), f3->GetZmax());
868 else if (f2)
869 f2->Save(f2->GetXmin(), f2->GetXmax(), f2->GetYmin(), f2->GetYmax(), 0, 0);
870 else
871 f1->Save(f1->GetXmin(), f1->GetXmax(), 0, 0, 0, 0);
872 };
873
874 auto create_stats = [&]() {
875 TPaveStats *stats = nullptr;
876 if (CanCreateObject("TPaveStats")) {
877 stats = new TPaveStats(
880 gStyle->GetStatX(),
881 gStyle->GetStatY(), "brNDC");
882
883 // do not set optfit and optstat, they calling pad->Update,
884 // values correctly set already in TPaveStats constructor
885 // stats->SetOptFit(gStyle->GetOptFit());
886 // stats->SetOptStat(gStyle->GetOptStat());
890 stats->SetTextFont(gStyle->GetStatFont());
891 if (gStyle->GetStatFont()%10 > 2)
895 stats->SetName("stats");
896
898 stats->SetTextAlign(12);
899 stats->SetBit(kCanDelete);
900 stats->SetBit(kMustCleanup);
901 }
902
903 return stats;
904 };
905
906 auto check_graph_funcs = [&](TGraph *gr, TList *funcs = nullptr) {
907 if (!funcs && gr)
909 if (!funcs)
910 return;
911
913 TPaveStats *stats = nullptr;
914 bool has_tf1 = false;
915
916 while (auto fobj = fiter()) {
917 if (fobj->InheritsFrom(TPaveStats::Class()))
918 stats = dynamic_cast<TPaveStats *> (fobj);
919 else if (fobj->InheritsFrom(TF1::Class())) {
921 has_tf1 = true;
922 }
923 }
924
925 if (!stats && has_tf1 && gr && !gr->TestBit(TGraph::kNoStats) && (gStyle->GetOptFit() > 0)) {
926 stats = create_stats();
927 if (stats) {
928 stats->SetOptStat(0);
929 stats->SetOptFit(gStyle->GetOptFit());
930 stats->SetParent(funcs);
931 funcs->Add(stats);
932 }
933 }
934 };
935
936 iter.Reset();
937
938 bool first_obj = true;
939
941 pad_status._has_specials = false;
942
943 while ((obj = iter()) != nullptr) {
944 if (obj->IsA() == TPad::Class()) {
945 flush_master();
946 CreatePadSnapshot(paddata.NewSubPad(), (TPad *)obj, version, nullptr);
947 } else if (!process_primitives) {
948 continue;
949 } else if (obj->InheritsFrom(TH1::Class())) {
950 flush_master();
951
952 TH1 *hist = static_cast<TH1 *>(obj);
953 hist->BufferEmpty();
954
955 TPaveStats *stats = nullptr;
956 TObject *palette = nullptr;
957
959 while (auto fobj = fiter()) {
960 if (fobj->InheritsFrom(TPaveStats::Class()))
961 stats = dynamic_cast<TPaveStats *> (fobj);
962 else if (fobj->InheritsFrom("TPaletteAxis"))
963 palette = fobj;
964 else if (fobj->InheritsFrom(TF1::Class()))
966 }
967
968 TString hopt = iter.GetOption();
969 TString o = hopt;
970 o.ToUpper();
971
972 if (!stats && (first_obj || o.Contains("SAMES")) && (gStyle->GetOptStat() > 0)) {
973 stats = create_stats();
974 if (stats) {
975 stats->SetParent(hist);
976 hist->GetListOfFunctions()->Add(stats);
977 }
978 }
979
980 if (!palette && CanCreateObject("TPaletteAxis") && checkNeedPalette(hist, o)) {
981 std::stringstream exec;
982 exec << "new TPaletteAxis(0,0,0,0, (TH1*)" << std::hex << std::showbase << (size_t)hist << ");";
983 palette = (TObject *)gROOT->ProcessLine(exec.str().c_str());
984 if (palette)
986 }
987
988 paddata.NewPrimitive(obj, hopt.Data()).SetSnapshot(TWebSnapshot::kObject, obj);
989
990 if (hist->GetDimension() == 2)
992
993 first_obj = false;
994 } else if (obj->InheritsFrom(TGraphPolar::Class())) {
995 flush_master();
996
997 auto polar = static_cast<TGraphPolar *>(obj);
998
1000
1001 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1002
1003 first_obj = false;
1004 } else if (obj->InheritsFrom(TGraphPolargram::Class())) {
1005 // do nothing, object must be streamed with graphpolar
1006 } else if (obj->InheritsFrom(TGraph::Class())) {
1007 flush_master();
1008
1009 TGraph *gr = static_cast<TGraph *>(obj);
1010
1012
1013 TString gropt = iter.GetOption();
1014
1015 // ensure histogram exists on server to draw it properly on clients side
1016 if (!IsReadOnly() && (first_obj || gropt.Index("A", 0, TString::kIgnoreCase) != kNPOS ||
1017 (gropt.Index("X+", 0, TString::kIgnoreCase) != kNPOS) || (gropt.Index("Y+", 0, TString::kIgnoreCase) != kNPOS)))
1018 gr->GetHistogram();
1019
1020 paddata.NewPrimitive(obj, gropt.Data()).SetSnapshot(TWebSnapshot::kObject, obj);
1021
1022 first_obj = false;
1023 } else if (obj->InheritsFrom(TGraph2D::Class())) {
1024 flush_master();
1025
1026 TGraph2D *gr2d = static_cast<TGraph2D *>(obj);
1027
1028 check_graph_funcs(nullptr, gr2d->GetListOfFunctions());
1029
1030 // ensure correct range of histogram
1031 if (!IsReadOnly() && first_obj) {
1032 TString gropt = iter.GetOption();
1033 gropt.ToUpper();
1034 Bool_t zscale = gropt.Contains("TRI1") || gropt.Contains("TRI2") || gropt.Contains("COL");
1035 Bool_t cont5_draw = gropt.Contains("CONT5");
1036 Bool_t real_draw = gropt.Contains("TRI") || gropt.Contains("LINE") || gropt.Contains("ERR") || gropt.Contains("P") || cont5_draw;
1037
1038 TString hopt = !real_draw ? iter.GetOption() : (cont5_draw ? "" : (zscale ? "lego2z" : "lego2"));
1039 if (title) hopt.Append(";;use_pad_title");
1040
1041 // if gr2d not draw - let create histogram with correspondent content
1042 auto hist = gr2d->GetHistogram(real_draw ? "empty" : "");
1043
1044 paddata.NewPrimitive(gr2d, hopt.Data(), "#hist").SetSnapshot(TWebSnapshot::kObject, hist);
1045 }
1046
1047 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1048 first_obj = false;
1049 } else if (obj->InheritsFrom(TMultiGraph::Class())) {
1050 flush_master();
1051
1052 TMultiGraph *mgr = static_cast<TMultiGraph *>(obj);
1053 TIter fiter(mgr->GetListOfFunctions());
1054 while (auto fobj = fiter()) {
1055 if (fobj->InheritsFrom(TF1::Class()))
1057 }
1058
1059 TIter giter(mgr->GetListOfGraphs());
1060 while (auto gobj = giter())
1061 check_graph_funcs(static_cast<TGraph *>(gobj));
1062
1063 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1064
1065 first_obj = false;
1066 } else if (obj->InheritsFrom(THStack::Class())) {
1067 flush_master();
1068
1069 THStack *hs = static_cast<THStack *>(obj);
1070
1071 TString hopt = iter.GetOption();
1072 hopt.ToLower();
1073 if (!hopt.Contains("nostack") && !hopt.Contains("candle") && !hopt.Contains("violin") && !hopt.Contains("pads")) {
1074 auto arr = hs->GetStack();
1075 arr->SetName(hs->GetName()); // mark list for JS
1076 paddata.NewPrimitive(arr, "__ignore_drawing__").SetSnapshot(TWebSnapshot::kObject, arr);
1077 }
1078
1079 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1080
1081 first_obj = hs->GetNhists() > 0; // real drawing only if there are histograms
1082 } else if (obj->InheritsFrom(TScatter::Class())) {
1083 flush_master();
1084
1085 TScatter *scatter = static_cast<TScatter *>(obj);
1086
1087 TObject *palette = nullptr;
1088
1089 TIter fiter(scatter->GetGraph()->GetListOfFunctions());
1090 while (auto fobj = fiter()) {
1091 if (fobj->InheritsFrom("TPaletteAxis"))
1092 palette = fobj;
1093 }
1094
1095 // ensure histogram exists on server to draw it properly on clients side
1096 if (!IsReadOnly() && first_obj)
1097 scatter->GetHistogram();
1098
1099 if (!palette && CanCreateObject("TPaletteAxis")) {
1100 std::stringstream exec;
1101 exec << "new TPaletteAxis(0,0,0,0,0,0);";
1102 palette = (TObject *)gROOT->ProcessLine(exec.str().c_str());
1103 if (palette)
1104 scatter->GetGraph()->GetListOfFunctions()->AddFirst(palette);
1105 }
1106
1107 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1108
1109 first_obj = false;
1110 } else if (obj->InheritsFrom(TF1::Class())) {
1111 flush_master();
1112 auto f1 = static_cast<TF1 *> (obj);
1113
1114 TString f1opt = iter.GetOption();
1115
1116 check_save_tf1(obj, true);
1117 if (fTF1UseSave > 1)
1118 f1opt.Append(";force_saved");
1119 else if (fTF1UseSave == 1)
1120 f1opt.Append(";prefer_saved");
1121
1122 if (first_obj) {
1123 auto hist = f1->GetHistogram();
1124 paddata.NewPrimitive(hist, "__ignore_drawing__").SetSnapshot(TWebSnapshot::kObject, hist);
1125 f1opt.Append(";webcanv_hist");
1126 }
1127
1128 if (f1->IsA() == TF2::Class())
1130
1131 paddata.NewPrimitive(f1, f1opt.Data()).SetSnapshot(TWebSnapshot::kObject, f1);
1132
1133 first_obj = false;
1134
1135 } else if (obj->InheritsFrom(TGaxis::Class())) {
1136 flush_master();
1137 auto gaxis = static_cast<TGaxis *> (obj);
1138 auto func = gaxis->GetFunction();
1139 if (func)
1140 paddata.NewPrimitive(func, "__ignore_drawing__").SetSnapshot(TWebSnapshot::kObject, func);
1141
1142 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1143 } else if (obj->InheritsFrom(TFrame::Class())) {
1144 flush_master();
1145 if (frame && (obj == frame)) {
1146 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1147 frame = nullptr; // add frame only once
1148 }
1149 } else if (IsJSSupportedClass(obj, usemaster)) {
1150 flush_master();
1151 paddata.NewPrimitive(obj, iter.GetOption()).SetSnapshot(TWebSnapshot::kObject, obj);
1152 } else {
1153 CreateObjectSnapshot(paddata, pad, obj, iter.GetOption(), usemaster ? &masterps : nullptr);
1154 }
1155 }
1156
1157 flush_master();
1158
1159 bool provide_colors = false;
1160
1161 if ((GetPaletteDelivery() > 2) || ((GetPaletteDelivery() == 2) && resfunc)) {
1162 // provide colors: either for each subpad (> 2) or only for canvas (== 2)
1164 } else if ((GetPaletteDelivery() == 1) && resfunc) {
1165 // check that colors really changing, using hash
1166
1168 auto hash = CalculateColorsHash();
1169 if ((hash != fColorsHash) || (fColorsVersion == 0)) {
1170 fColorsHash = hash;
1172 }
1173 }
1174
1176 }
1177
1178 // add colors after painting is performed - new colors may be generated only during painting
1179 if (provide_colors)
1181
1182 if (!resfunc)
1183 return;
1184
1185 // now hide all primitives to perform I/O
1186 std::vector<TList *> all_primitives(fAllPads.size());
1187 for (unsigned n = 0; n < fAllPads.size(); ++n) {
1188 all_primitives[n] = fAllPads[n]->fPrimitives;
1189 fAllPads[n]->fPrimitives = nullptr;
1190 }
1191
1192 // execute function to prevent storing of colors with custom TCanvas streamer
1194
1195 // invoke callback for streaming
1196 resfunc(&paddata);
1197
1198 // and restore back primitives - delete any temporary if necessary
1199 for (unsigned n = 0; n < fAllPads.size(); ++n) {
1200 if (fAllPads[n]->fPrimitives)
1201 delete fAllPads[n]->fPrimitives;
1202 fAllPads[n]->fPrimitives = all_primitives[n];
1203 }
1204 fAllPads.clear();
1205 fUsedObjs.clear();
1206}
1207
1208//////////////////////////////////////////////////////////////////////////////////////////////////
1209/// Add control message for specified connection
1210/// Same control message can be overwritten many time before it really sends to the client
1211/// If connid == 0, message will be add to all connections
1212/// After ctrl message is add to the output, short timer is activated and message send afterwards
1213
1214void TWebCanvas::AddCtrlMsg(unsigned connid, const std::string &key, const std::string &value)
1215{
1217
1218 for (auto &conn : fWebConn) {
1219 if (conn.match(connid)) {
1220 conn.fCtrl[key] = value;
1221 new_ctrl = kTRUE;
1222 }
1223 }
1224
1225 if (new_ctrl && fTimer->IsSlow())
1227}
1228
1229
1230//////////////////////////////////////////////////////////////////////////////////////////////////
1231/// Add message to send queue for specified connection
1232/// If connid == 0, message will be add to all connections
1233
1234void TWebCanvas::AddSendQueue(unsigned connid, const std::string &msg)
1235{
1236 for (auto &conn : fWebConn) {
1237 if (conn.match(connid))
1238 conn.fSend.emplace(msg);
1239 }
1240}
1241
1242
1243//////////////////////////////////////////////////////////////////////////////////////////////////
1244/// Check if any data should be send to client
1245/// If connid != 0, only selected connection will be checked
1246
1248{
1249 if (!Canvas())
1250 return kFALSE;
1251
1252 bool isMoreData = false, isAnySend = false;
1253
1254 for (auto &conn : fWebConn) {
1255
1256 bool isConnData = !conn.fCtrl.empty() || !conn.fSend.empty() ||
1257 ((conn.fCheckedVersion < fCanvVersion) && (conn.fSendVersion == conn.fDrawVersion));
1258
1259 while ((conn.is_batch() && !connid) || (conn.match(connid) && fWindow && fWindow->CanSend(conn.fConnId, true))) {
1260 // check if any control messages still there to keep timer running
1261
1262 std::string buf;
1263
1264 if (!conn.fCtrl.empty()) {
1266 conn.fCtrl.clear();
1267 } else if (!conn.fSend.empty()) {
1268 std::swap(buf, conn.fSend.front());
1269 conn.fSend.pop();
1270 } else if ((conn.fCheckedVersion < fCanvVersion) && (conn.fSendVersion == conn.fDrawVersion)) {
1271
1272 buf = "SNAP6:"s + std::to_string(fCanvVersion) + ":"s;
1273
1274 TCanvasWebSnapshot holder(IsReadOnly(), true, false); // readonly, set ids, batchmode
1275
1276 holder.SetFixedSize(fFixedSize); // set fixed size flag
1277
1278 // scripts send only when canvas drawn for the first time
1279 if (!conn.fSendVersion)
1280 holder.SetScripts(ProcessCustomScripts(false));
1281
1282 holder.SetHighlightConnect(Canvas()->HasConnection("Highlighted(TVirtualPad*,TObject*,Int_t,Int_t)"));
1283
1284 CreatePadSnapshot(holder, Canvas(), conn.fSendVersion, [&buf, &conn, this](TPadWebSnapshot *snap) {
1285 if (conn.is_batch()) {
1286 // for batch connection only calling of CreatePadSnapshot is important
1287 buf.clear();
1288 return;
1289 }
1290
1292 auto hash = json.Hash();
1293 if (conn.fLastSendHash && (conn.fLastSendHash == hash) && conn.fSendVersion) {
1294 // prevent looping when same data send many times
1295 buf.clear();
1296 } else {
1297 buf.append(json.Data());
1298 conn.fLastSendHash = hash;
1299 }
1300 });
1301
1302 conn.fCheckedVersion = fCanvVersion;
1303
1304 conn.fSendVersion = fCanvVersion;
1305
1306 if (buf.empty())
1307 conn.fDrawVersion = fCanvVersion;
1308 } else {
1309 isConnData = false;
1310 break;
1311 }
1312
1313 if (!buf.empty() && !conn.is_batch()) {
1314 fWindow->Send(conn.fConnId, buf);
1315 isAnySend = true;
1316 }
1317 }
1318
1319 if (isConnData)
1320 isMoreData = true;
1321 }
1322
1323 if (fTimer->IsSlow() && isMoreData)
1324 fTimer->SetSlow(kFALSE);
1325
1326 return isAnySend;
1327}
1328
1329//////////////////////////////////////////////////////////////////////////////////////////
1330/// Close web canvas - not implemented
1331
1333{
1334}
1335
1336//////////////////////////////////////////////////////////////////////////////////////////
1337/// Create web window for the canvas
1338
1340{
1341 if (fWindow)
1342 return;
1343
1345
1346 fWindow->SetConnLimit(0); // configure connections limit
1347
1348 fWindow->SetDefaultPage("file:rootui5sys/canv/canvas6.html");
1349
1350 fWindow->SetCallBacks(
1351 // connection
1352 [this](unsigned connid) {
1353 if (fWindow->GetConnectionId(0) == connid)
1354 fWebConn.emplace(fWebConn.begin() + 1, connid);
1355 else
1356 fWebConn.emplace_back(connid);
1357 CheckDataToSend(connid);
1358 },
1359 // data
1360 [this](unsigned connid, const std::string &arg) {
1361 ProcessData(connid, arg);
1363 },
1364 // disconnect
1365 [this](unsigned connid) {
1366 unsigned indx = 0;
1367 for (auto &c : fWebConn) {
1368 if (c.fConnId == connid) {
1369 fWebConn.erase(fWebConn.begin() + indx);
1370 break;
1371 }
1372 indx++;
1373 }
1374 });
1375}
1376
1377//////////////////////////////////////////////////////////////////////////////////////////
1378/// Show canvas in specified place.
1379/// If parameter args not specified, default ROOT web display will be used
1380
1382{
1384
1387
1388 auto w = Canvas()->GetWindowWidth(), h = Canvas()->GetWindowHeight();
1389 if ((w > 0) && (w < 50000) && (h > 0) && (h < 30000))
1390 fWindow->SetGeometry(w, h);
1391
1393}
1394
1395//////////////////////////////////////////////////////////////////////////////////////////
1396/// Show canvas in browser window
1397
1399{
1400 if (gROOT->IsWebDisplayBatch())
1401 return;
1402
1403 if (fWindow && !fWindow->HasConnection(0))
1404 fLastDrawVersion = 0;
1405
1407 args.SetWidgetKind("TCanvas");
1408 args.SetSize(Canvas()->GetWindowWidth(), Canvas()->GetWindowHeight());
1409 args.SetPos(Canvas()->GetWindowTopX(), Canvas()->GetWindowTopY());
1410
1411 ShowWebWindow(args);
1412}
1413
1414//////////////////////////////////////////////////////////////////////////////////////////
1415/// Function used to send command to browser to toggle menu, toolbar, editors, ...
1416
1417void TWebCanvas::ShowCmd(const std::string &arg, Bool_t show)
1418{
1419 AddCtrlMsg(0, arg, show ? "1"s : "0"s);
1420}
1421
1422//////////////////////////////////////////////////////////////////////////////////////////
1423/// Activate object in editor in web browser
1424
1426{
1427 if (!pad || !obj) return;
1428
1429 UInt_t hash = TString::Hash(&obj, sizeof(obj));
1430
1431 AddCtrlMsg(0, "edit"s, std::to_string(hash));
1432}
1433
1434//////////////////////////////////////////////////////////////////////////////////////////
1435/// Returns kTRUE if web canvas has graphical editor
1436
1438{
1439 return (fClientBits & TCanvas::kShowEditor) != 0;
1440}
1441
1442//////////////////////////////////////////////////////////////////////////////////////////
1443/// Returns kTRUE if web canvas has menu bar
1444
1446{
1447 return (fClientBits & TCanvas::kMenuBar) != 0;
1448}
1449
1450//////////////////////////////////////////////////////////////////////////////////////////
1451/// Returns kTRUE if web canvas has status bar
1452
1457
1458//////////////////////////////////////////////////////////////////////////////////////////
1459/// Returns kTRUE if tooltips are activated in web canvas
1460
1462{
1463 return (fClientBits & TCanvas::kShowToolTips) != 0;
1464}
1465
1466//////////////////////////////////////////////////////////////////////////////////////////
1467/// Set window position of web canvas
1468
1470{
1471 AddCtrlMsg(0, "x"s, std::to_string(x));
1472 AddCtrlMsg(0, "y"s, std::to_string(y));
1473}
1474
1475//////////////////////////////////////////////////////////////////////////////////////////
1476/// Set window size of web canvas
1477
1479{
1480 AddCtrlMsg(0, "w"s, std::to_string(w));
1481 AddCtrlMsg(0, "h"s, std::to_string(h));
1482}
1483
1484//////////////////////////////////////////////////////////////////////////////////////////
1485/// Set window title of web canvas
1486
1488{
1489 AddCtrlMsg(0, "title"s, newTitle);
1490}
1491
1492//////////////////////////////////////////////////////////////////////////////////////////
1493/// Set canvas size of web canvas
1494
1496{
1497 fFixedSize = kTRUE;
1498 AddCtrlMsg(0, "cw"s, std::to_string(cw));
1499 AddCtrlMsg(0, "ch"s, std::to_string(ch));
1500 if ((cw > 0) && (ch > 0)) {
1501 Canvas()->fCw = cw;
1502 Canvas()->fCh = ch;
1503 } else {
1504 // temporary value, will be reported back from client
1505 Canvas()->fCw = Canvas()->fWindowWidth;
1507 }
1508}
1509
1510//////////////////////////////////////////////////////////////////////////////////////////
1511/// Iconify browser window
1512
1514{
1515 AddCtrlMsg(0, "winstate"s, "iconify"s);
1516}
1517
1518//////////////////////////////////////////////////////////////////////////////////////////
1519/// Raise browser window
1520
1522{
1523 AddCtrlMsg(0, "winstate"s, "raise"s);
1524}
1525
1526//////////////////////////////////////////////////////////////////////////////////////////
1527/// Assign clients bits
1528
1537
1538//////////////////////////////////////////////////////////////////////////////////////////////////
1539/// Decode all pad options, which includes ranges plus objects options
1540
1542{
1543 if (IsReadOnly() || msg.empty())
1544 return kFALSE;
1545
1546 auto arr = TBufferJSON::FromJSON<std::vector<TWebPadOptions>>(msg);
1547
1548 if (!arr)
1549 return kFALSE;
1550
1552
1553 TPad *pad_with_execs = nullptr;
1554 TExec *hist_exec = nullptr;
1555
1556 for (unsigned n = 0; n < arr->size(); ++n) {
1557 auto &r = arr->at(n);
1558
1559 TPad *pad = dynamic_cast<TPad *>(FindPrimitive(r.snapid));
1560
1561 if (!pad)
1562 continue;
1563
1564 if (pad == Canvas()) {
1565 AssignStatusBits(r.bits);
1566 Canvas()->fCw = r.cw;
1567 Canvas()->fCh = r.ch;
1568 if (r.w.size() == 4)
1570 }
1571
1572 // only if get OPTIONS message from client allow to change gPad
1573 if (r.active && (pad != gPad) && process_execs)
1574 gPad = pad;
1575
1576 if ((pad->GetTickx() != r.tickx) || (pad->GetTicky() != r.ticky))
1577 pad->SetTicks(r.tickx, r.ticky);
1578 if ((pad->GetGridx() != (r.gridx > 0)) || (pad->GetGridy() != (r.gridy > 0)))
1579 pad->SetGrid(r.gridx, r.gridy);
1580 pad->fLogx = r.logx;
1581 pad->fLogy = r.logy;
1582 pad->fLogz = r.logz;
1583
1584 pad->SetLeftMargin(r.mleft);
1585 pad->SetRightMargin(r.mright);
1586 pad->SetTopMargin(r.mtop);
1587 pad->SetBottomMargin(r.mbottom);
1588
1589 if (r.ranges) {
1590 // avoid call of original methods, set members directly
1591 // pad->Range(r.px1, r.py1, r.px2, r.py2);
1592 // pad->RangeAxis(r.ux1, r.uy1, r.ux2, r.uy2);
1593
1594 pad->fX1 = r.px1;
1595 pad->fX2 = r.px2;
1596 pad->fY1 = r.py1;
1597 pad->fY2 = r.py2;
1598
1599 pad->fUxmin = r.ux1;
1600 pad->fUxmax = r.ux2;
1601 pad->fUymin = r.uy1;
1602 pad->fUymax = r.uy2;
1603 }
1604
1605 // pad->SetPad(r.mleft, r.mbottom, 1-r.mright, 1-r.mtop);
1606
1607 pad->fAbsXlowNDC = r.xlow;
1608 pad->fAbsYlowNDC = r.ylow;
1609 pad->fAbsWNDC = r.xup - r.xlow;
1610 pad->fAbsHNDC = r.yup - r.ylow;
1611
1612 if (pad == Canvas()) {
1613 pad->fXlowNDC = r.xlow;
1614 pad->fYlowNDC = r.ylow;
1615 pad->fXUpNDC = r.xup;
1616 pad->fYUpNDC = r.yup;
1617 pad->fWNDC = r.xup - r.xlow;
1618 pad->fHNDC = r.yup - r.ylow;
1619 } else {
1620 auto mother = pad->GetMother();
1621 if (mother->GetAbsWNDC() > 0. && mother->GetAbsHNDC() > 0.) {
1622 pad->fXlowNDC = (r.xlow - mother->GetAbsXlowNDC()) / mother->GetAbsWNDC();
1623 pad->fYlowNDC = (r.ylow - mother->GetAbsYlowNDC()) / mother->GetAbsHNDC();
1624 pad->fXUpNDC = (r.xup - mother->GetAbsXlowNDC()) / mother->GetAbsWNDC();
1625 pad->fYUpNDC = (r.yup - mother->GetAbsYlowNDC()) / mother->GetAbsHNDC();
1626 pad->fWNDC = (r.xup - r.xlow) / mother->GetAbsWNDC();
1627 pad->fHNDC = (r.yup - r.ylow) / mother->GetAbsHNDC();
1628 }
1629 }
1630
1631 if (r.phi || r.theta) {
1632 pad->fPhi = r.phi;
1633 pad->fTheta = r.theta;
1634 }
1635
1636 // copy of code from TPad::ResizePad()
1637
1638 Double_t pxlow = r.xlow * r.cw;
1639 Double_t pylow = (1-r.ylow) * r.ch;
1640 Double_t pxrange = (r.xup - r.xlow) * r.cw;
1641 Double_t pyrange = -1*(r.yup - r.ylow) * r.ch;
1642
1643 Double_t rounding = 0.00005;
1644 Double_t xrange = r.px2 - r.px1;
1645 Double_t yrange = r.py2 - r.py1;
1646
1647 if ((xrange != 0.) && (pxrange != 0)) {
1648 // Linear X axis
1649 pad->fXtoAbsPixelk = rounding + pxlow - pxrange*r.px1/xrange; //origin at left
1650 pad->fXtoPixelk = rounding + -pxrange*r.px1/xrange;
1651 pad->fXtoPixel = pxrange/xrange;
1652 pad->fAbsPixeltoXk = r.px1 - pxlow*xrange/pxrange;
1653 pad->fPixeltoXk = r.px1;
1654 pad->fPixeltoX = xrange/pxrange;
1655 }
1656
1657 if ((yrange != 0.) && (pyrange != 0.)) {
1658 // Linear Y axis
1659 pad->fYtoAbsPixelk = rounding + pylow - pyrange*r.py1/yrange; //origin at top
1660 pad->fYtoPixelk = rounding + -pyrange - pyrange*r.py1/yrange;
1661 pad->fYtoPixel = pyrange/yrange;
1662 pad->fAbsPixeltoYk = r.py1 - pylow*yrange/pyrange;
1663 pad->fPixeltoYk = r.py1;
1664 pad->fPixeltoY = yrange/pyrange;
1665 }
1666
1667 pad->SetFixedAspectRatio(kFALSE);
1668
1669 TObjLink *objlnk = nullptr;
1670
1671 TH1 *hist = static_cast<TH1 *>(FindPrimitive(sid_pad_histogram, 1, pad, &objlnk));
1672
1673 if (hist) {
1674
1675 TObject *hist_holder = objlnk ? objlnk->GetObject() : nullptr;
1676 if (hist_holder == hist)
1677 hist_holder = nullptr;
1678
1679 Bool_t no_entries = hist->GetEntries();
1681
1682 Double_t hmin = 0., hmax = 0.;
1683
1684 auto setAxisRange = [](TAxis *ax, Double_t r1, Double_t r2) {
1685 if (r1 != r2)
1686 ax->SetRangeUser(r1, r2);
1687 else if ((ax->GetFirst() == ax->GetLast()) || ((ax->GetFirst() > 0) && (ax->GetLast() <= ax->GetNbins())))
1688 // only if no underflow/overflow bins selected - let reset
1689 ax->SetRange(0, 0);
1690 };
1691
1692 setAxisRange(hist->GetXaxis(), r.zx1, r.zx2);
1693
1694 if (hist->GetDimension() == 1) {
1695 hmin = r.zy1;
1696 hmax = r.zy2;
1697 if ((hmin == hmax) && !no_entries && !is_stack) {
1698 // if there are no zooming on Y and histogram has no entries, hmin/hmax should be set to full range
1699 hmin = pad->fLogy ? TMath::Power(pad->fLogy < 2 ? 10 : pad->fLogy, r.uy1) : r.uy1;
1700 hmax = pad->fLogy ? TMath::Power(pad->fLogy < 2 ? 10 : pad->fLogy, r.uy2) : r.uy2;
1701 }
1702 } else {
1703 setAxisRange(hist->GetYaxis(), r.zy1, r.zy2);
1704 }
1705
1706 if (hist->GetDimension() == 2) {
1707 hmin = r.zz1;
1708 hmax = r.zz2;
1709 if ((hmin == hmax) && !no_entries) {
1710 // z scale is not transformed
1711 hmin = r.uz1;
1712 hmax = r.uz2;
1713 }
1714 } else if (hist->GetDimension() == 3) {
1715 setAxisRange(hist->GetZaxis(), r.zz1, r.zz2);
1716 }
1717
1718 if (hmin == hmax)
1719 hmin = hmax = -1111;
1720
1721 if (is_stack) {
1722 hist->SetMinimum(hmin);
1723 hist->SetMaximum(hmax);
1724 hist->SetBit(TH1::kIsZoomed, hmin != hmax);
1725 } else if (!hist_holder || (hist_holder->IsA() == TScatter::Class())) {
1726 hist->SetMinimum(hmin);
1727 hist->SetMaximum(hmax);
1728 } else {
1729 auto SetMember = [hist_holder](const char *name, Double_t value) {
1730 auto offset = hist_holder->IsA()->GetDataMemberOffset(name);
1731 if (offset > 0)
1732 *((Double_t *)((char*) hist_holder + offset)) = value;
1733 else
1734 ::Error("SetMember", "Cannot find %s data member in %s", name, hist_holder->ClassName());
1735 };
1736
1737 // directly set min/max in classes like THStack, TGraph, TMultiGraph
1738 SetMember("fMinimum", hmin);
1739 SetMember("fMaximum", hmax);
1740 }
1741
1742 TIter next(hist->GetListOfFunctions());
1743 while (auto fobj = next())
1744 if (!hist_exec && fobj->InheritsFrom(TExec::Class())) {
1745 hist_exec = (TExec *) fobj;
1747 }
1748 }
1749
1750 std::map<std::string, int> idmap;
1751
1752 for (auto &item : r.primitives) {
1753 auto iter = idmap.find(item.snapid);
1754 int idcnt = 1;
1755 if (iter == idmap.end())
1756 idmap[item.snapid] = 1;
1757 else
1758 idcnt = ++iter->second;
1759
1761 }
1762
1763 // without special objects no need for explicit update of the pad
1764 if (fPadsStatus[pad]._has_specials) {
1765 pad->Modified(kTRUE);
1767 }
1768
1769 if (process_execs && (gPad == pad))
1771 }
1772
1774
1775 if (fUpdatedSignal) fUpdatedSignal(); // invoke signal
1776
1777 return need_update;
1778}
1779
1780//////////////////////////////////////////////////////////////////////////////////////////////////
1781/// Process TExec objects in the pad
1782
1784{
1785 auto execs = pad ? pad->GetListOfExecs() : nullptr;
1786
1787 if ((!execs || !execs->GetSize()) && !extra)
1788 return;
1789
1790 auto saveps = gVirtualPS;
1791 TWebPS ps;
1792 gVirtualPS = &ps;
1793
1794 auto savex = gVirtualX;
1795 TVirtualX x;
1796 gVirtualX = &x;
1797
1798 TIter next(execs);
1799 while (auto obj = next()) {
1800 auto exec = dynamic_cast<TExec *>(obj);
1801 if (exec)
1802 exec->Exec();
1803 }
1804
1805 if (extra)
1806 extra->Exec();
1807
1809 gVirtualX = savex;
1810}
1811
1812//////////////////////////////////////////////////////////////////////////////////////////
1813/// Execute one or several methods for selected object
1814/// String can be separated by ";;" to let execute several methods at once
1816{
1817 std::string buf = lines;
1818
1819 Int_t indx = 0;
1820
1821 while (obj && !buf.empty()) {
1822 std::string sub = buf;
1823 auto pos = buf.find(";;");
1824 if (pos == std::string::npos) {
1825 sub = buf;
1826 buf.clear();
1827 } else {
1828 sub = buf.substr(0,pos);
1829 buf = buf.substr(pos+2);
1830 }
1831 if (sub.empty()) continue;
1832
1833 std::stringstream exec;
1834 exec << "((" << obj->ClassName() << " *) " << std::hex << std::showbase << (size_t)obj << ")->" << sub << ";";
1836 Info("ProcessLinesForObject", "Obj %s Execute %s", obj->GetName(), exec.str().c_str());
1837 gROOT->ProcessLine(exec.str().c_str());
1838 indx++;
1839 }
1840}
1841
1842//////////////////////////////////////////////////////////////////////////////////////////
1843/// Handle data from web browser
1844/// Returns kFALSE if message was not processed
1845
1846Bool_t TWebCanvas::ProcessData(unsigned connid, const std::string &arg)
1847{
1848 if (arg.empty())
1849 return kTRUE;
1850
1851 // try to identify connection for given WS request
1852 unsigned indx = 0; // first connection is batch and excluded
1853 while(++indx < fWebConn.size()) {
1854 if (fWebConn[indx].fConnId == connid)
1855 break;
1856 }
1857 if (indx >= fWebConn.size())
1858 return kTRUE;
1859
1860 Bool_t is_main_connection = indx == 1; // first connection allow to make changes
1861
1862 struct FlagGuard {
1863 Bool_t &flag;
1864 FlagGuard(Bool_t &_flag) : flag(_flag) { flag = true; }
1865 ~FlagGuard() { flag = false; }
1866 };
1867
1869
1870 const char *cdata = arg.c_str();
1871
1872 if (arg == "KEEPALIVE") {
1873 // do nothing
1874
1875 } else if (arg == "QUIT") {
1876
1877 // use window manager to correctly terminate http server
1878 fWindow->TerminateROOT();
1879
1880 } else if (arg.compare(0, 7, "READY6:") == 0) {
1881
1882 // this is reply on drawing of ROOT6 snapshot
1883 // it confirms when drawing of specific canvas version is completed
1884
1885 cdata += 7;
1886
1887 const char *separ = strchr(cdata, ':');
1888 if (!separ) {
1889 fWebConn[indx].fDrawVersion = std::stoll(cdata);
1890 } else {
1891 fWebConn[indx].fDrawVersion = std::stoll(std::string(cdata, separ - cdata));
1893 if (DecodePadOptions(separ+1, false))
1895 }
1896
1897 if (indx == 1)
1898 fLastDrawVersion = fWebConn[indx].fDrawVersion;
1899
1900 } else if (arg == "RELOAD") {
1901
1902 // trigger reload of canvas data
1903 fWebConn[indx].reset();
1904
1905 } else if (arg.compare(0, 5, "SAVE:") == 0) {
1906
1907 // save image produced by the client side - like png or svg
1908 const char *img = cdata + 5;
1909
1910 const char *separ = strchr(img, ':');
1911 if (separ) {
1913 img = separ + 1;
1914
1915 std::ofstream ofs(filename.Data());
1916
1917 int filelen = -1;
1918
1919 if (filename.Index(".svg") != kNPOS) {
1920 // ofs << "<?xml version=\"1.0\" standalone=\"no\"?>";
1921 ofs << img;
1922 filelen = strlen(img);
1923 } else {
1925 ofs.write(binary.Data(), binary.Length());
1926 filelen = binary.Length();
1927 }
1928 ofs.close();
1929
1930 Info("ProcessData", "File %s size %d has been created", filename.Data(), filelen);
1931 }
1932
1933 } else if (arg.compare(0, 8, "PRODUCE:") == 0) {
1934
1935 // create ROOT, PDF, ... files using native ROOT functionality
1936 Canvas()->Print(arg.c_str() + 8);
1937
1938 } else if (arg.compare(0, 8, "GETMENU:") == 0) {
1939
1940 TObject *obj = FindPrimitive(arg.substr(8));
1941 if (!obj)
1942 obj = Canvas();
1943
1944 TWebMenuItems items(arg.c_str() + 8);
1945 items.PopulateObjectMenu(obj, obj->IsA());
1946 std::string buf = "MENU:";
1947 buf.append(TBufferJSON::ToJSON(&items, 103).Data());
1948
1949 AddSendQueue(connid, buf);
1950
1951 } else if (arg.compare(0, 11, "STATUSBITS:") == 0) {
1952
1953 if (is_main_connection) {
1954 AssignStatusBits(std::stoul(arg.substr(11)));
1955 if (fUpdatedSignal) fUpdatedSignal(); // invoke signal
1956 }
1957
1958 } else if (arg.compare(0, 10, "HIGHLIGHT:") == 0) {
1959
1960 if (is_main_connection) {
1961 auto arr = TBufferJSON::FromJSON<std::vector<std::string>>(arg.substr(10));
1962 if (!arr || (arr->size() != 4)) {
1963 Error("ProcessData", "Wrong arguments count %d in highlight message", (int)(arr ? arr->size() : -1));
1964 } else {
1965 auto pad = dynamic_cast<TVirtualPad *>(FindPrimitive(arr->at(0)));
1966 auto obj = FindPrimitive(arr->at(1));
1967 int argx = std::stoi(arr->at(2));
1968 int argy = std::stoi(arr->at(3));
1969 if (pad && obj) {
1970 Canvas()->Highlighted(pad, obj, argx, argy);
1972 }
1973 }
1974 }
1975
1976 } else if (ROOT::RWebWindow::IsFileDialogMessage(arg)) {
1977
1979
1980 } else if (IsReadOnly() || !is_main_connection) {
1981
1982 ///////////////////////////////////////////////////////////////////////////////////////
1983 // all following messages are not allowed in readonly mode or for secondary connections
1984
1985 return kFALSE;
1986
1987 } else if (arg.compare(0, 9, "OPTIONS6:") == 0) {
1988
1989 if (DecodePadOptions(arg.substr(9), true))
1991
1992 } else if (arg.compare(0, 9, "FITPANEL:") == 0) {
1993
1994 std::string chid = arg.substr(9);
1995
1996 TH1 *hist = nullptr;
1997 TIter iter(Canvas()->GetListOfPrimitives());
1998 while (auto obj = iter()) {
1999 hist = dynamic_cast<TH1 *>(obj);
2000 if (hist) break;
2001 }
2002
2004 if (chid == "standalone")
2005 showcmd = "panel->Show()";
2006 else
2007 showcmd = TString::Format("auto wptr = (std::shared_ptr<ROOT::RWebWindow>*)0x%zx;"
2008 "panel->Show({*wptr, %u, %s})",
2009 (size_t) &fWindow, connid, chid.c_str());
2010
2011 auto cmd = TString::Format("auto panel = std::make_shared<ROOT::Experimental::RFitPanel>(\"FitPanel\");"
2012 "panel->AssignCanvas(\"%s\");"
2013 "panel->AssignHistogram((TH1 *)0x%zx);"
2014 "%s;panel->ClearOnClose(panel);",
2015 Canvas()->GetName(), (size_t) hist, showcmd.Data());
2016 gROOT->ProcessLine(cmd.Data());
2017 } else if (arg == "START_BROWSER"s) {
2018
2019 gROOT->ProcessLine("new TBrowser;");
2020
2021 } else if (arg.compare(0, 6, "EVENT:") == 0) {
2022 auto arr = TBufferJSON::FromJSON<std::vector<std::string>>(arg.substr(6));
2023 if (!arr || (arr->size() != 5)) {
2024 Error("ProcessData", "Wrong arguments count %d in event message", (int)(arr ? arr->size() : -1));
2025 } else {
2026 auto pad = dynamic_cast<TPad *>(FindPrimitive(arr->at(0)));
2027 std::string kind = arr->at(1);
2028 int event = -1;
2029 if (kind == "move"s) event = kMouseMotion;
2030 int argx = std::stoi(arr->at(2));
2031 int argy = std::stoi(arr->at(3));
2032 auto selobj = FindPrimitive(arr->at(4));
2033
2034 if ((event >= 0) && pad && (pad == gPad)) {
2035 Canvas()->fEvent = event;
2036 Canvas()->fEventX = argx;
2037 Canvas()->fEventY = argy;
2038
2039 Canvas()->fSelected = selobj;
2040
2042 }
2043 }
2044
2045 } else if (arg.compare(0, 8, "PRIMIT6:") == 0) {
2046
2047 auto opt = TBufferJSON::FromJSON<TWebObjectOptions>(arg.c_str() + 8);
2048
2049 if (opt) {
2050 TPad *modpad = ProcessObjectOptions(*opt, nullptr);
2051
2052 // indicate that pad was modified
2053 if (modpad)
2054 modpad->Modified();
2055 }
2056
2057 } else if (arg.compare(0, 11, "PADCLICKED:") == 0) {
2058
2059 auto click = TBufferJSON::FromJSON<TWebPadClick>(arg.c_str() + 11);
2060
2061 if (click) {
2062
2063 TPad *pad = dynamic_cast<TPad *>(FindPrimitive(click->padid));
2064
2065 if (pad && pad->InheritsFrom(TButton::Class())) {
2066 auto btn = (TButton *) pad;
2067 const char *mthd = btn->GetMethod();
2068 if (mthd && *mthd) {
2069 auto cpad = gROOT->GetSelectedPad();
2070 if (cpad)
2071 cpad->cd();
2072 gROOT->ProcessLine(mthd);
2073 }
2074 return kTRUE;
2075 }
2076
2077 if (pad && (pad != gPad)) {
2078 gPad = pad;
2082 }
2083
2084 if (!click->objid.empty()) {
2085 auto selobj = FindPrimitive(click->objid);
2087 Canvas()->fSelected = selobj;
2088 if (pad && selobj && fObjSelectSignal)
2090 }
2091
2092 if ((click->x >= 0) && (click->y >= 0)) {
2094 Canvas()->fEventX = click->x;
2095 Canvas()->fEventY = click->y;
2096 if (click->dbl && fPadDblClickedSignal)
2098 else if (!click->dbl && fPadClickedSignal)
2100 }
2101
2103 }
2104
2105 } else if (arg.compare(0, 8, "OBJEXEC:") == 0) {
2106
2107 auto buf = arg.substr(8);
2108 auto pos = buf.find(":");
2109
2110 if ((pos > 0) && (pos != std::string::npos)) {
2111 auto sid = buf.substr(0, pos);
2112 buf.erase(0, pos + 1);
2113
2114 TObjLink *lnk = nullptr;
2115 TPad *objpad = nullptr;
2116
2117 TObject *obj = FindPrimitive(sid, 1, nullptr, &lnk, &objpad);
2118
2119 if (obj && !buf.empty()) {
2120
2121 ProcessLinesForObject(obj, buf);
2122
2123 if (objpad)
2124 objpad->Modified();
2125 else
2126 Canvas()->Modified();
2127
2129 }
2130 }
2131
2132 } else if (arg.compare(0, 12, "EXECANDSEND:") == 0) {
2133
2134 // execute method and send data, used by drawing projections
2135
2136 std::string buf = arg.substr(12);
2137 std::string reply;
2138 TObject *obj = nullptr;
2139
2140 auto pos = buf.find(":");
2141
2142 if (pos > 0) {
2143 // only first client can execute commands
2144 reply = buf.substr(0, pos);
2145 buf.erase(0, pos + 1);
2146 pos = buf.find(":");
2147 if (pos > 0) {
2148 auto sid = buf.substr(0, pos);
2149 buf.erase(0, pos + 1);
2150 obj = FindPrimitive(sid);
2151 }
2152 }
2153
2154 if (obj && !buf.empty() && !reply.empty()) {
2155 std::stringstream exec;
2156 exec << "((" << obj->ClassName() << " *) " << std::hex << std::showbase << (size_t)obj
2157 << ")->" << buf << ";";
2158 if (gDebug > 0)
2159 Info("ProcessData", "Obj %s Exec %s", obj->GetName(), exec.str().c_str());
2160
2161 auto res = gROOT->ProcessLine(exec.str().c_str());
2162 TObject *resobj = (TObject *)(res);
2163 if (resobj) {
2164 std::string send = reply;
2165 send.append(":");
2166 send.append(TBufferJSON::ToJSON(resobj, 23).Data());
2167 AddSendQueue(connid, send);
2168 if (reply[0] == 'D')
2169 delete resobj; // delete object if first symbol in reply is D
2170 }
2171 }
2172
2173 } else if (arg.compare(0, 6, "CLEAR:") == 0) {
2174 std::string snapid = arg.substr(6);
2175
2176 TPad *pad = dynamic_cast<TPad *>(FindPrimitive(snapid));
2177
2178 if (pad) {
2179 pad->Clear();
2180 pad->Modified();
2182 } else {
2183 Error("ProcessData", "Not found pad with id %s to clear\n", snapid.c_str());
2184 }
2185 } else if (arg.compare(0, 7, "DIVIDE:") == 0) {
2186 auto arr = TBufferJSON::FromJSON<std::vector<std::string>>(arg.substr(7));
2187 if (arr && arr->size() == 2) {
2188 TPad *pad = dynamic_cast<TPad *>(FindPrimitive(arr->at(0)));
2189 int nn = 0, n1 = 0, n2 = 0;
2190
2191 std::string divide = arr->at(1);
2192 auto p = divide.find('x');
2193 if (p == std::string::npos)
2194 p = divide.find('X');
2195
2196 if (p != std::string::npos) {
2197 n1 = std::stoi(divide.substr(0,p));
2198 n2 = std::stoi(divide.substr(p+1));
2199 } else {
2200 nn = std::stoi(divide);
2201 }
2202
2203 if (pad && ((nn > 1) || (n1*n2 > 1))) {
2204 pad->Clear();
2205 pad->Modified();
2206 if (nn > 1)
2207 pad->DivideSquare(nn);
2208 else
2209 pad->Divide(n1, n2);
2210 pad->cd(1);
2212 }
2213 }
2214
2215 } else if (arg.compare(0, 8, "DRAWOPT:") == 0) {
2216
2217 auto arr = TBufferJSON::FromJSON<std::vector<std::string>>(arg.substr(8));
2218 if (arr && arr->size() == 2) {
2219 TObjLink *objlnk = nullptr;
2220 FindPrimitive(arr->at(0), 1, nullptr, &objlnk);
2221 if (objlnk)
2222 objlnk->SetOption(arr->at(1).c_str());
2223 }
2224
2225 } else if (arg.compare(0, 8, "RESIZED:") == 0) {
2226
2227 auto arr = TBufferJSON::FromJSON<std::vector<int>>(arg.substr(8));
2228 if (arr && arr->size() == 7) {
2229 // set members directly to avoid redrawing of the client again
2230 Canvas()->fCw = arr->at(4);
2231 Canvas()->fCh = arr->at(5);
2232 fFixedSize = arr->at(6) > 0;
2233 arr->resize(4);
2235 }
2236
2237 } else if (arg.compare(0, 7, "POPOBJ:") == 0) {
2238
2239 auto arr = TBufferJSON::FromJSON<std::vector<std::string>>(arg.substr(7));
2240 if (arr && arr->size() == 2) {
2241 TPad *pad = dynamic_cast<TPad *>(FindPrimitive(arr->at(0)));
2242 TObject *obj = FindPrimitive(arr->at(1), 0, pad);
2243 if (pad && obj && (obj != pad->GetListOfPrimitives()->Last())) {
2244 TIter next(pad->GetListOfPrimitives());
2245 while (auto o = next())
2246 if (obj == o) {
2247 TString opt = next.GetOption();
2248 pad->Remove(obj, kFALSE);
2249 pad->Add(obj, opt.Data());
2250 break;
2251 }
2252 }
2253 }
2254
2255 } else if (arg.compare(0, 8, "SHOWURL:") == 0) {
2256
2258 args.SetUrl(arg.substr(8));
2259 args.SetStandalone(false);
2260
2262
2263 } else if (arg == "INTERRUPT"s) {
2264
2265 gROOT->SetInterrupt();
2266
2267 } else {
2268
2269 // unknown message, probably should be processed by other implementation
2270 return kFALSE;
2271
2272 }
2273
2274 return kTRUE;
2275}
2276
2277//////////////////////////////////////////////////////////////////////////////////////////
2278/// Returns true if any pad in the canvas were modified
2279/// Reset modified flags, increment canvas version (if inc_version is true)
2280
2282{
2283 if (fPadsStatus.find(pad) == fPadsStatus.end())
2284 fPadsStatus[pad] = PadStatus{0, true, true};
2285
2286 auto &entry = fPadsStatus[pad];
2287 entry._detected = true;
2288 if (pad->IsModified()) {
2289 pad->Modified(kFALSE);
2290 entry._modified = true;
2291 }
2292
2293 TIter iter(pad->GetListOfPrimitives());
2294 while (auto obj = iter()) {
2295 if (obj->IsA() == TPad::Class())
2296 CheckPadModified(static_cast<TPad *>(obj));
2297 }
2298}
2299
2300//////////////////////////////////////////////////////////////////////////////////////////
2301/// Check if any pad on the canvas was modified
2302/// If yes, increment version of correspondent pad
2303/// Returns true when canvas really modified
2304
2306{
2307 // clear temporary flags
2308 for (auto &entry : fPadsStatus) {
2309 entry.second._detected = false;
2310 entry.second._modified = force_modified;
2311 }
2312
2313 // scan sub-pads
2315
2316 // remove no-longer existing pads
2317 bool is_any_modified = false;
2318 for(auto iter = fPadsStatus.begin(); iter != fPadsStatus.end(); ) {
2319 if (iter->second._modified)
2320 is_any_modified = true;
2321 if (!iter->second._detected)
2322 fPadsStatus.erase(iter++);
2323 else
2324 iter++;
2325 }
2326
2327 // if any pad modified, increment canvas version and set version of modified pads
2328 if (is_any_modified) {
2329 fCanvVersion++;
2330 for(auto &entry : fPadsStatus)
2331 if (entry.second._modified)
2332 entry.second.fVersion = fCanvVersion;
2333 }
2334
2335 return is_any_modified;
2336}
2337
2338//////////////////////////////////////////////////////////////////////////////////////////
2339/// Set window geometry as array with coordinates and dimensions
2340
2341void TWebCanvas::SetWindowGeometry(const std::vector<int> &arr)
2342{
2344 Canvas()->fWindowTopX = arr[0];
2345 Canvas()->fWindowTopY = arr[1];
2346 Canvas()->fWindowWidth = arr[2];
2347 Canvas()->fWindowHeight = arr[3];
2348 if (fWindow) {
2349 // position is unreliable and cannot be used
2350 // fWindow->SetPosition(arr[0], arr[1]);
2351 fWindow->SetGeometry(arr[2], arr[3]);
2352 }
2353}
2354
2355//////////////////////////////////////////////////////////////////////////////////////////
2356/// Returns window geometry including borders and menus
2357
2359{
2360 if (fWindowGeometry.size() == 4) {
2361 x = fWindowGeometry[0];
2362 y = fWindowGeometry[1];
2363 w = fWindowGeometry[2];
2364 h = fWindowGeometry[3];
2365 } else {
2366 x = Canvas()->fWindowTopX;
2367 y = Canvas()->fWindowTopY;
2368 w = Canvas()->fWindowWidth;
2369 h = Canvas()->fWindowHeight;
2370 }
2371 return 0;
2372}
2373
2374
2375//////////////////////////////////////////////////////////////////////////////////////////
2376/// if canvas or any subpad was modified,
2377/// scan all primitives in the TCanvas and subpads and convert them into
2378/// the structure which will be delivered to JSROOT client
2379
2381{
2383
2385
2386 if (!fProcessingData && !IsAsyncMode() && !async)
2388 else if (fWindow)
2389 fWindow->Sync();
2390
2391 return kTRUE;
2392}
2393
2394//////////////////////////////////////////////////////////////////////////////////////////
2395/// Increment canvas version and force sending data to client - do not wait for reply
2396
2398{
2399 CheckCanvasModified(true);
2400
2401 if (!fWindow) {
2402 TCanvasWebSnapshot holder(IsReadOnly(), false, true); // readonly, set ids, batchmode
2403
2404 holder.SetScripts(ProcessCustomScripts(true));
2405
2406 CreatePadSnapshot(holder, Canvas(), 0, nullptr);
2407 } else {
2409 }
2410}
2411
2412//////////////////////////////////////////////////////////////////////////////////////////
2413/// Wait when specified version of canvas was painted and confirmed by browser
2414
2416{
2417 if (!fWindow)
2418 return kTRUE;
2419
2420 // simple polling loop until specified version delivered to the clients
2421 // first 500 loops done without sleep, then with 1ms sleep and last 500 with 100 ms sleep
2422
2423 long cnt = 0, cnt_limit = GetLongerPolling() ? 5500 : 1500;
2424
2425 if (gDebug > 2)
2426 Info("WaitWhenCanvasPainted", "version %ld", (long)ver);
2427
2428 while (cnt++ < cnt_limit) {
2429
2430 // handle send operations, check connection timeouts
2431 fWindow->Sync();
2432
2433 if (!fWindow->HasConnection(0, false)) {
2434 if (gDebug > 2)
2435 Info("WaitWhenCanvasPainted", "no connections - abort");
2436 return kFALSE; // wait ~1 min if no new connection established
2437 }
2438
2439 if ((fWebConn.size() > 1) && (fWebConn[1].fDrawVersion >= ver)) {
2440 if (gDebug > 2)
2441 Info("WaitWhenCanvasPainted", "ver %ld got painted", (long)ver);
2442 return kTRUE;
2443 }
2444
2445 if (!fWindow->HasConnection(0) && (fLastDrawVersion > 0)) {
2446 if (gDebug > 2)
2447 Info("WaitWhenCanvasPainted", "ver %ld got painted before client disconnected", (long)fLastDrawVersion);
2448 return kTRUE;
2449 }
2450
2452 if (cnt > 500)
2453 gSystem->Sleep((cnt < cnt_limit - 500) ? 1 : 100); // increase sleep interval when do very often
2454 }
2455
2456 if (gDebug > 2)
2457 Info("WaitWhenCanvasPainted", "timeout");
2458
2459 return kFALSE;
2460}
2461
2462//////////////////////////////////////////////////////////////////////////////////////////
2463/// Create JSON painting output for given pad
2464/// Produce JSON can be used for offline drawing with JSROOT
2465
2467{
2468 TString res;
2469 if (!pad)
2470 return res;
2471
2472 TCanvas *c = dynamic_cast<TCanvas *>(pad);
2473 if (c) {
2475 } else {
2476 auto imp = std::make_unique<TWebCanvas>(pad->GetCanvas(), pad->GetName(), 0, 0, pad->GetWw(), pad->GetWh(), kTRUE);
2477
2478 TPadWebSnapshot holder(true, false, batchmode); // readonly, no ids, batchmode
2479
2480 imp->CreatePadSnapshot(holder, pad, 0, [&res, json_compression](TPadWebSnapshot *snap) {
2482 });
2483 }
2484
2485 return res;
2486}
2487
2488//////////////////////////////////////////////////////////////////////////////////////////
2489/// Create JSON painting output for given canvas
2490/// Produce JSON can be used for offline drawing with JSROOT
2491
2493{
2494 TString res;
2495
2496 if (!c)
2497 return res;
2498
2499 {
2500 auto imp = std::make_unique<TWebCanvas>(c, c->GetName(), 0, 0, c->GetWw(), c->GetWh(), kTRUE);
2501
2502 TCanvasWebSnapshot holder(true, false, batchmode); // readonly, no ids, batchmode
2503
2505
2506 imp->CreatePadSnapshot(holder, c, 0, [&res, json_compression](TPadWebSnapshot *snap) {
2508 });
2509 }
2510
2511 return res;
2512}
2513
2514//////////////////////////////////////////////////////////////////////////////////////////
2515/// Create JSON painting output for given canvas and store into the file
2516/// See TBufferJSON::ExportToFile() method for more details about option
2517/// If option string starts with symbol 'b', JSON for batch mode will be generated (default)
2518/// If option string starts with symbol 'i', JSON for interactive mode will be generated
2519
2521{
2522 Int_t res = 0;
2524 if (option) {
2525 if (*option == 'b') {
2526 batchmode = kTRUE;
2527 ++option;
2528 } else if (*option == 'i') {
2529 batchmode = kFALSE;
2530 ++option;
2531 }
2532 }
2533
2534 if (!c)
2535 return res;
2536
2537 {
2538 auto imp = std::make_unique<TWebCanvas>(c, c->GetName(), 0, 0, c->GetWw(), c->GetWh(), kTRUE);
2539
2540 TCanvasWebSnapshot holder(true, false, batchmode); // readonly, no ids, batchmode
2541
2543
2544 imp->CreatePadSnapshot(holder, c, 0, [&res, filename, option](TPadWebSnapshot *snap) {
2546 });
2547 }
2548
2549 return res;
2550}
2551
2552//////////////////////////////////////////////////////////////////////////////////////////
2553/// Create image using batch (headless) capability of Chrome or Firefox browsers
2554/// Supported png, jpeg, svg, pdf formats
2555
2557{
2558 if (!pad)
2559 return false;
2560
2562 if (!json.Length())
2563 return false;
2564
2565 TString fname = fileName;
2566 const char *endings[4] = {"(", "[", "]", ")"};
2567 const char *suffix = nullptr;
2568 for (int n = 0; (n < 4) && !suffix; ++n) {
2569 if (fname.EndsWith(endings[n])) {
2570 fname.Resize(fname.Length() - 1);
2571 suffix = endings[n];
2572 }
2573 }
2574
2576
2578 if (fmt.empty())
2579 return false;
2580
2581 if (suffix) {
2582 if (fmt != "pdf")
2583 return false;
2584 switch (*suffix) {
2585 case '(': gBatchMultiPdf = fname.Data(); flush_batch = kFALSE; break;
2586 case '[': gBatchMultiPdf = fname.Data(); append_batch = kFALSE; flush_batch = kFALSE; break;
2587 case ']': gBatchMultiPdf.clear(); append_batch = kFALSE; break;
2588 case ')': gBatchMultiPdf.clear(); fname.Append("+"); break;
2589 }
2590 } else if (fmt == "pdf") {
2591 if (!gBatchMultiPdf.empty()) {
2592 if (gBatchMultiPdf.compare(fileName) == 0) {
2595 suffix = "+"; // to let append to the batch
2596 if ((gBatchFiles.size() > 0) && (gBatchFiles.back().compare(0, fname.Length(), fname.Data()) == 0))
2597 fname.Append("+"); // .pdf+ means appending image to previous
2598 } else {
2599 ::Error("TWebCanvas::ProduceImage", "Cannot change PDF name when multi-page PDF active");
2600 return false;
2601 }
2602 }
2603 } else if (!gBatchMultiPdf.empty()) {
2604 ::Error("TWebCanvas::ProduceImage", "Cannot produce other images when multi-page PDF active");
2605 return false;
2606 }
2607
2608 if (!width && !height) {
2609 if ((pad->GetCanvas() == pad) || (pad->IsA() == TCanvas::Class())) {
2610 width = pad->GetWw();
2611 height = pad->GetWh();
2612 } else {
2613 width = (Int_t) (pad->GetAbsWNDC() * pad->GetCanvas()->GetWw());
2614 height = (Int_t) (pad->GetAbsHNDC() * pad->GetCanvas()->GetWh());
2615 }
2616 }
2617
2618 if (!suffix && (!gBatchImageMode || (fmt == "s.pdf") || (fmt == "json") || (fmt == "s.png")))
2620
2621 if (append_batch) {
2622 gBatchFiles.emplace_back(fname.Data());
2623 gBatchJsons.emplace_back(json);
2624 gBatchWidths.emplace_back(width);
2625 gBatchHeights.emplace_back(height);
2626 }
2627
2628 if (!flush_batch || (gBatchJsons.size() < gBatchImageMode))
2629 return true;
2630
2631 return FlushBatchImages();
2632}
2633
2634//////////////////////////////////////////////////////////////////////////////////////////
2635/// Create images for several pads using batch (headless) capability of Chrome or Firefox browsers
2636/// Supported png, jpeg, svg, pdf, webp formats
2637/// One can include %d qualifier which will be replaced by image index using printf functionality.
2638/// If for pdf format %d qualifier not specified, all images will be stored in single PDF file.
2639/// For all other formats %d qualifier will be add before extension automatically
2640
2641bool TWebCanvas::ProduceImages(std::vector<TPad *> pads, const char *filename, Int_t width, Int_t height)
2642{
2643 if (pads.empty())
2644 return false;
2645
2646 std::vector<std::string> jsons;
2647 std::vector<Int_t> widths, heights;
2648
2649 for (unsigned n = 0; n < pads.size(); ++n) {
2650 auto pad = pads[n];
2651
2653 if (!json.Length())
2654 continue;
2655
2656 Int_t w = width, h = height;
2657
2658 if (!w && !h) {
2659 if ((pad->GetCanvas() == pad) || (pad->IsA() == TCanvas::Class())) {
2660 w = pad->GetWw();
2661 h = pad->GetWh();
2662 } else {
2663 w = (Int_t) (pad->GetAbsWNDC() * pad->GetCanvas()->GetWw());
2664 h = (Int_t) (pad->GetAbsHNDC() * pad->GetCanvas()->GetWh());
2665 }
2666 }
2667
2668 jsons.emplace_back(json.Data());
2669 widths.emplace_back(w);
2670 heights.emplace_back(h);
2671 }
2672
2674
2675 if (!gBatchImageMode || (fmt == "json") || (fmt == "s.png") || (fmt == "s.pdf"))
2677
2679
2681 gBatchJsons.insert(gBatchJsons.end(), jsons.begin(), jsons.end());
2684 if (gBatchJsons.size() < gBatchImageMode)
2685 return true;
2686
2687 return FlushBatchImages();
2688}
2689
2690
2691//////////////////////////////////////////////////////////////////////////////////////////
2692/// Process data for single primitive
2693/// Returns object pad if object was modified
2694
2696{
2697 TObjLink *lnk = nullptr;
2698 TPad *objpad = nullptr;
2699 TObject *obj = FindPrimitive(item.snapid, idcnt, pad, &lnk, &objpad);
2700
2701 if (item.fcust.compare("exec") == 0) {
2702 auto pos = item.opt.find("(");
2703 if (obj && (pos != std::string::npos) && obj->IsA()->GetMethodAllAny(item.opt.substr(0,pos).c_str())) {
2704 std::stringstream exec;
2705 exec << "((" << obj->ClassName() << " *) " << std::hex << std::showbase
2706 << (size_t)obj << ")->" << item.opt << ";";
2707 if (gDebug > 0)
2708 Info("ProcessObjectOptions", "Obj %s Execute %s", obj->GetName(), exec.str().c_str());
2709 gROOT->ProcessLine(exec.str().c_str());
2710 } else {
2711 Error("ProcessObjectOptions", "Fail to execute %s for object %p %s", item.opt.c_str(), obj, obj ? obj->ClassName() : "---");
2712 objpad = nullptr;
2713 }
2714 return objpad;
2715 }
2716
2717 bool modified = false;
2718
2719 if (obj && lnk) {
2720 auto pos = item.opt.find(";;use_"); // special coding of extra options
2721 if (pos != std::string::npos) item.opt.resize(pos);
2722
2723 if (gDebug > 0)
2724 Info("ProcessObjectOptions", "Set draw option %s for object %s %s", item.opt.c_str(),
2725 obj->ClassName(), obj->GetName());
2726
2727 lnk->SetOption(item.opt.c_str());
2728
2729 modified = true;
2730 }
2731
2732 if (item.fcust.compare(0,10,"auto_exec:") == 0) {
2733 ProcessLinesForObject(obj, item.fcust.substr(10));
2734 } else if (item.fcust.compare("frame") == 0) {
2735 if (obj && obj->InheritsFrom(TFrame::Class())) {
2736 TFrame *frame = static_cast<TFrame *>(obj);
2737 if (item.fopt.size() >= 4) {
2738 frame->SetX1(item.fopt[0]);
2739 frame->SetY1(item.fopt[1]);
2740 frame->SetX2(item.fopt[2]);
2741 frame->SetY2(item.fopt[3]);
2742 modified = true;
2743 }
2744 }
2745 } else if (item.fcust.compare(0,4,"pave") == 0) {
2746 if (obj && obj->InheritsFrom(TPave::Class())) {
2747 TPave *pave = static_cast<TPave *>(obj);
2748 if ((item.fopt.size() >= 4) && objpad) {
2750
2751 // first time need to overcome init problem
2752 pave->ConvertNDCtoPad();
2753
2754 pave->SetX1NDC(item.fopt[0]);
2755 pave->SetY1NDC(item.fopt[1]);
2756 pave->SetX2NDC(item.fopt[2]);
2757 pave->SetY2NDC(item.fopt[3]);
2758 modified = true;
2759
2760 pave->ConvertNDCtoPad();
2761 }
2762 if ((item.fcust.length() > 4) && pave->InheritsFrom(TPaveStats::Class())) {
2763 // add text lines for statsbox
2764 auto stats = static_cast<TPaveStats *>(pave);
2765 stats->Clear();
2766 size_t pos_start = 6, pos_end;
2767 while ((pos_end = item.fcust.find(";;", pos_start)) != std::string::npos) {
2768 stats->AddText(item.fcust.substr(pos_start, pos_end - pos_start).c_str());
2769 pos_start = pos_end + 2;
2770 }
2771 stats->AddText(item.fcust.substr(pos_start).c_str());
2772 }
2773 }
2774 } else if (item.fcust.compare(0,9,"func_fail") == 0) {
2775 if (fTF1UseSave <= 0) {
2776 fTF1UseSave = 1;
2777 modified = true;
2778 }
2779 }
2780
2781 return modified ? objpad : nullptr;
2782}
2783
2784//////////////////////////////////////////////////////////////////////////////////////////////////
2785/// Search of object with given id in list of primitives
2786/// One could specify pad where search could be start
2787/// Also if object is in list of primitives, one could ask for entry link for such object,
2788/// This can allow to change draw option
2789
2791{
2792 if (sid.empty() || (sid == "0"s))
2793 return nullptr;
2794
2795 if (!pad)
2796 pad = Canvas();
2797
2798 std::string subelement;
2799 long unsigned id = 0;
2800 bool search_hist = (sid == sid_pad_histogram);
2801 if (!search_hist) {
2802 auto separ = sid.find("#");
2803
2804 if (separ == std::string::npos) {
2805 id = std::stoul(sid);
2806 } else {
2807 subelement = sid.substr(separ + 1);
2808 id = std::stoul(sid.substr(0, separ));
2809 }
2810 if (TString::Hash(&pad, sizeof(pad)) == id)
2811 return pad;
2812 }
2813
2814 for (auto lnk = pad->GetListOfPrimitives()->FirstLink(); lnk != nullptr; lnk = lnk->Next()) {
2815 TObject *obj = lnk->GetObject();
2816 if (!obj) continue;
2817
2818 if (!search_hist && (TString::Hash(&obj, sizeof(obj)) != id)) {
2819 if (obj->IsA() == TPad::Class()) {
2820 obj = FindPrimitive(sid, idcnt, (TPad *)obj, objlnk, objpad);
2821 if (objpad && !*objpad)
2822 *objpad = pad;
2823 if (obj)
2824 return obj;
2825 }
2826 continue;
2827 }
2828
2829 // one may require to access n-th object
2830 if (!search_hist && --idcnt > 0)
2831 continue;
2832
2833 if (objpad)
2834 *objpad = pad;
2835
2836 if (objlnk)
2837 *objlnk = lnk;
2838
2839 if (search_hist)
2840 subelement = "hist";
2841
2842 auto getHistogram = [](TObject *container) -> TH1* {
2843 auto offset = container->IsA()->GetDataMemberOffset("fHistogram");
2844 if (offset > 0)
2845 return *((TH1 **)((char *)container + offset));
2846 ::Error("getHistogram", "Cannot access fHistogram data member in %s", container->ClassName());
2847 return nullptr;
2848 };
2849
2850 while(!subelement.empty() && obj) {
2851 // do not return link if sub-selement is searched - except for histogram
2852 if (!search_hist && objlnk)
2853 *objlnk = nullptr;
2854
2855 std::string kind = subelement;
2856 auto separ = kind.find("#");
2857 if (separ == std::string::npos) {
2858 subelement.clear();
2859 } else {
2860 kind.resize(separ);
2861 subelement = subelement.substr(separ + 1);
2862 }
2863
2864 TH1 *h1 = obj->InheritsFrom(TH1::Class()) ? static_cast<TH1 *>(obj) : nullptr;
2865 TGraph *gr = obj->InheritsFrom(TGraph::Class()) ? static_cast<TGraph *>(obj) : nullptr;
2866 TGraph2D *gr2d = obj->InheritsFrom(TGraph2D::Class()) ? static_cast<TGraph2D *>(obj) : nullptr;
2867 TScatter *scatter = obj->InheritsFrom(TScatter::Class()) ? static_cast<TScatter *>(obj) : nullptr;
2868 TMultiGraph *mg = obj->InheritsFrom(TMultiGraph::Class()) ? static_cast<TMultiGraph *>(obj) : nullptr;
2869 THStack *hs = obj->InheritsFrom(THStack::Class()) ? static_cast<THStack *>(obj) : nullptr;
2870 TF1 *f1 = obj->InheritsFrom(TF1::Class()) ? static_cast<TF1 *>(obj) : nullptr;
2871
2872 if (kind.compare("hist") == 0) {
2873 if (h1)
2874 obj = h1;
2875 else if (gr)
2876 obj = getHistogram(gr);
2877 else if (mg)
2878 obj = getHistogram(mg);
2879 else if (hs && (hs->GetNhists() > 0))
2880 obj = getHistogram(hs);
2881 else if (scatter)
2882 obj = getHistogram(scatter);
2883 else if (f1)
2884 obj = getHistogram(f1);
2885 else if (gr2d)
2886 obj = getHistogram(gr2d);
2887 else
2888 obj = nullptr;
2889 } else if (kind.compare("x") == 0) {
2890 obj = h1 ? h1->GetXaxis() : nullptr;
2891 } else if (kind.compare("y") == 0) {
2892 obj = h1 ? h1->GetYaxis() : nullptr;
2893 } else if (kind.compare("z") == 0) {
2894 obj = h1 ? h1->GetZaxis() : nullptr;
2895 } else if ((kind.compare(0,5,"func_") == 0) || (kind.compare(0,5,"indx_") == 0)) {
2896 auto funcname = kind.substr(5);
2897 TList *col = nullptr;
2898 if (h1)
2899 col = h1->GetListOfFunctions();
2900 else if (gr)
2901 col = gr->GetListOfFunctions();
2902 else if (mg)
2903 col = mg->GetListOfFunctions();
2904 else if (scatter->GetGraph())
2905 col = scatter->GetGraph()->GetListOfFunctions();
2906 if (!col)
2907 obj = nullptr;
2908 else if (kind.compare(0,5,"func_") == 0)
2909 obj = col->FindObject(funcname.c_str());
2910 else
2911 obj = col->At(std::stoi(funcname));
2912 } else if (kind.compare("polargram") == 0) {
2913 auto polar = dynamic_cast<TGraphPolar *>(obj);
2914 obj = polar ? polar->GetPolargram() : nullptr;
2915 } else if (kind.compare(0,7,"graphs_") == 0) {
2916 TList *graphs = mg ? mg->GetListOfGraphs() : nullptr;
2917 obj = graphs ? graphs->At(std::stoi(kind.substr(7))) : nullptr;
2918 } else if (kind.compare(0,6,"hists_") == 0) {
2919 TList *hists = hs ? hs->GetHists() : nullptr;
2920 obj = hists ? hists->At(std::stoi(kind.substr(6))) : nullptr;
2921 } else if (kind.compare(0,6,"stack_") == 0) {
2922 auto stack = hs ? hs->GetStack() : nullptr;
2923 obj = stack ? stack->At(std::stoi(kind.substr(6))) : nullptr;
2924 } else if (kind.compare(0,7,"member_") == 0) {
2925 auto member = kind.substr(7);
2926 auto offset = obj->IsA() ? obj->IsA()->GetDataMemberOffset(member.c_str()) : 0;
2927 obj = (offset > 0) ? *((TObject **)((char *) obj + offset)) : nullptr;
2928 } else {
2929 obj = nullptr;
2930 }
2931 }
2932
2933 if (!search_hist || obj)
2934 return obj;
2935 }
2936
2937 return nullptr;
2938}
2939
2940//////////////////////////////////////////////////////////////////////////////////////////////////
2941/// Static method to create TWebCanvas instance
2942/// Used by plugin manager
2943
2945{
2946 Bool_t readonly = gEnv->GetValue("WebGui.FullCanvas", (Int_t) 1) == 0;
2947
2948 auto imp = new TWebCanvas(c, name, x, y, width, height, readonly);
2949
2950 c->fWindowTopX = x;
2951 c->fWindowTopY = y;
2952 c->fWindowWidth = width;
2953 c->fWindowHeight = height;
2954 if (!gROOT->IsBatch() && (height > 25))
2955 height -= 25;
2956 c->fCw = width;
2957 c->fCh = height;
2958
2959 return imp;
2960}
2961
2962//////////////////////////////////////////////////////////////////////////////////////////////////
2963/// Create TCanvas and assign TWebCanvas implementation to it
2964/// Canvas is not displayed automatically, therefore canv->Show() method must be called
2965/// Or canvas can be embed in other widgets.
2966
2968{
2969 auto canvas = new TCanvas(kFALSE);
2970 canvas->SetName(name);
2971 canvas->SetTitle(title);
2972 canvas->ResetBit(TCanvas::kShowEditor);
2973 canvas->ResetBit(TCanvas::kShowToolBar);
2974 canvas->SetBit(TCanvas::kMenuBar, kTRUE);
2975 canvas->SetCanvas(canvas);
2976 canvas->SetBatch(kTRUE); // mark canvas as batch
2977 canvas->SetEditable(kTRUE); // ensure fPrimitives are created
2978
2979 // copy gStyle attributes
2980 canvas->SetFillColor(gStyle->GetCanvasColor());
2981 canvas->SetFillStyle(1001);
2982 canvas->SetGrid(gStyle->GetPadGridX(),gStyle->GetPadGridY());
2983 canvas->SetTicks(gStyle->GetPadTickX(),gStyle->GetPadTickY());
2984 canvas->SetLogx(gStyle->GetOptLogx());
2985 canvas->SetLogy(gStyle->GetOptLogy());
2986 canvas->SetLogz(gStyle->GetOptLogz());
2987 canvas->SetBottomMargin(gStyle->GetPadBottomMargin());
2988 canvas->SetTopMargin(gStyle->GetPadTopMargin());
2989 canvas->SetLeftMargin(gStyle->GetPadLeftMargin());
2990 canvas->SetRightMargin(gStyle->GetPadRightMargin());
2991 canvas->SetBorderSize(gStyle->GetCanvasBorderSize());
2992 canvas->SetBorderMode(gStyle->GetCanvasBorderMode());
2993
2994 auto imp = static_cast<TWebCanvas *> (NewCanvas(canvas, name, 0, 0, width, height));
2995
2996 canvas->SetCanvasImp(imp);
2997
2998 canvas->cd();
2999
3000 {
3002 auto l1 = gROOT->GetListOfCleanups();
3003 if (!l1->FindObject(canvas))
3004 l1->Add(canvas);
3005 auto l2 = gROOT->GetListOfCanvases();
3006 if (!l2->FindObject(canvas))
3007 l2->Add(canvas);
3008 }
3009
3010 // ensure creation of web window
3011 imp->CreateWebWindow();
3012
3013 return canvas;
3014}
3015
@ kMouseMotion
Definition Buttons.h:23
@ kButton1Double
Definition Buttons.h:24
@ kButton1Up
Definition Buttons.h:19
nlohmann::json json
#define c(i)
Definition RSha256.hxx:101
#define h(i)
Definition RSha256.hxx:106
int Int_t
Signed integer 4 bytes (int)
Definition RtypesCore.h:59
unsigned int UInt_t
Unsigned integer 4 bytes (unsigned int)
Definition RtypesCore.h:60
short Font_t
Font number (short)
Definition RtypesCore.h:95
constexpr Bool_t kFALSE
Definition RtypesCore.h:108
constexpr Ssiz_t kNPOS
The equivalent of std::string::npos for the ROOT class TString.
Definition RtypesCore.h:131
long long Long64_t
Portable signed long integer 8 bytes.
Definition RtypesCore.h:83
constexpr Bool_t kTRUE
Definition RtypesCore.h:107
ROOT::Detail::TRangeCast< T, true > TRangeDynCast
TRangeDynCast is an adapter class that allows the typed iteration through a TCollection.
R__EXTERN TEnv * gEnv
Definition TEnv.h:170
void Info(const char *location, const char *msgfmt,...)
Use this function for informational messages.
Definition TError.cxx:241
void Error(const char *location, const char *msgfmt,...)
Use this function in case an error occurred.
Definition TError.cxx:208
winID h TVirtualViewer3D TVirtualGLPainter p
Option_t Option_t option
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void data
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t hmin
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char filename
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t hmax
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char Pixmap_t Pixmap_t PictureAttributes_t attr const char char ret_data h unsigned char height h offset
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t r
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void value
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void funcs
Option_t Option_t width
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t src
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t height
Option_t Option_t TPoint TPoint const char text
char name[80]
Definition TGX11.cxx:110
@ kCanDelete
Definition TObject.h:372
@ kMustCleanup
Definition TObject.h:373
Int_t gDebug
Global variable setting the debug level. Set to 0 to disable, increase it in steps of 1 to increase t...
Definition TROOT.cxx:627
R__EXTERN TVirtualMutex * gROOTMutex
Definition TROOT.h:63
#define gROOT
Definition TROOT.h:414
R__EXTERN TStyle * gStyle
Definition TStyle.h:442
@ kReadPermission
Definition TSystem.h:55
R__EXTERN TSystem * gSystem
Definition TSystem.h:582
#define R__LOCKGUARD(mutex)
R__EXTERN TVirtualPS * gVirtualPS
Definition TVirtualPS.h:82
#define gPad
#define gVirtualX
Definition TVirtualX.h:337
static std::vector< WebFont_t > gWebFonts
static const std::string sid_pad_histogram
Color * colors
Definition X3DBuffer.c:21
const_iterator begin() const
const_iterator end() const
Holds different arguments for starting browser with RWebDisplayHandle::Display() method.
EBrowserKind GetBrowserKind() const
returns configured browser kind, see EBrowserKind for supported values
void SetStandalone(bool on=true)
Set standalone mode for running browser, default on When disabled, normal browser window (or just tab...
RWebDisplayArgs & SetWidgetKind(const std::string &kind)
set widget kind
RWebDisplayArgs & SetSize(int w, int h)
set preferable web window width and height
RWebDisplayArgs & SetUrl(const std::string &url)
set window url
RWebDisplayArgs & SetPos(int x=-1, int y=-1)
set preferable web window x and y position, negative is default
@ kCEF
Chromium Embedded Framework - local display with CEF libs.
@ kQt6
Qt6 QWebEngine libraries - Chromium code packed in qt6.
static bool ProduceImages(const std::string &fname, const std::vector< std::string > &jsons, const std::vector< int > &widths, const std::vector< int > &heights, const char *batch_file=nullptr)
Produce image file(s) using JSON data as source Invokes JSROOT drawing functionality in headless brow...
static std::vector< std::string > ProduceImagesNames(const std::string &fname, unsigned nfiles=1)
Produce vector of file names for specified file pattern Depending from supported file forma.
static std::string GetImageFormat(const std::string &fname)
Detect image format There is special handling of ".screenshot.pdf" and ".screenshot....
static bool ProduceImage(const std::string &fname, const std::string &json, int width=800, int height=600, const char *batch_file=nullptr)
Produce image file using JSON data as source Invokes JSROOT drawing functionality in headless browser...
static std::unique_ptr< RWebDisplayHandle > Display(const RWebDisplayArgs &args)
Create web display.
static std::shared_ptr< RWebWindow > Create()
Create new RWebWindow Using default RWebWindowsManager.
static unsigned ShowWindow(std::shared_ptr< RWebWindow > window, const RWebDisplayArgs &args="")
Static method to show web window Has to be used instead of RWebWindow::Show() when window potentially...
static bool EmbedFileDialog(const std::shared_ptr< RWebWindow > &window, unsigned connid, const std::string &args)
Create dialog instance to use as embedded dialog inside provided widget Loads libROOTBrowserv7 and tr...
static bool IsFileDialogMessage(const std::string &msg)
Check if this could be the message send by client to start new file dialog If returns true,...
static std::map< std::string, std::string > GetServerLocations()
Returns server locations as <std::string, std::string> Key is location name (with slash at the end) a...
Array of integers (32 bits per element).
Definition TArrayI.h:27
static TClass * Class()
virtual void SetFillColor(Color_t fcolor)
Set the fill area color.
Definition TAttFill.h:38
virtual void SetFillStyle(Style_t fstyle)
Set the fill area style.
Definition TAttFill.h:40
virtual void SetTextAlign(Short_t align=11)
Set the text alignment.
Definition TAttText.h:44
virtual void SetTextColor(Color_t tcolor=1)
Set the text color.
Definition TAttText.h:46
virtual void SetTextFont(Font_t tfont=62)
Set the text font.
Definition TAttText.h:48
virtual void SetTextSize(Float_t tsize=1)
Set the text size.
Definition TAttText.h:49
Class to manage histogram axis.
Definition TAxis.h:32
static TString Decode(const char *data)
Decode a base64 string date into a generic TString.
Definition TBase64.cxx:130
static TString Encode(const char *data)
Transform data into a null terminated base64 string.
Definition TBase64.cxx:106
virtual void SetY2(Double_t y2)
Definition TBox.h:65
virtual void SetX1(Double_t x1)
Definition TBox.h:62
virtual void SetX2(Double_t x2)
Definition TBox.h:63
virtual void SetY1(Double_t y1)
Definition TBox.h:64
static Int_t ExportToFile(const char *filename, const TObject *obj, const char *option=nullptr)
Convert object into JSON and store in text file Returns size of the produce file Used in TObject::Sav...
static TString ToJSON(const T *obj, Int_t compact=0, const char *member_name=nullptr)
Definition TBufferJSON.h:77
@ kNoSpaces
no new lines plus remove all spaces around "," and ":" symbols
Definition TBufferJSON.h:39
@ kMapAsObject
store std::map, std::unordered_map as JSON object
Definition TBufferJSON.h:41
@ kSameSuppression
zero suppression plus compress many similar values together
Definition TBufferJSON.h:45
A TButton object is a user interface object.
Definition TButton.h:18
static TClass * Class()
ABC describing GUI independent main window (with menubar, scrollbars and a drawing area).
Definition TCanvasImp.h:30
TCanvas * Canvas() const
Definition TCanvasImp.h:58
friend class TCanvas
Definition TCanvasImp.h:31
The Canvas class.
Definition TCanvas.h:23
UInt_t fCw
Width of the canvas along X (pixels)
Definition TCanvas.h:43
UInt_t GetWindowHeight() const
Definition TCanvas.h:162
void SetClickSelectedPad(TPad *pad)
Definition TCanvas.h:211
Int_t fWindowTopX
Top X position of window (in pixels)
Definition TCanvas.h:39
Int_t fEventX
! Last X mouse position in canvas
Definition TCanvas.h:46
TVirtualPadPainter * GetCanvasPainter()
Access and (probably) creation of pad painter.
Definition TCanvas.cxx:2613
UInt_t fWindowWidth
Width of window (including borders, etc.)
Definition TCanvas.h:41
Int_t fEventY
! Last Y mouse position in canvas
Definition TCanvas.h:47
UInt_t fWindowHeight
Height of window (including menubar, borders, etc.)
Definition TCanvas.h:42
TObject * fSelected
! Currently selected object
Definition TCanvas.h:49
UInt_t fCh
Height of the canvas along Y (pixels)
Definition TCanvas.h:44
UInt_t GetWindowWidth() const
Definition TCanvas.h:161
Int_t fWindowTopY
Top Y position of window (in pixels)
Definition TCanvas.h:40
void SetClickSelected(TObject *obj)
Definition TCanvas.h:209
@ kShowToolTips
Definition TCanvas.h:97
@ kShowToolBar
Definition TCanvas.h:92
@ kShowEventStatus
Definition TCanvas.h:89
@ kMenuBar
Definition TCanvas.h:91
@ kShowEditor
Definition TCanvas.h:93
virtual void Highlighted(TVirtualPad *pad, TObject *obj, Int_t x, Int_t y)
Emit Highlighted() signal.
Definition TCanvas.cxx:1610
static TClass * Class()
Int_t fEvent
! Type of current or last handled event
Definition TCanvas.h:45
TClass instances represent classes, structs and namespaces in the ROOT type system.
Definition TClass.h:84
Bool_t InheritsFrom(const char *cl) const override
Return kTRUE if this class inherits from a class with name "classname".
Definition TClass.cxx:4901
virtual Int_t GetSize() const
Return the capacity of the collection, i.e.
The color creation and management class.
Definition TColor.h:22
static const TArrayI & GetPalette()
Static function returning the current active palette.
Definition TColor.cxx:1521
static TClass * Class()
static Bool_t DefinedColors(Int_t set_always_on=0)
Static method returning kTRUE if some new colors have been defined after initialisation or since the ...
Definition TColor.cxx:1542
static TClass * Class()
virtual Int_t GetValue(const char *name, Int_t dflt) const
Returns the integer value for a resource.
Definition TEnv.cxx:503
TExec is a utility class that can be used to execute a C++ command when some event happens in a pad.
Definition TExec.h:26
virtual void Exec(const char *command="")
Execute the command referenced by this object.
Definition TExec.cxx:142
static TClass * Class()
1-Dim function class
Definition TF1.h:182
virtual Double_t GetXmax() const
Definition TF1.h:525
virtual TH1 * GetHistogram() const
Return a pointer to the histogram used to visualise the function Note that this histogram is managed ...
Definition TF1.cxx:1634
static TClass * Class()
@ kNotDraw
Definition TF1.h:297
virtual Bool_t IsValid() const
Return kTRUE if the function is valid.
Definition TF1.cxx:2931
virtual void Save(Double_t xmin, Double_t xmax, Double_t ymin, Double_t ymax, Double_t zmin, Double_t zmax)
Save values of function in array fSave.
Definition TF1.cxx:3211
TClass * IsA() const override
Definition TF1.h:694
virtual Double_t GetXmin() const
Definition TF1.h:521
Bool_t HasSave() const
Return true if function has data in fSave buffer.
Definition TF1.h:403
A 2-Dim function with parameters.
Definition TF2.h:29
void Save(Double_t xmin, Double_t xmax, Double_t ymin, Double_t ymax, Double_t zmin, Double_t zmax) override
Save values of function in array fSave.
Definition TF2.cxx:860
static TClass * Class()
TF3 defines a 3D Function with Parameters.
Definition TF3.h:28
Define a Frame.
Definition TFrame.h:19
static TClass * Class()
The axis painter class.
Definition TGaxis.h:26
static TClass * Class()
Graphics object made of three arrays X, Y and Z with the same number of points each.
Definition TGraph2D.h:41
static TClass * Class()
To draw a polar graph.
Definition TGraphPolar.h:23
static TClass * Class()
To draw polar axis.
static TClass * Class()
A TGraph is an object made of two arrays X and Y with npoints each.
Definition TGraph.h:41
static TClass * Class()
@ kNoStats
Don't draw stats box.
Definition TGraph.h:74
TList * GetListOfFunctions() const
Definition TGraph.h:125
virtual TH1F * GetHistogram() const
Returns a pointer to the histogram used to draw the axis Takes into account the two following cases.
Definition TGraph.cxx:1458
TH1 is the base class of all histogram classes in ROOT.
Definition TH1.h:109
TAxis * GetZaxis()
Definition TH1.h:573
static TClass * Class()
virtual Int_t GetDimension() const
Definition TH1.h:527
@ kNoTitle
Don't draw the histogram title.
Definition TH1.h:408
@ kIsZoomed
Bit set when zooming on Y axis.
Definition TH1.h:407
TAxis * GetXaxis()
Definition TH1.h:571
virtual void SetMaximum(Double_t maximum=-1111)
Definition TH1.h:652
TAxis * GetYaxis()
Definition TH1.h:572
virtual void SetMinimum(Double_t minimum=-1111)
Definition TH1.h:653
virtual Double_t GetEntries() const
Return the current number of entries.
Definition TH1.cxx:4438
TList * GetListOfFunctions() const
Definition TH1.h:488
virtual Int_t BufferEmpty(Int_t action=0)
Fill histogram with all entries in the buffer.
Definition TH1.cxx:1389
The Histogram stack class.
Definition THStack.h:40
static TClass * Class()
static char * ReadFileContent(const char *filename, Int_t &len)
Reads content of file from the disk.
Option_t * GetOption() const
void Reset()
A doubly linked list.
Definition TList.h:38
TObject * FindObject(const char *name) const override
Find an object in this list using its name.
Definition TList.cxx:708
void Add(TObject *obj) override
Definition TList.h:81
TObject * At(Int_t idx) const override
Returns the object at position idx. Returns 0 if idx is out of range.
Definition TList.cxx:487
void AddFirst(TObject *obj) override
Add object at the beginning of the list.
Definition TList.cxx:97
A TMultiGraph is a collection of TGraph (or derived) objects.
Definition TMultiGraph.h:34
TList * GetListOfGraphs() const
Definition TMultiGraph.h:67
static TClass * Class()
TList * GetListOfFunctions()
Return pointer to list of functions.
const char * GetName() const override
Returns name of object.
Definition TNamed.h:49
An array of TObjects.
Definition TObjArray.h:31
Mother of all ROOT objects.
Definition TObject.h:42
virtual const char * GetName() const
Returns name of object.
Definition TObject.cxx:458
R__ALWAYS_INLINE Bool_t TestBit(UInt_t f) const
Definition TObject.h:204
virtual const char * ClassName() const
Returns name of class to which the object belongs.
Definition TObject.cxx:226
void SetBit(UInt_t f, Bool_t set)
Set or unset the user status bits as specified in f.
Definition TObject.cxx:882
virtual Bool_t InheritsFrom(const char *classname) const
Returns kTRUE if object inherits from class "classname".
Definition TObject.cxx:544
virtual const char * GetTitle() const
Returns title of object.
Definition TObject.cxx:502
virtual TClass * IsA() const
Definition TObject.h:248
virtual void Paint(Option_t *option="")
This method must be overridden if a class wants to paint itself.
Definition TObject.cxx:626
The most important graphics class in the ROOT system.
Definition TPad.h:28
static TClass * Class()
void Modified(Bool_t flag=true) override
Mark pad modified Will be repainted when TCanvas::Update() will be called next time.
Definition TPad.cxx:7511
void Print(const char *filename="") const override
This method is equivalent to SaveAs("filename"). See TPad::SaveAs for details.
Definition TPad.cxx:4968
The histogram statistics painter class.
Definition TPaveStats.h:18
virtual void SetStatFormat(const char *format="6.4g")
Change (i.e. set) the format for printing statistics.
void SetOptStat(Int_t stat=1)
Set the stat option.
virtual void SetFitFormat(const char *format="5.4g")
Change (i.e. set) the format for printing fit parameters in statistics box.
void SetParent(TObject *obj) override
Definition TPaveStats.h:53
void SetOptFit(Int_t fit=1)
Set the fit option.
static TClass * Class()
A Pave (see TPave) with text, lines or/and boxes inside.
Definition TPaveText.h:21
virtual TText * AddText(Double_t x1, Double_t y1, const char *label)
Add a new Text line to this pavetext at given coordinates.
static TClass * Class()
void Clear(Option_t *option="") override
Clear all lines in this pavetext.
virtual TText * GetLine(Int_t number) const
Get Pointer to line number in this pavetext.
A TBox with a bordersize and a shadow option.
Definition TPave.h:19
virtual void SetName(const char *name="")
Definition TPave.h:81
virtual void SetBorderSize(Int_t bordersize=4)
Sets the border size of the TPave box and shadow.
Definition TPave.h:79
static TClass * Class()
Option_t * GetOption() const override
Definition TPave.h:59
A TScatter is able to draw four variables scatter plot on a single plot.
Definition TScatter.h:32
static TClass * Class()
Basic string class.
Definition TString.h:138
Ssiz_t Length() const
Definition TString.h:425
const char * Data() const
Definition TString.h:384
@ kBoth
Definition TString.h:284
@ kIgnoreCase
Definition TString.h:285
Ssiz_t Last(char c) const
Find last occurrence of a character c.
Definition TString.cxx:938
void ToUpper()
Change string to upper case.
Definition TString.cxx:1202
UInt_t Hash(ECaseCompare cmp=kExact) const
Return hash value.
Definition TString.cxx:684
TString & Append(const char *cs)
Definition TString.h:581
static TString Format(const char *fmt,...)
Static method which formats a string using a printf style format descriptor and return a TString.
Definition TString.cxx:2384
Bool_t Contains(const char *pat, ECaseCompare cmp=kExact) const
Definition TString.h:641
Int_t GetOptLogy() const
Definition TStyle.h:250
Int_t GetOptStat() const
Definition TStyle.h:247
Color_t GetStatTextColor() const
Definition TStyle.h:260
Int_t GetOptTitle() const
Definition TStyle.h:248
Int_t GetPadTickX() const
Definition TStyle.h:219
Float_t GetStatFontSize() const
Definition TStyle.h:263
Float_t GetStatX() const
Definition TStyle.h:266
Float_t GetPadRightMargin() const
Definition TStyle.h:216
Style_t GetTitleFont(Option_t *axis="X") const
Return title font.
Definition TStyle.cxx:1217
Float_t GetStatY() const
Definition TStyle.h:267
Color_t GetTitleFillColor() const
Definition TStyle.h:273
Style_t GetTitleStyle() const
Definition TStyle.h:275
Bool_t GetPadGridY() const
Definition TStyle.h:218
Color_t GetStatColor() const
Definition TStyle.h:259
Float_t GetPadLeftMargin() const
Definition TStyle.h:215
Bool_t GetPadGridX() const
Definition TStyle.h:217
Float_t GetStatH() const
Definition TStyle.h:269
static TClass * Class()
Int_t GetPadTickY() const
Definition TStyle.h:220
Width_t GetTitleBorderSize() const
Definition TStyle.h:277
Color_t GetCanvasColor() const
Definition TStyle.h:190
Float_t GetPadBottomMargin() const
Definition TStyle.h:213
Width_t GetStatBorderSize() const
Definition TStyle.h:261
Color_t GetTitleTextColor() const
Definition TStyle.h:274
Int_t GetOptLogx() const
Definition TStyle.h:249
Style_t GetStatStyle() const
Definition TStyle.h:264
Float_t GetStatW() const
Definition TStyle.h:268
const char * GetFitFormat() const
Definition TStyle.h:201
Int_t GetCanvasBorderMode() const
Definition TStyle.h:192
const char * GetStatFormat() const
Definition TStyle.h:265
Width_t GetCanvasBorderSize() const
Definition TStyle.h:191
Int_t GetOptFit() const
Definition TStyle.h:246
Style_t GetStatFont() const
Definition TStyle.h:262
Int_t GetOptLogz() const
Definition TStyle.h:251
Float_t GetTitleFontSize() const
Definition TStyle.h:276
Float_t GetPadTopMargin() const
Definition TStyle.h:214
virtual Bool_t ExpandPathName(TString &path)
Expand a pathname getting rid of special shell characters like ~.
Definition TSystem.cxx:1285
virtual Bool_t AccessPathName(const char *path, EAccessMode mode=kFileExists)
Returns FALSE if one can access a file using the specified access mode.
Definition TSystem.cxx:1307
virtual void Sleep(UInt_t milliSec)
Sleep milliSec milli seconds.
Definition TSystem.cxx:435
virtual Bool_t ProcessEvents()
Process pending events (GUI, timers, sockets).
Definition TSystem.cxx:414
Base class for several text objects.
Definition TText.h:22
static Long_t SelfId()
Static method returning the id for the current thread.
Definition TThread.cxx:552
Handles synchronous and a-synchronous timer events.
Definition TTimer.h:51
virtual void TurnOn()
Add the timer to the system timer list.
Definition TTimer.cxx:246
void SetTime(Long_t milliSec)
Definition TTimer.h:91
See TView3D.
Definition TView.h:25
static TView * CreateView(Int_t system=1, const Double_t *rmin=nullptr, const Double_t *rmax=nullptr)
Create a concrete default 3-d view via the plug-in manager.
Definition TView.cxx:26
virtual void SetAutoRange(Bool_t autorange=kTRUE)=0
TVirtualPS is an abstract interface to Postscript, PDF, SVG.
Definition TVirtualPS.h:30
To make it possible to use GL for 2D graphic in a TPad/TCanvas.
small helper class to store/restore gPad context in TPad methods
Definition TVirtualPad.h:61
TVirtualPad is an abstract base class for the Pad and Canvas classes.
Definition TVirtualPad.h:51
Semi-Abstract base class defining a generic interface to the underlying, low level,...
Definition TVirtualX.h:46
void SetSlow(Bool_t slow=kTRUE)
TWebCanvasTimer(TWebCanvas &canv)
Bool_t IsSlow() const
void Timeout() override
used to send control messages to clients
TWebCanvas & fCanv
Basic TCanvasImp ABI implementation for Web-based Graphics Provides painting of main ROOT classes in ...
Definition TWebCanvas.h:35
TVirtualPadPainter * CreatePadPainter() override
Creates web-based pad painter.
void ForceUpdate() override
Increment canvas version and force sending data to client - do not wait for reply.
static TCanvas * CreateWebCanvas(const char *name, const char *title, UInt_t width=1200, UInt_t height=800)
Create TCanvas and assign TWebCanvas implementation to it Canvas is not displayed automatically,...
static void AddCustomClass(const std::string &clname, bool with_derived=false)
Assign custom class.
static TString CreatePadJSON(TPad *pad, Int_t json_compression=0, Bool_t batchmode=kFALSE)
Create JSON painting output for given pad Produce JSON can be used for offline drawing with JSROOT.
void SetCanvasSize(UInt_t w, UInt_t h) override
Set canvas size of web canvas.
UInt_t fColorsHash
! last hash of colors/palette
Definition TWebCanvas.h:107
Int_t fTF1UseSave
! use save buffer for TF1/TF2, 0:off, 1:prefer, 2:force
Definition TWebCanvas.h:108
void ShowCmd(const std::string &arg, Bool_t show)
Function used to send command to browser to toggle menu, toolbar, editors, ...
Long64_t fColorsVersion
! current colors/palette version, checked every time when new snapshot created
Definition TWebCanvas.h:106
virtual Bool_t IsReadOnly() const
Definition TWebCanvas.h:197
std::shared_ptr< ROOT::RWebWindow > fWindow
Definition TWebCanvas.h:88
virtual Bool_t IsJSSupportedClass(TObject *obj, Bool_t many_primitives=kFALSE)
Returns kTRUE when object is fully supported on JSROOT side In ROOT7 Paint function will just return ...
void AddCtrlMsg(unsigned connid, const std::string &key, const std::string &value)
Add control message for specified connection Same control message can be overwritten many time before...
static void SetCustomScripts(const std::string &src)
Configures custom script for canvas.
ObjectSelectSignal_t fObjSelectSignal
! signal emitted when new object selected in the pad
Definition TWebCanvas.h:116
PadClickedSignal_t fPadClickedSignal
! signal emitted when simple mouse click performed on the pad
Definition TWebCanvas.h:114
void SetLongerPolling(Bool_t on)
Definition TWebCanvas.h:253
UInt_t fStyleHash
! last hash of gStyle
Definition TWebCanvas.h:105
virtual Bool_t CanCreateObject(const std::string &)
Definition TWebCanvas.h:171
void ShowWebWindow(const ROOT::RWebDisplayArgs &user_args="")
Show canvas in specified place.
Int_t fPrimitivesMerge
! number of PS primitives, which will be merged together
Definition TWebCanvas.h:98
void Show() override
Show canvas in browser window.
Bool_t WaitWhenCanvasPainted(Long64_t ver)
Wait when specified version of canvas was painted and confirmed by browser.
static UInt_t gBatchImageMode
! configured batch size
Definition TWebCanvas.h:123
static std::string gCustomScripts
! custom JavaScript code or URL on JavaScript files to load before start drawing
Definition TWebCanvas.h:143
Bool_t IsAsyncMode() const
Definition TWebCanvas.h:257
Long64_t fLastDrawVersion
! last draw version
Definition TWebCanvas.h:92
UInt_t CalculateColorsHash()
Calculate hash function for all colors and palette.
void SetWindowGeometry(const std::vector< int > &arr)
Set window geometry as array with coordinates and dimensions.
Bool_t HasStatusBar() const override
Returns kTRUE if web canvas has status bar.
static std::vector< std::string > gCustomClasses
! list of custom classes, which can be delivered as is to client
Definition TWebCanvas.h:144
void CreateWebWindow()
Create web window for the canvas.
void Close() override
Close web canvas - not implemented.
static bool ProduceImages(std::vector< TPad * > pads, const char *filename, Int_t width=0, Int_t height=0)
Create images for several pads using batch (headless) capability of Chrome or Firefox browsers Suppor...
Bool_t HasMenuBar() const override
Returns kTRUE if web canvas has menu bar.
Int_t InitWindow() override
Initialize window for the web canvas At this place canvas is not yet register to the list of canvases...
void CheckPadModified(TPad *pad)
Returns true if any pad in the canvas were modified Reset modified flags, increment canvas version (i...
void RaiseWindow() override
Raise browser window.
static bool ProduceImage(TPad *pad, const char *filename, Int_t width=0, Int_t height=0)
Create image using batch (headless) capability of Chrome or Firefox browsers Supported png,...
void ActivateInEditor(TPad *pad, TObject *obj)
Activate object in editor in web browser.
std::vector< WebConn > fWebConn
! connections
Definition TWebCanvas.h:83
PadSignal_t fActivePadChangedSignal
! signal emitted when active pad changed in the canvas
Definition TWebCanvas.h:113
Bool_t GetLongerPolling() const
Definition TWebCanvas.h:254
UInt_t fClientBits
! latest status bits from client like editor visible or not
Definition TWebCanvas.h:93
std::function< void(TPadWebSnapshot *)> PadPaintingReady_t
Function called when pad painting produced.
Definition TWebCanvas.h:55
Int_t fPaletteDelivery
! colors palette delivery 0:never, 1:once, 2:always, 3:per subpad
Definition TWebCanvas.h:97
Bool_t fProcessingData
! flag used to prevent blocking methods when process data is invoked
Definition TWebCanvas.h:102
Bool_t HasToolTips() const override
Returns kTRUE if tooltips are activated in web canvas.
std::vector< TPad * > fAllPads
! list of all pads recognized during streaming
Definition TWebCanvas.h:94
friend class TWebCanvasTimer
Definition TWebCanvas.h:37
TWebCanvasTimer * fTimer
! timer to submit control messages
Definition TWebCanvas.h:84
static std::vector< std::string > gBatchJsons
! converted jsons batch job
Definition TWebCanvas.h:126
Long64_t fCanvVersion
! actual canvas version, changed with every new Modified() call
Definition TWebCanvas.h:91
std::vector< int > fWindowGeometry
! last received window geometry
Definition TWebCanvas.h:109
TPad * ProcessObjectOptions(TWebObjectOptions &item, TPad *pad, int idcnt=1)
Process data for single primitive Returns object pad if object was modified.
void CreateObjectSnapshot(TPadWebSnapshot &master, TPad *pad, TObject *obj, const char *opt, TWebPS *masterps=nullptr)
Creates representation of the object for painting in web browser.
std::map< TObject *, bool > fUsedObjs
! map of used objects during streaming
Definition TWebCanvas.h:95
void AddColorsPalette(TPadWebSnapshot &master)
Add special canvas objects with list of colors and color palette.
Long64_t fStyleVersion
! current gStyle object version, checked every time when new snapshot created
Definition TWebCanvas.h:104
static std::string gBatchMultiPdf
! name of current multi-page pdf file
Definition TWebCanvas.h:124
static void BatchImageMode(UInt_t n=100)
Configure batch image mode for web graphics.
std::vector< std::unique_ptr< ROOT::RWebDisplayHandle > > fHelpHandles
! array of handles for help widgets
Definition TWebCanvas.h:118
void AddSendQueue(unsigned connid, const std::string &msg)
Add message to send queue for specified connection If connid == 0, message will be add to all connect...
void SetWindowPosition(Int_t x, Int_t y) override
Set window position of web canvas.
UpdatedSignal_t fUpdatedSignal
! signal emitted when canvas updated or state is changed
Definition TWebCanvas.h:112
Int_t fJsonComp
! compression factor for messages send to the client
Definition TWebCanvas.h:99
static std::vector< int > gBatchWidths
! batch job widths
Definition TWebCanvas.h:127
~TWebCanvas() override
Destructor.
static std::string ProcessCustomScripts(bool batch)
For batch mode special handling of scripts are required Headless browser not able to load modules fro...
Bool_t fReadOnly
!< configured display
Definition TWebCanvas.h:90
std::map< TPad *, PadStatus > fPadsStatus
! map of pads in canvas and their status flags
Definition TWebCanvas.h:86
static Font_t AddFont(const char *name, const char *ttffile, Int_t precision=2)
Add font to static list of fonts supported by the canvas Name specifies name of the font,...
void AddCustomFonts(TPadWebSnapshot &master)
Add special canvas objects with custom fonts.
Bool_t CheckDataToSend(unsigned connid=0)
Check if any data should be send to client If connid != 0, only selected connection will be checked.
Bool_t PerformUpdate(Bool_t async) override
if canvas or any subpad was modified, scan all primitives in the TCanvas and subpads and convert them...
void AssignStatusBits(UInt_t bits)
Assign clients bits.
virtual Bool_t ProcessData(unsigned connid, const std::string &arg)
Handle data from web browser Returns kFALSE if message was not processed.
void ProcessLinesForObject(TObject *obj, const std::string &lines)
Execute one or several methods for selected object String can be separated by ";;" to let execute sev...
TWebCanvas(TCanvas *c, const char *name, Int_t x, Int_t y, UInt_t width, UInt_t height, Bool_t readonly=kTRUE)
Constructor.
static std::vector< int > gBatchHeights
! batch job heights
Definition TWebCanvas.h:128
static Int_t StoreCanvasJSON(TCanvas *c, const char *filename, const char *option="")
Create JSON painting output for given canvas and store into the file See TBufferJSON::ExportToFile() ...
static const std::string & GetCustomScripts()
Returns configured custom script.
void Iconify() override
Iconify browser window.
void SetWindowTitle(const char *newTitle) override
Set window title of web canvas.
UInt_t GetWindowGeometry(Int_t &x, Int_t &y, UInt_t &w, UInt_t &h) override
Returns window geometry including borders and menus.
static std::vector< std::string > gBatchFiles
! file names for batch job
Definition TWebCanvas.h:125
static TCanvasImp * NewCanvas(TCanvas *c, const char *name, Int_t x, Int_t y, UInt_t width, UInt_t height)
Static method to create TWebCanvas instance Used by plugin manager.
static TString CreateCanvasJSON(TCanvas *c, Int_t json_compression=0, Bool_t batchmode=kFALSE)
Create JSON painting output for given canvas Produce JSON can be used for offline drawing with JSROOT...
Bool_t HasEditor() const override
Returns kTRUE if web canvas has graphical editor.
Int_t fStyleDelivery
! gStyle delivery to clients: 0:never, 1:once, 2:always
Definition TWebCanvas.h:96
PadClickedSignal_t fPadDblClickedSignal
! signal emitted when simple mouse click performed on the pad
Definition TWebCanvas.h:115
void ProcessExecs(TPad *pad, TExec *extra=nullptr)
Process TExec objects in the pad.
static bool FlushBatchImages()
Flush batch images.
void CreatePadSnapshot(TPadWebSnapshot &paddata, TPad *pad, Long64_t version, PadPaintingReady_t func)
Create snapshot for pad and all primitives Callback function is used to create JSON in the middle of ...
Bool_t CheckCanvasModified(bool force_modified=false)
Check if any pad on the canvas was modified If yes, increment version of correspondent pad Returns tr...
virtual Bool_t DecodePadOptions(const std::string &, bool process_execs=false)
Decode all pad options, which includes ranges plus objects options.
Int_t GetPaletteDelivery() const
Definition TWebCanvas.h:248
void SetWindowSize(UInt_t w, UInt_t h) override
Set window size of web canvas.
static bool IsCustomClass(const TClass *cl)
Checks if class belongs to custom.
TObject * FindPrimitive(const std::string &id, int idcnt=1, TPad *pad=nullptr, TObjLink **objlnk=nullptr, TPad **objpad=nullptr)
Search of object with given id in list of primitives One could specify pad where search could be star...
Bool_t fFixedSize
! is canvas size fixed
Definition TWebCanvas.h:110
Int_t GetStyleDelivery() const
Definition TWebCanvas.h:245
Class used to transport drawing options from the client.
Bool_t IsEmptyPainting() const
Definition TWebPS.h:32
TWebPainting * TakePainting()
Definition TWebPS.h:34
TWebPainting * GetPainting()
Definition TWebPS.h:33
Implement TVirtualPadPainter which abstracts painting operations.
Object used to store paint operations and deliver them to JSROOT.
void SetObjectName(const std::string &objname)
void SetClassName(const std::string &classname)
@ kStyle
gStyle object
@ kObject
object itself
@ kSVG
list of SVG primitives
@ kSubPad
subpad
@ kFont
custom web font
@ kColors
list of ROOT colors + palette
Double_t y[n]
Definition legend1.C:17
Double_t x[n]
Definition legend1.C:17
const Int_t n
Definition legend1.C:16
TGraphErrors * gr
Definition legend1.C:25
TH1F * h1
Definition legend1.C:5
TF1 * f1
Definition legend1.C:11
LongDouble_t Power(LongDouble_t x, LongDouble_t y)
Returns x raised to the power y.
Definition TMath.h:732
TString fName
TString fFormat
WebFont_t()=default
TString fData
WebFont_t(Int_t indx, const TString &name, const TString &fmt, const TString &data)