#include "TGImageMap.h"
#include "TRefCnt.h"
#include "TGMenu.h"
#include "TGToolTip.h"
#include "TList.h"
#include "TArrayS.h"
ClassImp(TGRegion)
ClassImp(TGRegionWithId)
ClassImpQ(TGImageMap)
TGRegionWithId  *gCurrentRegion; 
static TGRegion *gEmptyRegion = 0;
static Int_t gPointerX;  
static Int_t gPointerY;  
class TGRegionData : public TRefCnt {
friend class TGRegion;
private:
   Region_t   fRgn;     
   Bool_t     fIsNull;  
public:
   TGRegionData() { AddReference(); }
   ~TGRegionData() { }
   TGRegionData &operator=(const TGRegionData &r);
};
TGRegionData &TGRegionData::operator=(const TGRegionData &r)
{
   
   fRefs   = r.fRefs;
   fRgn    = r.fRgn;
   fIsNull = r.fIsNull;
   return *this;
}
TGRegion::TGRegion()
{
   
   if (!gEmptyRegion)                      
      gEmptyRegion = new TGRegion(kTRUE);
   fData = gEmptyRegion->fData;
   fData->AddReference();
}
TGRegion::TGRegion(Bool_t is_null)
{
   
   fData          = new TGRegionData;
   fData->fRgn    = gVirtualX->CreateRegion();
   fData->fIsNull = is_null;
}
TGRegion::TGRegion(Int_t x, Int_t y, UInt_t w, UInt_t h, ERegionType)
{
   
   fData          = new TGRegionData;
   fData->fRgn    = gVirtualX->CreateRegion();
   fData->fIsNull = kFALSE;
   Rectangle_t xr;
   xr.fX      = (Short_t) x;
   xr.fY      = (Short_t) y;
   xr.fWidth  = (UShort_t) w;
   xr.fHeight = (UShort_t) h;
   gVirtualX->UnionRectWithRegion(&xr, fData->fRgn, fData->fRgn);
}
TGRegion::TGRegion(Int_t n, TPoint *points, Bool_t winding)
{
   
   fData            = new TGRegionData;
   fData->fIsNull   = kFALSE;
   Point_t *gpoints = new Point_t[n];
   for (int i = 0; i < n; i++) {
      gpoints[i].fX = (Short_t) points[i].GetX();
      gpoints[i].fY = (Short_t) points[i].GetY();
   }
   fData->fRgn = gVirtualX->PolygonRegion(gpoints, n, winding);
}
TGRegion::TGRegion(const TArrayS &x, const TArrayS &y, Bool_t winding)
{
   
   fData          = new TGRegionData;
   fData->fIsNull = kFALSE;
   Int_t n = x.GetSize();
   if (n != y.GetSize()) {
      Error("TGRegion", "x and y arrays must have same length");
      return;
   }
   Point_t *gpoints = new Point_t[n];
   for (int i = 0; i < n; i++) {
      gpoints[i].fX = x.GetArray()[i];
      gpoints[i].fY = y.GetArray()[i];
   }
   fData->fRgn = gVirtualX->PolygonRegion(gpoints, n, winding);
}
TGRegion::TGRegion(Int_t n, Int_t *x, Int_t *y, Bool_t winding)
{
   
   fData          = new TGRegionData;
   fData->fIsNull = kFALSE;
   Point_t *gpoints = new Point_t[n];
   for (int i = 0; i < n; i++) {
      gpoints[i].fX = x[i];
      gpoints[i].fY = y[i];
   }
   fData->fRgn = gVirtualX->PolygonRegion(gpoints, n, winding);
}
TGRegion::TGRegion(const TGRegion &r) : TObject(r)
{
   
   fData = r.fData;
   fData->AddReference();
}
TGRegion::~TGRegion()
{
   
   if (fData->RemoveReference() <= 0) {
      gVirtualX->DestroyRegion(fData->fRgn);
      delete fData;
   }
}
TGRegion &TGRegion::operator=(const TGRegion &r)
{
   
   if (this != &r) {
      r.fData->AddReference();
      if (fData->RemoveReference() <= 0) {
         gVirtualX->DestroyRegion(fData->fRgn);
         delete fData;
      }
      fData = r.fData;
   }
   return *this;
}
TGRegion TGRegion::CopyRegion() const
{
   
   TGRegion r(fData->fIsNull);
   gVirtualX->UnionRegion(fData->fRgn, r.fData->fRgn, r.fData->fRgn);
   return r;
}
Bool_t TGRegion::IsNull() const
{
   
   return fData->fIsNull;
}
Bool_t TGRegion::IsEmpty() const
{
   
   return fData->fIsNull || gVirtualX->EmptyRegion(fData->fRgn);
}
Bool_t TGRegion::Contains(const TPoint &p) const
{
   
   return gVirtualX->PointInRegion((Int_t)p.GetX(), (Int_t)p.GetY(), fData->fRgn);
}
Bool_t TGRegion::Contains(Int_t x, Int_t y) const
{
   
   return gVirtualX->PointInRegion(x, y, fData->fRgn);
}
TGRegion TGRegion::Unite(const TGRegion &r) const
{
   
   TGRegion result(kFALSE);
   gVirtualX->UnionRegion(fData->fRgn, r.fData->fRgn, result.fData->fRgn);
   return result;
}
TGRegion TGRegion::Intersect(const TGRegion &r) const
{
   
   TGRegion result(kFALSE);
   gVirtualX->IntersectRegion(fData->fRgn, r.fData->fRgn, result.fData->fRgn);
   return result;
}
TGRegion TGRegion::Subtract(const TGRegion &r) const
{
   
   TGRegion result(kFALSE);
   gVirtualX->SubtractRegion(fData->fRgn, r.fData->fRgn, result.fData->fRgn);
   return result;
}
TGRegion TGRegion::Eor(const TGRegion &r) const
{
   
   
   TGRegion result(kFALSE);
   gVirtualX->XorRegion(fData->fRgn, r.fData->fRgn, result.fData->fRgn);
   return result;
}
TGDimension TGRegion::GetDimension() const
{
   
   Rectangle_t r = { 0, 0, 0, 0 };
   gVirtualX->GetRegionBox(fData->fRgn, &r);
   return TGDimension(r.fWidth, r.fHeight);
}
TGPosition TGRegion::GetPosition() const
{
   
   Rectangle_t r = { 0, 0, 0, 0 };
   gVirtualX->GetRegionBox(fData->fRgn, &r);
   return TGPosition(r.fX, r.fY);
}
Bool_t TGRegion::operator==(const TGRegion &r) const
{
   
   return fData == r.fData ?
             kTRUE : gVirtualX->EqualRegion(fData->fRgn, r.fData->fRgn);
}
TGRegionWithId::TGRegionWithId() : TGRegion()
{
   
   fId    = 0;
   fTip   = 0;
   fPopup = 0;
}
TGRegionWithId::TGRegionWithId(Int_t id, Int_t x, Int_t y,
                               UInt_t w, UInt_t h, ERegionType type) :
   TGRegion(x, y, w, h, type)
{
   
   fId    = id;
   fTip   = 0;
   fPopup = 0;
}
TGRegionWithId::TGRegionWithId(Int_t id, Int_t n, TPoint *points,
                               Bool_t winding) :
   TGRegion(n, points, winding)
{
   
   fId    = id;
   fTip   = 0;
   fPopup = 0;
}
TGRegionWithId::TGRegionWithId(const TGRegionWithId ®) : TGRegion(reg)
{
   
   fId    = reg.GetId();
   fTip   = 0;
   fPopup = 0;
}
TGRegionWithId::TGRegionWithId(const TGRegion ®, Int_t id) :
   TGRegion(reg)
{
   
   fId    = id;
   fTip   = 0;
   fPopup = 0;
}
TGRegionWithId::~TGRegionWithId()
{
   
   delete fTip;
}
void TGRegionWithId::DisplayPopup()
{
   
   if (fPopup) fPopup->PlaceMenu(gPointerX, gPointerY, kFALSE, kTRUE);
}
void TGRegionWithId::SetToolTipText(const char *text, Long_t delayms,
                                    const TGFrame *frame)
{
   
   
   
   if (fTip) {
      delete fTip;
      fTip = 0;
   }
   if (text && strlen(text))
      fTip = new TGToolTip(gClient->GetDefaultRoot(), frame, text, delayms);
}
TGImageMap::TGImageMap(const TGWindow *p, const TGPicture *pic) :
   TGPictureButton(p, pic)
{
   
   fCursorMouseOut  = kPointer;
   fCursorMouseOver = kHand;
   fListOfRegions   = new TList;
   fTrash           = new TList;
   fMainTip         = 0;
   fNavMode = kNavRegions;
   SetDisabledPicture(fPic);
   SetState(kButtonDisabled);
   gVirtualX->GrabButton(fId, kAnyButton, kAnyModifier,
                         kButtonPressMask | kButtonReleaseMask |
                         kPointerMotionMask, kNone, kNone);
   AddInput(kKeyPressMask | kKeyReleaseMask | kPointerMotionMask |
            kStructureNotifyMask | kLeaveWindowMask);
   SetWindowName();
}
TGImageMap::TGImageMap(const TGWindow *p, const TString &pic) :
   TGPictureButton(p, pic.Data())
{
   
   fCursorMouseOut  = kPointer;
   fCursorMouseOver = kHand;
   fListOfRegions   = new TList;
   fTrash           = new TList;
   fMainTip         = 0;
   fNavMode = kNavRegions;
   SetDisabledPicture(fPic);
   SetState(kButtonDisabled);
   gVirtualX->GrabButton(fId, kAnyButton, kAnyModifier,
                         kButtonPressMask | kButtonReleaseMask |
                         kPointerMotionMask, kNone, kNone);
   AddInput(kKeyPressMask | kKeyReleaseMask | kPointerMotionMask |
            kStructureNotifyMask | kLeaveWindowMask);
   SetWindowName();
}
TGImageMap::~TGImageMap()
{
   
   delete fMainTip;
   fTrash->Delete();
   delete fTrash;
   fListOfRegions->Delete();
   delete fListOfRegions;
}
void TGImageMap::AddRegion(const TGRegion ®ion, Int_t id)
{
   
   fListOfRegions->Add(new TGRegionWithId(region, id));
}
TGPopupMenu *TGImageMap::CreatePopup(Int_t id)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   TGPopupMenu    *popup = 0;
   TGPopupMenu    *newpopup = 0;
   while ((region = (TGRegionWithId*)next())) {
      if (id == region->GetId()) {
         popup = region->GetPopup();
         if (!popup && !newpopup) {
            newpopup = new TGPopupMenu(this);
            fTrash->Add(newpopup);
         }
         if (newpopup) region->SetPopup(newpopup);
      }
   }
   return newpopup ? newpopup : popup;
}
TGPopupMenu *TGImageMap::GetPopup(Int_t id)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   while ((region = (TGRegionWithId*)next())) {
      if (id == region->GetId()) return region->GetPopup();
   }
   return 0;
}
Bool_t TGImageMap::HandleMotion(Event_t *event)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   if (fNavMode != kNavRegions) return kTRUE;
   gPointerX = event->fX;
   gPointerY = event->fY;
   while ((region = (TGRegionWithId*)next())) {
      if (region->Contains(gPointerX, gPointerY)) {
         if (fLastVisited == region->GetId()) return kTRUE;
         if (fLastVisited) OnMouseOut(fLastVisited);
         fLastVisited = region->GetId();
         fTip = region->GetToolTipText();
         gCurrentRegion = region;
         OnMouseOver(fLastVisited);
         return kTRUE;
      }
   }
   if (fLastVisited) {
      OnMouseOut(fLastVisited);
      fTip = fMainTip;
   }
   fLastVisited = 0;  
   return kTRUE;
}
Bool_t TGImageMap::HandleDoubleClick(Event_t *event)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   if (fTip) fTip->Hide();
   if (event->fCode != kButton1 ) return kTRUE;
   if (fNavMode != kNavRegions) return kTRUE;
   gPointerX = event->fX;
   gPointerY = event->fY;
   while ((region = (TGRegionWithId*)next())) {
      if (region->Contains(gPointerX, gPointerY)) {
         DoubleClicked(region->GetId());
         gCurrentRegion = region;
         return kTRUE;
      }
   }
   DoubleClicked();
   return kTRUE;
}
Bool_t TGImageMap::HandleButton(Event_t *event)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   TGPopupMenu *pop;
   if (fTip) fTip->Hide();
   if (fNavMode != kNavRegions) return kTRUE;
   gPointerX = event->fX;
   gPointerY = event->fY;
   while ((region = (TGRegionWithId*)next())) {
      if (region->Contains(gPointerX, gPointerY)) {
         gCurrentRegion = region;
         if (event->fType == kButtonPress) {
            if (event->fCode == kButton1 )
               RegionClicked(region->GetId());
            else if (event->fCode == kButton3 ) {
               pop = region->GetPopup();
               if (pop) pop->PlaceMenu(gPointerX, gPointerY, kFALSE, kTRUE);
            }
         }
         return kTRUE;
      }
   }
   if (event->fType == kButtonPress)
      Clicked();
   return kTRUE;
}
void TGImageMap::SetToolTipText(const char *text, Long_t delayms)
{
   
   if (fMainTip) delete fMainTip;
   fMainTip = 0;
   if (text && strlen(text))
      fMainTip = new TGToolTip(fClient->GetDefaultRoot(), this, text, delayms);
}
void TGImageMap::SetToolTipText(Int_t id, const char *text, Long_t delayms)
{
   
   TIter next(fListOfRegions);
   TGRegionWithId *region;
   while ((region = (TGRegionWithId*)next())) {
      if (id == region->GetId())
         region->SetToolTipText(text, delayms, this);
   }
}
void TGImageMap::OnMouseOver(Int_t id)
{
   
   
   if (fTip) fTip->Reset();
   if (fMainTip) fMainTip->Hide();
   gVirtualX->SetCursor(fId, gVirtualX->CreateCursor(fCursorMouseOver));
   Emit("OnMouseOver(Int_t)", id);
}
void TGImageMap::OnMouseOut(Int_t id)
{
   
   
   if(fTip) fTip->Hide();
   if(fMainTip) fMainTip->Reset();
   gVirtualX->SetCursor(fId,gVirtualX->CreateCursor(fCursorMouseOut));
   Emit("OnMouseOut(Int_t)",id);
}
void TGImageMap::RegionClicked(Int_t id)
{
   
   
   Emit("RegionClicked(Int_t)",id);
}
void TGImageMap::DoubleClicked()
{
   
   
   Emit("DoubleClicked()");
}
void TGImageMap::DoubleClicked(Int_t id)
{
   
   
   Emit("DoubleClicked(Int_t)",id);
}
This page has been automatically generated. If you have any comments or suggestions about the page layout send a mail to ROOT support, or contact the developers with any questions or problems regarding ROOT.