ctkTransferFunctionControlPointsItem.cpp 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) 2010 Kitware Inc.
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.commontk.org/LICENSE
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. /// Qt includes
  15. #include <QColor>
  16. #include <QDebug>
  17. #include <QLinearGradient>
  18. #include <QGraphicsSceneMouseEvent>
  19. #include <QPainter>
  20. #include <QtGlobal>
  21. #include <QVariant>
  22. /// CTK includes
  23. #include "ctkTransferFunctionControlPointsItem.h"
  24. #include "ctkTransferFunctionScene.h"
  25. #include "ctkTransferFunctionWidget.h"
  26. //-----------------------------------------------------------------------------
  27. class ctkTransferFunctionControlPointsItemPrivate:
  28. public ctkPrivate<ctkTransferFunctionControlPointsItem>
  29. {
  30. public:
  31. ctkTransferFunctionControlPointsItemPrivate();
  32. void init();
  33. QList<QPointF> ControlPoints;
  34. QSize PointSize;
  35. int SelectedPoint;
  36. };
  37. //-----------------------------------------------------------------------------
  38. ctkTransferFunctionControlPointsItemPrivate::ctkTransferFunctionControlPointsItemPrivate()
  39. {
  40. this->PointSize = QSize(12,12);
  41. this->SelectedPoint = -1;
  42. }
  43. //-----------------------------------------------------------------------------
  44. void ctkTransferFunctionControlPointsItemPrivate::init()
  45. {
  46. CTK_P(ctkTransferFunctionControlPointsItem);
  47. p->setAcceptedMouseButtons(Qt::LeftButton);
  48. }
  49. //-----------------------------------------------------------------------------
  50. ctkTransferFunctionControlPointsItem::ctkTransferFunctionControlPointsItem(QGraphicsItem* parentGraphicsItem)
  51. :ctkTransferFunctionItem(parentGraphicsItem)
  52. {
  53. CTK_INIT_PRIVATE(ctkTransferFunctionControlPointsItem);
  54. ctk_d()->init();
  55. }
  56. //-----------------------------------------------------------------------------
  57. ctkTransferFunctionControlPointsItem::ctkTransferFunctionControlPointsItem(
  58. ctkTransferFunction* transferFunction, QGraphicsItem* parentItem)
  59. :ctkTransferFunctionItem(transferFunction, parentItem)
  60. {
  61. CTK_INIT_PRIVATE(ctkTransferFunctionControlPointsItem);
  62. ctk_d()->init();
  63. }
  64. //-----------------------------------------------------------------------------
  65. ctkTransferFunctionControlPointsItem::~ctkTransferFunctionControlPointsItem()
  66. {
  67. }
  68. //-----------------------------------------------------------------------------
  69. void ctkTransferFunctionControlPointsItem::paint(
  70. QPainter * painter, const QStyleOptionGraphicsItem * option, QWidget * widget)
  71. {
  72. CTK_D(ctkTransferFunctionControlPointsItem);
  73. int count = this->transferFunction() ? this->transferFunction()->count() : 0;
  74. if (count <= 0)
  75. {
  76. return;
  77. }
  78. ctkTransferFunctionScene* tfScene = dynamic_cast<ctkTransferFunctionScene*>(this->scene());
  79. Q_ASSERT(tfScene);
  80. const QPainterPath& curve = tfScene->curve();
  81. QPen pen(QColor(255, 255, 255, 191), 1);
  82. pen.setCosmetic(true);
  83. painter->setPen(pen);
  84. painter->drawPath(curve);
  85. d->ControlPoints = tfScene->points();
  86. painter->setBrush(QBrush(QColor(191, 191, 191, 127)));
  87. painter->save();
  88. QTransform transform = painter->transform();
  89. painter->setTransform(QTransform());
  90. foreach(const QPointF& point, d->ControlPoints)
  91. {
  92. QPointF pos = transform.map(point);
  93. painter->drawEllipse(pos.x() - d->PointSize.width() / 2,
  94. pos.y() - d->PointSize.height() / 2,
  95. d->PointSize.width(), d->PointSize.width());
  96. //points.addEllipse(point, d->PointSize.width(), d->PointSize.height());
  97. }
  98. painter->restore();
  99. }
  100. //-----------------------------------------------------------------------------
  101. void ctkTransferFunctionControlPointsItem::mousePressEvent(QGraphicsSceneMouseEvent* e)
  102. {
  103. CTK_D(ctkTransferFunctionControlPointsItem);
  104. QWidget* w = e->widget();
  105. ctkTransferFunctionWidget* view = qobject_cast<ctkTransferFunctionWidget*>(e->widget()->parentWidget());
  106. Q_ASSERT(view);
  107. // e->pos() is ok, pointArea should be in the world coordiate
  108. QRect pointViewArea(QPoint(-d->PointSize.width() / 2, -d->PointSize.height() / 2), d->PointSize);
  109. QPolygonF pointItemArea = this->mapFromScene(view->mapToScene(pointViewArea));
  110. d->SelectedPoint = -1;
  111. for(int i = 0; i < d->ControlPoints.count(); ++i)
  112. {
  113. if (pointItemArea.translated(d->ControlPoints[i]).containsPoint(e->pos(), Qt::OddEvenFill))
  114. {
  115. d->SelectedPoint = i;
  116. break;
  117. }
  118. }
  119. if (d->SelectedPoint >= 0)
  120. {
  121. return;
  122. }
  123. ctkTransferFunctionScene* tfScene = dynamic_cast<ctkTransferFunctionScene*>(this->scene());
  124. Q_ASSERT(tfScene);
  125. // convert coordinates
  126. QPointF tfPos = tfScene->mapPointFromScene(e->pos());
  127. // add point to transfer function
  128. // returns index
  129. int index = this->transferFunction()->insertControlPoint( tfPos.x());
  130. //NOT WORKING IN COMPOSITE
  131. // update value of the point
  132. if (!QSharedPointer<ctkControlPoint>(this->transferFunction()->controlPoint(index))->value().canConvert<QColor>())
  133. {
  134. this->transferFunction()->setControlPointValue( index, tfPos.y());
  135. }
  136. }
  137. //-----------------------------------------------------------------------------
  138. void ctkTransferFunctionControlPointsItem::mouseMoveEvent(QGraphicsSceneMouseEvent* e)
  139. {
  140. CTK_D(ctkTransferFunctionControlPointsItem);
  141. if (d->SelectedPoint < 0)
  142. {
  143. e->ignore();
  144. return;
  145. }
  146. ctkTransferFunctionScene* tfScene = dynamic_cast<ctkTransferFunctionScene*>(this->scene());
  147. Q_ASSERT(tfScene);
  148. QPointF newPos = tfScene->mapPointFromScene(e->pos());
  149. // Deal with borders
  150. if(d->SelectedPoint == 0 || d->SelectedPoint == this->transferFunction()->count() )
  151. {
  152. // BEHAVIOR TO BE IMPLEMENTED
  153. // int borderBehavior = tfScene->borderBehavior();
  154. // LockBorder
  155. // Create new point on border at same height
  156. // Create new point on border on top
  157. // Create new point on border on bottom
  158. qDebug() << "border" ;
  159. return;
  160. }
  161. else
  162. {
  163. // TO BE IMPLEMENTED
  164. //int movePointsBehavior = tfScene->movePointsBehavior();
  165. // initialize to BLOCK_MOVE for now
  166. int movePointsBehavior = STOP_MOVE;
  167. switch( movePointsBehavior ){
  168. case STOP_MOVE:
  169. stopPoints(newPos);
  170. break;
  171. case SWITCH_MOVE:
  172. switchPoints(newPos);
  173. break;
  174. case DRAW_MOVE:
  175. drawPoints(newPos);
  176. break;
  177. case FUSION_MOVE:
  178. fusionPoints(newPos);
  179. break;
  180. default:
  181. break;
  182. }
  183. }
  184. }
  185. //-----------------------------------------------------------------------------
  186. void ctkTransferFunctionControlPointsItem::stopPoints( QPointF iPointF )
  187. {
  188. CTK_D(ctkTransferFunctionControlPointsItem);
  189. if( this->transferFunction()->controlPoint(d->SelectedPoint - 1)->x() > iPointF.x() ||
  190. this->transferFunction()->controlPoint(d->SelectedPoint + 1)->x() < iPointF.x())
  191. {
  192. return;
  193. }
  194. updatePointPosition(iPointF);
  195. }
  196. //-----------------------------------------------------------------------------
  197. void ctkTransferFunctionControlPointsItem::switchPoints( QPointF iPointF )
  198. {
  199. CTK_D(ctkTransferFunctionControlPointsItem);
  200. // Increment or decrement selected point?
  201. // Don't need to check borders since it is done just before calling this method...
  202. if( this->transferFunction()->controlPoint(d->SelectedPoint - 1)->x() > iPointF.x() )
  203. {
  204. // NOT WORKING IF COMPOSITE TRANSFER FUNCTION
  205. double value = this->transferFunction()->value(d->SelectedPoint-1 ).toDouble();
  206. d->SelectedPoint -= 1;
  207. this->transferFunction()->setControlPointValue(d->SelectedPoint+1, value);
  208. }
  209. else if ( this->transferFunction()->controlPoint(d->SelectedPoint + 1)->x() < iPointF.x() )
  210. {
  211. double value = this->transferFunction()->value(d->SelectedPoint + 1 ).toDouble();
  212. d->SelectedPoint += 1;
  213. this->transferFunction()->setControlPointValue(d->SelectedPoint-1, value);
  214. }
  215. updatePointPosition(iPointF);
  216. }
  217. //-----------------------------------------------------------------------------
  218. void ctkTransferFunctionControlPointsItem::drawPoints( QPointF iPointF )
  219. {
  220. CTK_D(ctkTransferFunctionControlPointsItem);
  221. // Increment or decrement selected point
  222. if( this->transferFunction()->controlPoint(d->SelectedPoint - 1)->x() > iPointF.x() )
  223. {
  224. //change the selected point
  225. d->SelectedPoint = d->SelectedPoint -1;
  226. }
  227. else if ( this->transferFunction()->controlPoint(d->SelectedPoint + 1)->x() < iPointF.x() )
  228. {
  229. d->SelectedPoint = d->SelectedPoint +1;
  230. }
  231. updatePointPosition(iPointF);
  232. }
  233. //-----------------------------------------------------------------------------
  234. void ctkTransferFunctionControlPointsItem::fusionPoints( QPointF iPointF )
  235. {
  236. // TO BE IMPLEMENTED
  237. // if 2 points are to close: delete one..?
  238. }
  239. //-----------------------------------------------------------------------------
  240. void ctkTransferFunctionControlPointsItem::updatePointPosition( QPointF iPoint )
  241. {
  242. CTK_D(ctkTransferFunctionControlPointsItem);
  243. this->transferFunction()->setControlPointPos(d->SelectedPoint, iPoint.x());
  244. // TEST NOT WORKING IN COMPOSITE TRANSFER FUNCTION
  245. if (!QSharedPointer<ctkControlPoint>(this->transferFunction()->controlPoint(d->SelectedPoint))->value().canConvert<QColor>())
  246. {
  247. this->transferFunction()->setControlPointValue(d->SelectedPoint, iPoint.y());
  248. }
  249. }
  250. //-----------------------------------------------------------------------------
  251. void ctkTransferFunctionControlPointsItem::mouseReleaseEvent(QGraphicsSceneMouseEvent* e)
  252. {
  253. CTK_D(ctkTransferFunctionControlPointsItem);
  254. if (d->SelectedPoint < 0)
  255. {
  256. e->ignore();
  257. return;
  258. }
  259. d->SelectedPoint = -1;
  260. }