Blob Blame Raw
#include "ext/OverallDesigner.h"
//#include "ext/StrokeParametricDeformer.h"
#include "ext/StrokeDeformation.h"
#include "ext/SmoothDeformation.h"
#include "ext/CornerDeformation.h"
#include "ext/StraightCornerDeformation.h"
#include "ext/Selector.h"
#include "ext/ContextStatus.h"

#include "tcurves.h"

#include <tstroke.h>
#include <tgl.h>
#include <tgeometry.h>
#include <drawutil.h>
#include <sstream>

/*
#ifdef  _DEBUG
#undef  _DEBUG
#define RESTORE_DEBUG
#endif
//*/

//-----------------------------------------------------------------------------

namespace {
void extglDrawText(const TPointD &p, const std::string &s,
                   void *character = GLUT_BITMAP_TIMES_ROMAN_10) {
  glPushMatrix();
  glTranslated(p.x, p.y, 0);
  if (character <= GLUT_STROKE_MONO_ROMAN) {
    double factor = 0.07;
    glScaled(factor, factor, factor);
    for (int i = 0; i < (int)s.size(); i++)
      glutStrokeCharacter(character, s[i]);
  } else
    for (int i = 0; i < (int)s.size(); i++)
      glutBitmapCharacter(character, s[i]);

  glPopMatrix();
}

//---------------------------------------------------------------------------

void drawSign(const TPointD &p, const TPointD &v, double size) {
  if (size == 0.0) return;
  size = fabs(size);

  TPointD v90  = rotate90(v);
  TPointD v270 = rotate270(v);

  glBegin(GL_LINES);
  glVertex2d(p.x, p.y);
  v90 = normalize(v90);
  v90 = v90 * size;
  v90 = p + v90;
  glVertex2d(v90.x, v90.y);
  glEnd();
  glBegin(GL_LINES);
  glVertex2d(p.x, p.y);
  v270 = normalize(v270);
  v270 = v270 * size;
  v270 = p + v270;
  glVertex2d(v270.x, v270.y);
  glEnd();
}

//---------------------------------------------------------------------------

void drawCross(const TPointD &p, double size) {
  TPointD v1 = TPointD(1, 1);
  TPointD v2 = TPointD(1, -1);

  drawSign(p, v1, size);
  drawSign(p, v2, size);
}

//---------------------------------------------------------------------------

/**
   * Try to verify if it is possible to call
   * some method of this stroke.
   */
bool isValid(const TStroke *s) {
  if (!s) return false;
  const int cpCount = s->getControlPointCount();
  int i;
  for (i               = 0; i < cpCount; ++i) s->getControlPoint(i);
  const int chunkCount = s->getChunkCount();
  for (i = 0; i < chunkCount; ++i) s->getChunk(i);
  return true;
}

//---------------------------------------------------------------------------

void showCP(const TStroke *s, double pixelSize) {
  if (!s) return;

  // show control points
  const int countCP = s->getControlPointCount();
  TThickPoint p1;

  for (int i = 0; i < countCP; ++i) {
    p1 = s->getControlPoint(i);
    if (i & 1)
      glColor3d(1.0, 0.0, 0.0);
    else
      glColor3d(1.0, 1.0, 0.0);

    drawCross(convert(p1), p1.thick * pixelSize);
    tglDrawCircle(convert(p1), p1.thick * pixelSize);
  }
}

//---------------------------------------------------------------------------

void drawStrokeCenterLine(const TStroke *stroke, double pixelSize,
                          const ToonzExt::Interval &vals) {
  double from = vals.first, to = vals.second;

  if (!stroke || pixelSize < 0.0) return;

  from = std::max(std::min(from, 1.0), 0.0);
  to   = std::max(std::min(to, 1.0), 0.0);

  if (from < to) {
    drawStrokeCenterline(*stroke, pixelSize, from, to);
  } else {
    drawStrokeCenterline(*stroke, pixelSize, from, 1.0);
    drawStrokeCenterline(*stroke, pixelSize, 0.0, to);
  }
  tglDrawDisk(stroke->getPoint(from), 5.0 * pixelSize);
  tglDrawDisk(stroke->getPoint(to), 5.0 * pixelSize);

#ifdef _DEBUG
  glColor3d(1.0, 0.0, 0.0);
  tglDrawDisk(stroke->getPoint(0.0), 9.0 * pixelSize);
#endif
}

//---------------------------------------------------------------------------

void showCorners(const TStroke *s, int cornerSize, double pixelSize) {
  if (!s) return;

  const TPointD offset(20, 20);

  // show corners
  std::vector<double> corners;

  ToonzExt::cornersDetector(s, cornerSize, corners);
  if (corners.empty()) return;

  int i, size = corners.size();

  glColor3d(0.8, 1.0, 0.3);
  // find interval with corner like extremes
  for (i = 0; i < size; ++i) {
    double tmp   = corners[i];
    TPointD pnt  = s->getPoint(tmp);
    double large = 5.0;
    drawCross(pnt, large * pixelSize);

    glColor3d(0.7, 0.7, 0.7);
    std::ostringstream oss;
    oss << "[" << tmp << "]";
    extglDrawText(TPointD(pnt.x, pnt.y) + offset, oss.str());
  }

  corners.clear();

  ToonzExt::straightCornersDetector(s, corners);
  if (corners.empty()) return;

  size = corners.size();
  glColor3d(0.8, 1.0, 0.3);
  // find interval with corner like extremes
  for (i = 0; i < size; ++i) {
    double tmp   = corners[i];
    TPointD pnt  = s->getPoint(tmp);
    double large = 5.0;
    drawCross(pnt, large * pixelSize);

    glColor3d(0.7, 0.2, 0.2);
    std::ostringstream oss;
    oss << "[" << tmp << "]";
    extglDrawText(TPointD(pnt.x, pnt.y) - offset, oss.str());
  }
}
}

//-----------------------------------------------------------------------------

ToonzExt::OverallDesigner::OverallDesigner(int x, int y) : x_(x), y_(y) {
  pixelSize_ = sqrt(this->getPixelSize2());
  scale_     = pixelSize_ != 0.0 ? pixelSize_ : 1.0;
}

//-----------------------------------------------------------------------------

ToonzExt::OverallDesigner::~OverallDesigner() {}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::draw(ToonzExt::SmoothDeformation *sd) {
#ifdef _DEBUG
  glPushMatrix();
  {
    this->setPosition();
    TCubic c;
    c.setP0(TPointD(0.0, 0.0));
    c.setP1(TPointD(4.0, 12.0));
    c.setP2(TPointD(12.0, 12.0));
    c.setP3(TPointD(16.0, 0.0));
    glColor3d(1.0, 0.0, 1.0);
    tglDraw(c, 100, GL_LINE);
  }
  glPopMatrix();
#endif
}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::draw(ToonzExt::CornerDeformation *sd) {
#ifdef _DEBUG
  glPushMatrix();
  {
    this->setPosition();
    TCubic c;
    c.setP0(TPointD(0.0, 0.0));
    c.setP1(TPointD(0.0, 12.0));
    c.setP2(TPointD(6.0, 12.0));
    c.setP3(TPointD(12.0, 12.0));
    glColor3d(1.0, 0.0, 1.0);
    tglDraw(c, 100, GL_LINE);
    c.setP0(TPointD(12.0, 12.0));
    c.setP1(TPointD(6.0, 8.0));
    c.setP2(TPointD(6.0, 4.0));
    c.setP3(TPointD(12.0, 0.0));
    tglDraw(c, 100, GL_LINE);
  }
  glPopMatrix();
#endif
}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::draw(ToonzExt::StraightCornerDeformation *sd) {
#ifdef _DEBUG
  glPushMatrix();
  {
    this->setPosition();
    glColor3d(0.0, 1.0, 1.0);
    glBegin(GL_LINE_STRIP);
    tglVertex(TPointD(0.0, 0.0));
    tglVertex(TPointD(8.0, 12.0));
    tglVertex(TPointD(16.0, 0.0));
    glEnd();
  }
  glPopMatrix();
#endif
}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::draw(ToonzExt::StrokeDeformation *sd) {
  if (sd) {
    const TStroke *s;

    // glColor3d(1.0,0.0,1.0);
    s = sd->getCopiedStroke();
    if (s) {
      const ContextStatus *status = sd->getStatus();
      double w = 0.0, pixelSize = 1.0;

      if (status) {
        w         = status->w_;
        pixelSize = status->pixelSize_ < 0 ? 1.0 : status->pixelSize_;
      }
#ifdef _DEBUG
      drawCross(s->getPoint(0), 2 * pixelSize);

      tglDrawCircle(s->getPoint(w), 8 * pixelSize);
      drawCross(s->getPoint(w), 8 * pixelSize);
#endif
      ToonzExt::Interval ex = sd->getExtremes();
      drawStrokeCenterLine(s, pixelSize_, ex);
      if (status) {
#ifdef _DEBUG
        glColor3d(0, 0, 0);
        showCorners(s, status->cornerSize_, pixelSize);

        glColor3d(0, 0, 0);
        TPointD offset = normalize(TPointD(1.0, 1.0)) * 20.0;
        std::ostringstream oss;
        oss << "(" << this->x_ << "," << this->y_ << ")\n{" << w << ",{"
            << sd->getExtremes().first << "," << sd->getExtremes().second
            << "}}";
        extglDrawText(TPointD(x_, y_) + offset, oss.str());

        glColor3d(0.5, 1.0, 0.5);
        showCP(s, pixelSize);
#endif
      }
    }

    /*
glColor3d(1.0,1.0,0.0);
s = sd->getStroke();
if(s)
{
drawCross( s->getPoint(0),
     4);

drawStrokeCenterLine(s,
               pixelSize_,
               ToonzExt::Interval(0,1));
}
*/

    s = sd->getTransformedStroke();

    glColor3d(1.0, 0.0, 0.0);
    if (s) {
#ifdef _DEBUG
      isValid(s);
#endif
      drawStrokeCenterline(*s, pixelSize_);
    }

#ifdef _DEBUG
    {
      const TStroke *c = sd->getCopiedStroke(), *s = sd->getStroke();
      if (c && s) {
        // glColor3d(1,1,0);
        // tglDrawDisk(s->getPoint(0.0),
        //            5*pixelSize_);

        int count =
            std::min(c->getControlPointCount(), s->getControlPointCount());
        for (int i = 0; i < count; ++i) {
          TThickPoint ccp = c->getControlPoint(i);
          TThickPoint scp = s->getControlPoint(i);
        }
      }
    }
#endif
  }
}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::setPosition() {
  TPointD offset(1, -1);
  offset = normalize(offset) * (20.0 * scale_);
  glTranslated(x_ + offset.x, y_ + offset.y, 0.0);
  glScalef(scale_, scale_, scale_);
}

//-----------------------------------------------------------------------------

void ToonzExt::OverallDesigner::draw(ToonzExt::Selector *selector) {
  if (!selector) return;

  const TStroke *ref = selector->getStroke();

  if (!ref) return;

  double length_at_w         = ref->getLength(selector->getW()),
         emi_selector_length = selector->getLength() * 0.5,
         stroke_length       = ref->getLength();

  ToonzExt::Interval interval;

  if (ref->isSelfLoop()) {
    interval.first = length_at_w - emi_selector_length;
    if (interval.first < 0.0) interval.first = stroke_length + interval.first;

    interval.first = ref->getParameterAtLength(interval.first);

    interval.second = length_at_w + emi_selector_length;
    if (interval.second > stroke_length)
      interval.second = interval.second - stroke_length;

    interval.second = ref->getParameterAtLength(interval.second);
  } else {
    interval.first = ref->getParameterAtLength(
        std::max(0.0, length_at_w - emi_selector_length));
    interval.second = ref->getParameterAtLength(
        std::min(stroke_length, length_at_w + emi_selector_length));
  }

  float prev_line_width = 1.0;

  glGetFloatv(GL_LINE_WIDTH, &prev_line_width);

  glLineWidth(2.0);
  drawStrokeCenterLine(ref, pixelSize_, interval);
  glLineWidth(prev_line_width);
}

//-----------------------------------------------------------------------------
//  End Of File
//-----------------------------------------------------------------------------
/*
#ifdef  RESTORE_DEBUG
#undef  RESTORE_DEBUG
#define _DEBUG  1
#endif
//*/