1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672 1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914 1915 1916 1917 1918 1919 1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982 1983 1984 1985 1986 1987 1988 1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035 2036 2037 2038 2039 2040 2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054 2055 2056 2057 2058 2059 2060 2061 2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077 2078 2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114 2115 2116 2117 2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172 2173 2174 2175 2176 2177 2178 2179 2180 2181 2182 2183 2184 2185 2186 2187 2188 2189 2190 2191 2192 2193 2194 2195 2196 2197 2198 2199 2200 2201 2202 2203 2204 2205 2206 2207 2208 2209
|
/****************************************************************************
Copyright (C) 2002-2014 Gilles Debunne. All rights reserved.
This file is part of the QGLViewer library version 2.6.3.
http://www.libqglviewer.com - contact@libqglviewer.com
This file may be used under the terms of the GNU General Public License
versions 2.0 or 3.0 as published by the Free Software Foundation and
appearing in the LICENSE file included in the packaging of this file.
In addition, as a special exception, Gilles Debunne gives you certain
additional rights, described in the file GPL_EXCEPTION in this package.
libQGLViewer uses dual licensing. Commercial/proprietary software must
purchase a libQGLViewer Commercial License.
This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
*****************************************************************************/
#include "domUtils.h"
#include "camera.h"
#include "qglviewer.h"
#include "manipulatedCameraFrame.h"
using namespace std;
using namespace qglviewer;
/*! Default constructor.
sceneCenter() is set to (0,0,0) and sceneRadius() is set to 1.0. type() is Camera::PERSPECTIVE,
with a \c M_PI/4 fieldOfView().
See IODistance(), physicalDistanceToScreen(), physicalScreenWidth() and focusDistance()
documentations for default stereo parameter values. */
Camera::Camera()
: frame_(NULL), fieldOfView_(M_PI/4.0), modelViewMatrixIsUpToDate_(false), projectionMatrixIsUpToDate_(false)
{
// #CONNECTION# Camera copy constructor
interpolationKfi_ = new KeyFrameInterpolator;
// Requires the interpolationKfi_
setFrame(new ManipulatedCameraFrame());
// #CONNECTION# All these default values identical in initFromDOMElement.
// Requires fieldOfView() to define focusDistance()
setSceneRadius(1.0);
// Initial value (only scaled after this)
orthoCoef_ = tan(fieldOfView()/2.0);
// Also defines the pivotPoint(), which changes orthoCoef_. Requires a frame().
setSceneCenter(Vec(0.0, 0.0, 0.0));
// Requires fieldOfView() when called with ORTHOGRAPHIC. Attention to projectionMatrix_ below.
setType(PERSPECTIVE);
// #CONNECTION# initFromDOMElement default values
setZNearCoefficient(0.005);
setZClippingCoefficient(sqrt(3.0));
// Dummy values
setScreenWidthAndHeight(600, 400);
// Stereo parameters
setIODistance(0.062);
setPhysicalScreenWidth(0.5);
// focusDistance is set from setFieldOfView()
// #CONNECTION# Camera copy constructor
for (unsigned short j=0; j<16; ++j)
{
modelViewMatrix_[j] = ((j%5 == 0) ? 1.0 : 0.0);
// #CONNECTION# computeProjectionMatrix() is lazy and assumes 0.0 almost everywhere.
projectionMatrix_[j] = 0.0;
}
computeProjectionMatrix();
}
/*! Virtual destructor.
The frame() is deleted, but the different keyFrameInterpolator() are \e not deleted (in case they
are shared). */
Camera::~Camera()
{
delete frame_;
delete interpolationKfi_;
}
/*! Copy constructor. Performs a deep copy using operator=(). */
Camera::Camera(const Camera& camera)
: QObject(), frame_(NULL)
{
// #CONNECTION# Camera constructor
interpolationKfi_ = new KeyFrameInterpolator;
// Requires the interpolationKfi_
setFrame(new ManipulatedCameraFrame(*camera.frame()));
for (unsigned short j=0; j<16; ++j)
{
modelViewMatrix_[j] = ((j%5 == 0) ? 1.0 : 0.0);
// #CONNECTION# computeProjectionMatrix() is lazy and assumes 0.0 almost everywhere.
projectionMatrix_[j] = 0.0;
}
(*this)=camera;
}
/*! Equal operator.
All the parameters of \p camera are copied. The frame() pointer is not modified, but its
Frame::position() and Frame::orientation() are set to those of \p camera.
\attention The Camera screenWidth() and screenHeight() are set to those of \p camera. If your
Camera is associated with a QGLViewer, you should update these value after the call to this method:
\code
*(camera()) = otherCamera;
camera()->setScreenWidthAndHeight(width(), height());
\endcode
The same applies to sceneCenter() and sceneRadius(), if needed. */
Camera& Camera::operator=(const Camera& camera)
{
setScreenWidthAndHeight(camera.screenWidth(), camera.screenHeight());
setFieldOfView(camera.fieldOfView());
setSceneRadius(camera.sceneRadius());
setSceneCenter(camera.sceneCenter());
setZNearCoefficient(camera.zNearCoefficient());
setZClippingCoefficient(camera.zClippingCoefficient());
setType(camera.type());
// Stereo parameters
setIODistance(camera.IODistance());
setFocusDistance(camera.focusDistance());
setPhysicalScreenWidth(camera.physicalScreenWidth());
orthoCoef_ = camera.orthoCoef_;
projectionMatrixIsUpToDate_ = false;
// frame_ and interpolationKfi_ pointers are not shared.
frame_->setReferenceFrame(NULL);
frame_->setPosition(camera.position());
frame_->setOrientation(camera.orientation());
interpolationKfi_->resetInterpolation();
kfi_ = camera.kfi_;
computeProjectionMatrix();
computeModelViewMatrix();
return *this;
}
/*! Sets Camera screenWidth() and screenHeight() (expressed in pixels).
You should not call this method when the Camera is associated with a QGLViewer, since the
latter automatically updates these values when it is resized (hence overwritting your values).
Non-positive dimension are silently replaced by a 1 pixel value to ensure frustrum coherence.
If your Camera is used without a QGLViewer (offscreen rendering, shadow maps), use setAspectRatio()
instead to define the projection matrix. */
void Camera::setScreenWidthAndHeight(int width, int height)
{
// Prevent negative and zero dimensions that would cause divisions by zero.
screenWidth_ = width > 0 ? width : 1;
screenHeight_ = height > 0 ? height : 1;
projectionMatrixIsUpToDate_ = false;
}
/*! Returns the near clipping plane distance used by the Camera projection matrix.
The clipping planes' positions depend on the sceneRadius() and sceneCenter() rather than being fixed
small-enough and large-enough values. A good scene dimension approximation will hence result in an
optimal precision of the z-buffer.
The near clipping plane is positioned at a distance equal to zClippingCoefficient() * sceneRadius()
in front of the sceneCenter():
\code
zNear = distanceToSceneCenter() - zClippingCoefficient()*sceneRadius();
\endcode
In order to prevent negative or too small zNear() values (which would degrade the z precision),
zNearCoefficient() is used when the Camera is inside the sceneRadius() sphere:
\code
const qreal zMin = zNearCoefficient() * zClippingCoefficient() * sceneRadius();
if (zNear < zMin)
zNear = zMin;
// With an ORTHOGRAPHIC type, the value is simply clamped to 0.0
\endcode
See also the zFar(), zClippingCoefficient() and zNearCoefficient() documentations.
If you need a completely different zNear computation, overload the zNear() and zFar() methods in a
new class that publicly inherits from Camera and use QGLViewer::setCamera():
\code
class myCamera :: public qglviewer::Camera
{
virtual qreal Camera::zNear() const { return 0.001; };
virtual qreal Camera::zFar() const { return 100.0; };
}
\endcode
See the <a href="../examples/standardCamera.html">standardCamera example</a> for an application.
\attention The value is always positive although the clipping plane is positioned at a negative z
value in the Camera coordinate system. This follows the \c gluPerspective standard. */
qreal Camera::zNear() const
{
const qreal zNearScene = zClippingCoefficient() * sceneRadius();
qreal z = distanceToSceneCenter() - zNearScene;
// Prevents negative or null zNear values.
const qreal zMin = zNearCoefficient() * zNearScene;
if (z < zMin)
switch (type())
{
case Camera::PERSPECTIVE : z = zMin; break;
case Camera::ORTHOGRAPHIC : z = 0.0; break;
}
return z;
}
/*! Returns the far clipping plane distance used by the Camera projection matrix.
The far clipping plane is positioned at a distance equal to zClippingCoefficient() * sceneRadius()
behind the sceneCenter():
\code
zFar = distanceToSceneCenter() + zClippingCoefficient()*sceneRadius();
\endcode
See the zNear() documentation for details. */
qreal Camera::zFar() const
{
return distanceToSceneCenter() + zClippingCoefficient() * sceneRadius();
}
/*! Sets the vertical fieldOfView() of the Camera (in radians).
Note that focusDistance() is set to sceneRadius() / tan(fieldOfView()/2) by this method. */
void Camera::setFieldOfView(qreal fov) {
fieldOfView_ = fov;
setFocusDistance(sceneRadius() / tan(fov/2.0));
projectionMatrixIsUpToDate_ = false;
}
/*! Defines the Camera type().
Changing the camera Type alters the viewport and the objects' sizes can be changed.
This method garantees that the two frustum match in a plane normal to viewDirection(), passing through the pivotPoint().
Prefix the type with \c Camera if needed, as in:
\code
camera()->setType(Camera::ORTHOGRAPHIC);
// or even qglviewer::Camera::ORTHOGRAPHIC if you do not use namespace
\endcode */
void Camera::setType(Type type)
{
// make ORTHOGRAPHIC frustum fit PERSPECTIVE (at least in plane normal to viewDirection(), passing
// through RAP). Done only when CHANGING type since orthoCoef_ may have been changed with a
// setPivotPoint() in the meantime.
if ( (type == Camera::ORTHOGRAPHIC) && (type_ == Camera::PERSPECTIVE) )
orthoCoef_ = tan(fieldOfView()/2.0);
type_ = type;
projectionMatrixIsUpToDate_ = false;
}
/*! Sets the Camera frame().
If you want to move the Camera, use setPosition() and setOrientation() or one of the Camera
positioning methods (lookAt(), fitSphere(), showEntireScene()...) instead.
If you want to save the Camera position(), there's no need to call this method either. Use
addKeyFrameToPath() and playPath() instead.
This method is actually mainly useful if you derive the ManipulatedCameraFrame class and want to
use an instance of your new class to move the Camera.
A \c NULL \p mcf pointer will silently be ignored. The calling method is responsible for
deleting the previous frame() pointer if needed in order to prevent memory leaks. */
void Camera::setFrame(ManipulatedCameraFrame* const mcf)
{
if (!mcf)
return;
if (frame_) {
disconnect(frame_, SIGNAL(modified()), this, SLOT(onFrameModified()));
}
frame_ = mcf;
interpolationKfi_->setFrame(frame());
connect(frame_, SIGNAL(modified()), this, SLOT(onFrameModified()));
onFrameModified();
}
/*! Returns the distance from the Camera center to sceneCenter(), projected along the Camera Z axis.
Used by zNear() and zFar() to optimize the Z range. */
qreal Camera::distanceToSceneCenter() const
{
return fabs((frame()->coordinatesOf(sceneCenter())).z);
}
/*! Returns the \p halfWidth and \p halfHeight of the Camera orthographic frustum.
These values are only valid and used when the Camera is of type() Camera::ORTHOGRAPHIC. They are
expressed in OpenGL units and are used by loadProjectionMatrix() to define the projection matrix
using:
\code
glOrtho( -halfWidth, halfWidth, -halfHeight, halfHeight, zNear(), zFar() )
\endcode
These values are proportional to the Camera (z projected) distance to the pivotPoint().
When zooming on the object, the Camera is translated forward \e and its frustum is narrowed, making
the object appear bigger on screen, as intuitively expected.
Overload this method to change this behavior if desired, as is done in the
<a href="../examples/standardCamera.html">standardCamera example</a>. */
void Camera::getOrthoWidthHeight(GLdouble& halfWidth, GLdouble& halfHeight) const
{
const qreal dist = orthoCoef_ * fabs(cameraCoordinatesOf(pivotPoint()).z);
//#CONNECTION# fitScreenRegion
halfWidth = dist * ((aspectRatio() < 1.0) ? 1.0 : aspectRatio());
halfHeight = dist * ((aspectRatio() < 1.0) ? 1.0/aspectRatio() : 1.0);
}
/*! Computes the projection matrix associated with the Camera.
If type() is Camera::PERSPECTIVE, defines a \c GL_PROJECTION matrix similar to what would \c
gluPerspective() do using the fieldOfView(), window aspectRatio(), zNear() and zFar() parameters.
If type() is Camera::ORTHOGRAPHIC, the projection matrix is as what \c glOrtho() would do.
Frustum's width and height are set using getOrthoWidthHeight().
Both types use zNear() and zFar() to place clipping planes. These values are determined from
sceneRadius() and sceneCenter() so that they best fit the scene size.
Use getProjectionMatrix() to retrieve this matrix. Overload loadProjectionMatrix() if you want your
Camera to use an exotic projection matrix.
\note You must call this method if your Camera is not associated with a QGLViewer and is used for
offscreen computations (using (un)projectedCoordinatesOf() for instance). loadProjectionMatrix()
does it otherwise. */
void Camera::computeProjectionMatrix() const
{
if (projectionMatrixIsUpToDate_) return;
const qreal ZNear = zNear();
const qreal ZFar = zFar();
switch (type())
{
case Camera::PERSPECTIVE:
{
// #CONNECTION# all non null coefficients were set to 0.0 in constructor.
const qreal f = 1.0/tan(fieldOfView()/2.0);
projectionMatrix_[0] = f/aspectRatio();
projectionMatrix_[5] = f;
projectionMatrix_[10] = (ZNear + ZFar) / (ZNear - ZFar);
projectionMatrix_[11] = -1.0;
projectionMatrix_[14] = 2.0 * ZNear * ZFar / (ZNear - ZFar);
projectionMatrix_[15] = 0.0;
// same as gluPerspective( 180.0*fieldOfView()/M_PI, aspectRatio(), zNear(), zFar() );
break;
}
case Camera::ORTHOGRAPHIC:
{
GLdouble w, h;
getOrthoWidthHeight(w,h);
projectionMatrix_[0] = 1.0/w;
projectionMatrix_[5] = 1.0/h;
projectionMatrix_[10] = -2.0/(ZFar - ZNear);
projectionMatrix_[11] = 0.0;
projectionMatrix_[14] = -(ZFar + ZNear)/(ZFar - ZNear);
projectionMatrix_[15] = 1.0;
// same as glOrtho( -w, w, -h, h, zNear(), zFar() );
break;
}
}
projectionMatrixIsUpToDate_ = true;
}
/*! Computes the modelView matrix associated with the Camera's position() and orientation().
This matrix converts from the world coordinates system to the Camera coordinates system, so that
coordinates can then be projected on screen using the projection matrix (see computeProjectionMatrix()).
Use getModelViewMatrix() to retrieve this matrix.
\note You must call this method if your Camera is not associated with a QGLViewer and is used for
offscreen computations (using (un)projectedCoordinatesOf() for instance). loadModelViewMatrix()
does it otherwise. */
void Camera::computeModelViewMatrix() const
{
if (modelViewMatrixIsUpToDate_) return;
const Quaternion q = frame()->orientation();
const qreal q00 = 2.0 * q[0] * q[0];
const qreal q11 = 2.0 * q[1] * q[1];
const qreal q22 = 2.0 * q[2] * q[2];
const qreal q01 = 2.0 * q[0] * q[1];
const qreal q02 = 2.0 * q[0] * q[2];
const qreal q03 = 2.0 * q[0] * q[3];
const qreal q12 = 2.0 * q[1] * q[2];
const qreal q13 = 2.0 * q[1] * q[3];
const qreal q23 = 2.0 * q[2] * q[3];
modelViewMatrix_[0] = 1.0 - q11 - q22;
modelViewMatrix_[1] = q01 - q23;
modelViewMatrix_[2] = q02 + q13;
modelViewMatrix_[3] = 0.0;
modelViewMatrix_[4] = q01 + q23;
modelViewMatrix_[5] = 1.0 - q22 - q00;
modelViewMatrix_[6] = q12 - q03;
modelViewMatrix_[7] = 0.0;
modelViewMatrix_[8] = q02 - q13;
modelViewMatrix_[9] = q12 + q03;
modelViewMatrix_[10] = 1.0 - q11 - q00;
modelViewMatrix_[11] = 0.0;
const Vec t = q.inverseRotate(frame()->position());
modelViewMatrix_[12] = -t.x;
modelViewMatrix_[13] = -t.y;
modelViewMatrix_[14] = -t.z;
modelViewMatrix_[15] = 1.0;
modelViewMatrixIsUpToDate_ = true;
}
/*! Loads the OpenGL \c GL_PROJECTION matrix with the Camera projection matrix.
The Camera projection matrix is computed using computeProjectionMatrix().
When \p reset is \c true (default), the method clears the previous projection matrix by calling \c
glLoadIdentity before setting the matrix. Setting \p reset to \c false is useful for \c GL_SELECT
mode, to combine the pushed matrix with a picking matrix. See QGLViewer::beginSelection() for details.
This method is used by QGLViewer::preDraw() (called before user's QGLViewer::draw() method) to
set the \c GL_PROJECTION matrix according to the viewer's QGLViewer::camera() settings.
Use getProjectionMatrix() to retrieve this matrix. Overload this method if you want your Camera to
use an exotic projection matrix. See also loadModelViewMatrix().
\attention \c glMatrixMode is set to \c GL_PROJECTION.
\attention If you use several OpenGL contexts and bypass the Qt main refresh loop, you should call
QGLWidget::makeCurrent() before this method in order to activate the right OpenGL context. */
void Camera::loadProjectionMatrix(bool reset) const
{
// WARNING: makeCurrent must be called by every calling method
glMatrixMode(GL_PROJECTION);
if (reset)
glLoadIdentity();
computeProjectionMatrix();
glMultMatrixd(projectionMatrix_);
}
/*! Loads the OpenGL \c GL_MODELVIEW matrix with the modelView matrix corresponding to the Camera.
Calls computeModelViewMatrix() to compute the Camera's modelView matrix.
This method is used by QGLViewer::preDraw() (called before user's QGLViewer::draw() method) to
set the \c GL_MODELVIEW matrix according to the viewer's QGLViewer::camera() position() and
orientation().
As a result, the vertices used in QGLViewer::draw() can be defined in the so called world
coordinate system. They are multiplied by this matrix to get converted to the Camera coordinate
system, before getting projected using the \c GL_PROJECTION matrix (see loadProjectionMatrix()).
When \p reset is \c true (default), the method loads (overwrites) the \c GL_MODELVIEW matrix. Setting
\p reset to \c false simply calls \c glMultMatrixd (might be useful for some applications).
Overload this method or simply call glLoadMatrixd() at the beginning of QGLViewer::draw() if you
want your Camera to use an exotic modelView matrix. See also loadProjectionMatrix().
getModelViewMatrix() returns the 4x4 modelView matrix.
\attention glMatrixMode is set to \c GL_MODELVIEW
\attention If you use several OpenGL contexts and bypass the Qt main refresh loop, you should call
QGLWidget::makeCurrent() before this method in order to activate the right OpenGL context. */
void Camera::loadModelViewMatrix(bool reset) const
{
// WARNING: makeCurrent must be called by every calling method
glMatrixMode(GL_MODELVIEW);
computeModelViewMatrix();
if (reset)
glLoadMatrixd(modelViewMatrix_);
else
glMultMatrixd(modelViewMatrix_);
}
/*! Same as loadProjectionMatrix() but for a stereo setup.
Only the Camera::PERSPECTIVE type() is supported for stereo mode. See
QGLViewer::setStereoDisplay().
Uses focusDistance(), IODistance(), and physicalScreenWidth() to compute cameras
offset and asymmetric frustums.
When \p leftBuffer is \c true, computes the projection matrix associated to the left eye (right eye
otherwise). See also loadModelViewMatrixStereo().
See the <a href="../examples/stereoViewer.html">stereoViewer</a> and the <a
href="../examples/contribs.html#anaglyph">anaglyph</a> examples for an illustration.
To retrieve this matrix, use a code like:
\code
glMatrixMode(GL_PROJECTION);
glPushMatrix();
loadProjectionMatrixStereo(left_or_right);
glGetDoublev(GL_PROJECTION_MATRIX, m);
glPopMatrix();
\endcode
Note that getProjectionMatrix() always returns the mono-vision matrix.
\attention glMatrixMode is set to \c GL_PROJECTION. */
void Camera::loadProjectionMatrixStereo(bool leftBuffer) const
{
qreal left, right, bottom, top;
qreal screenHalfWidth, halfWidth, side, shift, delta;
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
switch (type())
{
case Camera::PERSPECTIVE:
// compute half width of screen,
// corresponding to zero parallax plane to deduce decay of cameras
screenHalfWidth = focusDistance() * tan(horizontalFieldOfView() / 2.0);
shift = screenHalfWidth * IODistance() / physicalScreenWidth();
// should be * current y / y total
// to take into account that the window doesn't cover the entire screen
// compute half width of "view" at znear and the delta corresponding to
// the shifted camera to deduce what to set for asymmetric frustums
halfWidth = zNear() * tan(horizontalFieldOfView() / 2.0);
delta = shift * zNear() / focusDistance();
side = leftBuffer ? -1.0 : 1.0;
left = -halfWidth + side * delta;
right = halfWidth + side * delta;
top = halfWidth / aspectRatio();
bottom = -top;
glFrustum(left, right, bottom, top, zNear(), zFar() );
break;
case Camera::ORTHOGRAPHIC:
qWarning("Camera::setProjectionMatrixStereo: Stereo not available with Ortho mode");
break;
}
}
/*! Same as loadModelViewMatrix() but for a stereo setup.
Only the Camera::PERSPECTIVE type() is supported for stereo mode. See
QGLViewer::setStereoDisplay().
The modelView matrix is almost identical to the mono-vision one. It is simply translated along its
horizontal axis by a value that depends on stereo parameters (see focusDistance(),
IODistance(), and physicalScreenWidth()).
When \p leftBuffer is \c true, computes the modelView matrix associated to the left eye (right eye
otherwise).
loadProjectionMatrixStereo() explains how to retrieve to resulting matrix.
See the <a href="../examples/stereoViewer.html">stereoViewer</a> and the <a
href="../examples/contribs.html#anaglyph">anaglyph</a> examples for an illustration.
\attention glMatrixMode is set to \c GL_MODELVIEW. */
void Camera::loadModelViewMatrixStereo(bool leftBuffer) const
{
// WARNING: makeCurrent must be called by every calling method
glMatrixMode(GL_MODELVIEW);
qreal halfWidth = focusDistance() * tan(horizontalFieldOfView() / 2.0);
qreal shift = halfWidth * IODistance() / physicalScreenWidth(); // * current window width / full screen width
computeModelViewMatrix();
if (leftBuffer)
modelViewMatrix_[12] -= shift;
else
modelViewMatrix_[12] += shift;
glLoadMatrixd(modelViewMatrix_);
}
/*! Fills \p m with the Camera projection matrix values.
Based on computeProjectionMatrix() to make sure the Camera projection matrix is up to date.
This matrix only reflects the Camera's internal parameters and it may differ from the \c
GL_PROJECTION matrix retrieved using \c glGetDoublev(GL_PROJECTION_MATRIX, m). It actually
represents the state of the \c GL_PROJECTION after QGLViewer::preDraw(), at the beginning of
QGLViewer::draw(). If you modified the \c GL_PROJECTION matrix (for instance using
QGLViewer::startScreenCoordinatesSystem()), the two results differ.
The result is an OpenGL 4x4 matrix, which is given in \e column-major order (see \c glMultMatrix
man page for details).
See also getModelViewMatrix() and setFromProjectionMatrix(). */
void Camera::getProjectionMatrix(GLdouble m[16]) const
{
computeProjectionMatrix();
for (unsigned short i=0; i<16; ++i)
m[i] = projectionMatrix_[i];
}
/*! Overloaded getProjectionMatrix(GLdouble m[16]) method using a \c GLfloat array instead. */
void Camera::getProjectionMatrix(GLfloat m[16]) const
{
static GLdouble mat[16];
getProjectionMatrix(mat);
for (unsigned short i=0; i<16; ++i)
m[i] = float(mat[i]);
}
/*! Fills \p m with the Camera modelView matrix values.
First calls computeModelViewMatrix() to define the Camera modelView matrix.
Note that this matrix may \e not be the one you would get from a \c
glGetDoublev(GL_MODELVIEW_MATRIX, m). It actually represents the state of the \c
GL_MODELVIEW after QGLViewer::preDraw(), at the \e beginning of QGLViewer::draw(). It converts from
the world to the Camera coordinate system. As soon as you modify the \c GL_MODELVIEW in your
QGLViewer::draw() method (using glTranslate, glRotate... or similar methods), the two matrices differ.
The result is an OpenGL 4x4 matrix, which is given in \e column-major order (see \c glMultMatrix
man page for details).
See also getProjectionMatrix() and setFromModelViewMatrix(). */
void Camera::getModelViewMatrix(GLdouble m[16]) const
{
// May not be needed, but easier like this.
// Prevents from retrieving matrix in stereo mode -> overwrites shifted value.
computeModelViewMatrix();
for (unsigned short i=0; i<16; ++i)
m[i] = modelViewMatrix_[i];
}
/*! Overloaded getModelViewMatrix(GLdouble m[16]) method using a \c GLfloat array instead. */
void Camera::getModelViewMatrix(GLfloat m[16]) const
{
static GLdouble mat[16];
getModelViewMatrix(mat);
for (unsigned short i=0; i<16; ++i)
m[i] = float(mat[i]);
}
/*! Fills \p m with the product of the ModelView and Projection matrices.
Calls getModelViewMatrix() and getProjectionMatrix() and then fills \p m with the product of these two matrices. */
void Camera::getModelViewProjectionMatrix(GLdouble m[16]) const
{
GLdouble mv[16];
GLdouble proj[16];
getModelViewMatrix(mv);
getProjectionMatrix(proj);
for (unsigned short i=0; i<4; ++i)
{
for (unsigned short j=0; j<4; ++j)
{
qreal sum = 0.0;
for (unsigned short k=0; k<4; ++k)
sum += proj[i+4*k]*mv[k+4*j];
m[i+4*j] = sum;
}
}
}
/*! Overloaded getModelViewProjectionMatrix(GLdouble m[16]) method using a \c GLfloat array instead. */
void Camera::getModelViewProjectionMatrix(GLfloat m[16]) const
{
static GLdouble mat[16];
getModelViewProjectionMatrix(mat);
for (unsigned short i=0; i<16; ++i)
m[i] = float(mat[i]);
}
/*! Sets the sceneRadius() value. Negative values are ignored.
\attention This methods also sets focusDistance() to sceneRadius() / tan(fieldOfView()/2) and
flySpeed() to 1% of sceneRadius(). */
void Camera::setSceneRadius(qreal radius)
{
if (radius <= 0.0)
{
qWarning("Scene radius must be positive - Ignoring value");
return;
}
sceneRadius_ = radius;
projectionMatrixIsUpToDate_ = false;
setFocusDistance(sceneRadius() / tan(fieldOfView()/2.0));
frame()->setFlySpeed(0.01*sceneRadius());
}
/*! Similar to setSceneRadius() and setSceneCenter(), but the scene limits are defined by a (world
axis aligned) bounding box. */
void Camera::setSceneBoundingBox(const Vec& min, const Vec& max)
{
setSceneCenter((min+max)/2.0);
setSceneRadius(0.5*(max-min).norm());
}
/*! Sets the sceneCenter().
\attention This method also sets the pivotPoint() to sceneCenter(). */
void Camera::setSceneCenter(const Vec& center)
{
sceneCenter_ = center;
setPivotPoint(sceneCenter());
projectionMatrixIsUpToDate_ = false;
}
/*! setSceneCenter() to the result of pointUnderPixel(\p pixel).
Returns \c true if a pointUnderPixel() was found and sceneCenter() was actually changed.
See also setPivotPointFromPixel(). See the pointUnderPixel() documentation. */
bool Camera::setSceneCenterFromPixel(const QPoint& pixel)
{
bool found;
Vec point = pointUnderPixel(pixel, found);
if (found)
setSceneCenter(point);
return found;
}
#ifndef DOXYGEN
void Camera::setRevolveAroundPoint(const Vec& point) {
qWarning("setRevolveAroundPoint() is deprecated, use setPivotPoint() instead");
setPivotPoint(point);
}
bool Camera::setRevolveAroundPointFromPixel(const QPoint& pixel) {
qWarning("setRevolveAroundPointFromPixel() is deprecated, use setPivotPointFromPixel() instead");
return setPivotPointFromPixel(pixel);
}
Vec Camera::revolveAroundPoint() const {
qWarning("revolveAroundPoint() is deprecated, use pivotPoint() instead");
return pivotPoint();
}
#endif
/*! Changes the pivotPoint() to \p point (defined in the world coordinate system). */
void Camera::setPivotPoint(const Vec& point)
{
const qreal prevDist = fabs(cameraCoordinatesOf(pivotPoint()).z);
// If frame's RAP is set directly, projectionMatrixIsUpToDate_ should also be
// set to false to ensure proper recomputation of the ORTHO projection matrix.
frame()->setPivotPoint(point);
// orthoCoef_ is used to compensate for changes of the pivotPoint, so that the image does
// not change when the pivotPoint is changed in ORTHOGRAPHIC mode.
const qreal newDist = fabs(cameraCoordinatesOf(pivotPoint()).z);
// Prevents division by zero when rap is set to camera position
if ((prevDist > 1E-9) && (newDist > 1E-9))
orthoCoef_ *= prevDist / newDist;
projectionMatrixIsUpToDate_ = false;
}
/*! The pivotPoint() is set to the point located under \p pixel on screen.
Returns \c true if a pointUnderPixel() was found. If no point was found under \p pixel, the
pivotPoint() is left unchanged.
\p pixel is expressed in Qt format (origin in the upper left corner of the window). See
pointUnderPixel().
See also setSceneCenterFromPixel(). */
bool Camera::setPivotPointFromPixel(const QPoint& pixel)
{
bool found;
Vec point = pointUnderPixel(pixel, found);
if (found)
setPivotPoint(point);
return found;
}
/*! Returns the ratio between pixel and OpenGL units at \p position.
A line of \c n * pixelGLRatio() OpenGL units, located at \p position in the world coordinates
system, will be projected with a length of \c n pixels on screen.
Use this method to scale objects so that they have a constant pixel size on screen. The following
code will draw a 20 pixel line, starting at sceneCenter() and always directed along the screen
vertical direction:
\code
glBegin(GL_LINES);
glVertex3fv(sceneCenter());
glVertex3fv(sceneCenter() + 20 * pixelGLRatio(sceneCenter()) * camera()->upVector());
glEnd();
\endcode */
qreal Camera::pixelGLRatio(const Vec& position) const
{
switch (type())
{
case Camera::PERSPECTIVE :
return 2.0 * fabs((frame()->coordinatesOf(position)).z) * tan(fieldOfView()/2.0) / screenHeight();
case Camera::ORTHOGRAPHIC :
{
GLdouble w, h;
getOrthoWidthHeight(w,h);
return 2.0 * h / screenHeight();
}
}
// Bad compilers complain
return 1.0;
}
/*! Changes the Camera fieldOfView() so that the entire scene (defined by QGLViewer::sceneCenter()
and QGLViewer::sceneRadius()) is visible from the Camera position().
The position() and orientation() of the Camera are not modified and you first have to orientate the
Camera in order to actually see the scene (see lookAt(), showEntireScene() or fitSphere()).
This method is especially useful for \e shadow \e maps computation. Use the Camera positioning
tools (setPosition(), lookAt()) to position a Camera at the light position. Then use this method to
define the fieldOfView() so that the shadow map resolution is optimally used:
\code
// The light camera needs size hints in order to optimize its fieldOfView
lightCamera->setSceneRadius(sceneRadius());
lightCamera->setSceneCenter(sceneCenter());
// Place the light camera.
lightCamera->setPosition(lightFrame->position());
lightCamera->lookAt(sceneCenter());
lightCamera->setFOVToFitScene();
\endcode
See the (soon available) shadowMap contribution example for a practical implementation.
\attention The fieldOfView() is clamped to M_PI/2.0. This happens when the Camera is at a distance
lower than sqrt(2.0) * sceneRadius() from the sceneCenter(). It optimizes the shadow map
resolution, although it may miss some parts of the scene. */
void Camera::setFOVToFitScene()
{
if (distanceToSceneCenter() > sqrt(2.0)*sceneRadius())
setFieldOfView(2.0 * asin(sceneRadius() / distanceToSceneCenter()));
else
setFieldOfView(M_PI / 2.0);
}
/*! Makes the Camera smoothly zoom on the pointUnderPixel() \p pixel.
Nothing happens if no pointUnderPixel() is found. Otherwise a KeyFrameInterpolator is created that
animates the Camera on a one second path that brings the Camera closer to the point under \p pixel.
See also interpolateToFitScene(). */
void Camera::interpolateToZoomOnPixel(const QPoint& pixel)
{
const qreal coef = 0.1;
bool found;
Vec target = pointUnderPixel(pixel, found);
if (!found)
return;
if (interpolationKfi_->interpolationIsStarted())
interpolationKfi_->stopInterpolation();
interpolationKfi_->deletePath();
interpolationKfi_->addKeyFrame(*(frame()));
interpolationKfi_->addKeyFrame(Frame(0.3*frame()->position() + 0.7*target, frame()->orientation()), 0.4);
// Small hack: attach a temporary frame to take advantage of lookAt without modifying frame
static ManipulatedCameraFrame* tempFrame = new ManipulatedCameraFrame();
ManipulatedCameraFrame* const originalFrame = frame();
tempFrame->setPosition(coef*frame()->position() + (1.0-coef)*target);
tempFrame->setOrientation(frame()->orientation());
setFrame(tempFrame);
lookAt(target);
setFrame(originalFrame);
interpolationKfi_->addKeyFrame(*(tempFrame), 1.0);
interpolationKfi_->startInterpolation();
}
/*! Interpolates the Camera on a one second KeyFrameInterpolator path so that the entire scene fits
the screen at the end.
The scene is defined by its sceneCenter() and its sceneRadius(). See showEntireScene().
The orientation() of the Camera is not modified. See also interpolateToZoomOnPixel(). */
void Camera::interpolateToFitScene()
{
if (interpolationKfi_->interpolationIsStarted())
interpolationKfi_->stopInterpolation();
interpolationKfi_->deletePath();
interpolationKfi_->addKeyFrame(*(frame()));
// Small hack: attach a temporary frame to take advantage of lookAt without modifying frame
static ManipulatedCameraFrame* tempFrame = new ManipulatedCameraFrame();
ManipulatedCameraFrame* const originalFrame = frame();
tempFrame->setPosition(frame()->position());
tempFrame->setOrientation(frame()->orientation());
setFrame(tempFrame);
showEntireScene();
setFrame(originalFrame);
interpolationKfi_->addKeyFrame(*(tempFrame));
interpolationKfi_->startInterpolation();
}
/*! Smoothly interpolates the Camera on a KeyFrameInterpolator path so that it goes to \p fr.
\p fr is expressed in world coordinates. \p duration tunes the interpolation speed (default is
1 second).
See also interpolateToFitScene() and interpolateToZoomOnPixel(). */
void Camera::interpolateTo(const Frame& fr, qreal duration)
{
if (interpolationKfi_->interpolationIsStarted())
interpolationKfi_->stopInterpolation();
interpolationKfi_->deletePath();
interpolationKfi_->addKeyFrame(*(frame()));
interpolationKfi_->addKeyFrame(fr, duration);
interpolationKfi_->startInterpolation();
}
/*! Returns the coordinates of the 3D point located at pixel (x,y) on screen.
Calls a \c glReadPixel to get the pixel depth and applies an unprojectedCoordinatesOf() to the
result. \p found indicates whether a point was found or not (i.e. background pixel, result's depth
is zFar() in that case).
\p x and \p y are expressed in pixel units with an origin in the upper left corner. Use
screenHeight() - y to convert to OpenGL standard.
\attention This method assumes that a GL context is available, and that its content was drawn using
the Camera (i.e. using its projection and modelview matrices). This method hence cannot be used for
offscreen Camera computations. Use cameraCoordinatesOf() and worldCoordinatesOf() to perform
similar operations in that case.
\note The precision of the z-Buffer highly depends on how the zNear() and zFar() values are fitted
to your scene. Loose boundaries will result in imprecision along the viewing direction. */
Vec Camera::pointUnderPixel(const QPoint& pixel, bool& found) const
{
float depth;
// Qt uses upper corner for its origin while GL uses the lower corner.
glReadPixels(pixel.x(), screenHeight()-1-pixel.y(), 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT, &depth);
found = depth < 1.0;
Vec point(pixel.x(), pixel.y(), depth);
point = unprojectedCoordinatesOf(point);
return point;
}
/*! Moves the Camera so that the entire scene is visible.
Simply calls fitSphere() on a sphere defined by sceneCenter() and sceneRadius().
You will typically use this method in QGLViewer::init() after you defined a new sceneRadius(). */
void Camera::showEntireScene()
{
fitSphere(sceneCenter(), sceneRadius());
}
/*! Moves the Camera so that its sceneCenter() is projected on the center of the window. The
orientation() and fieldOfView() are unchanged.
Simply projects the current position on a line passing through sceneCenter(). See also
showEntireScene().*/
void Camera::centerScene()
{
frame()->projectOnLine(sceneCenter(), viewDirection());
}
/*! Sets the Camera orientation(), so that it looks at point \p target (defined in the world
coordinate system).
The Camera position() is not modified. Simply setViewDirection().
See also setUpVector(), setOrientation(), showEntireScene(), fitSphere() and fitBoundingBox(). */
void Camera::lookAt(const Vec& target)
{
setViewDirection(target - position());
}
/*! Moves the Camera so that the sphere defined by (\p center, \p radius) is visible and fits in the frustum.
The Camera is simply translated to center the sphere in the screen and make it fit the frustum. Its
orientation() and its fieldOfView() are unchanged.
You should therefore orientate the Camera before you call this method. See lookAt(),
setOrientation() and setUpVector(). */
void Camera::fitSphere(const Vec& center, qreal radius)
{
qreal distance = 0.0;
switch (type())
{
case Camera::PERSPECTIVE :
{
const qreal yview = radius / sin(fieldOfView() / 2.0);
const qreal xview = radius / sin(horizontalFieldOfView() / 2.0);
distance = qMax(xview, yview);
break;
}
case Camera::ORTHOGRAPHIC :
{
distance = ((center-pivotPoint()) * viewDirection()) + (radius / orthoCoef_);
break;
}
}
Vec newPos(center - distance * viewDirection());
frame()->setPositionWithConstraint(newPos);
}
/*! Moves the Camera so that the (world axis aligned) bounding box (\p min, \p max) is entirely
visible, using fitSphere(). */
void Camera::fitBoundingBox(const Vec& min, const Vec& max)
{
qreal diameter = qMax(fabs(max[1]-min[1]), fabs(max[0]-min[0]));
diameter = qMax(fabs(max[2]-min[2]), diameter);
fitSphere(0.5*(min+max), 0.5*diameter);
}
/*! Moves the Camera so that the rectangular screen region defined by \p rectangle (pixel units,
with origin in the upper left corner) fits the screen.
The Camera is translated (its orientation() is unchanged) so that \p rectangle is entirely
visible. Since the pixel coordinates only define a \e frustum in 3D, it's the intersection of this
frustum with a plane (orthogonal to the viewDirection() and passing through the sceneCenter())
that is used to define the 3D rectangle that is eventually fitted. */
void Camera::fitScreenRegion(const QRect& rectangle)
{
const Vec vd = viewDirection();
const qreal distToPlane = distanceToSceneCenter();
const QPoint center = rectangle.center();
Vec orig, dir;
convertClickToLine( center, orig, dir );
Vec newCenter = orig + distToPlane / (dir*vd) * dir;
convertClickToLine( QPoint(rectangle.x(), center.y()), orig, dir );
const Vec pointX = orig + distToPlane / (dir*vd) * dir;
convertClickToLine( QPoint(center.x(), rectangle.y()), orig, dir );
const Vec pointY = orig + distToPlane / (dir*vd) * dir;
qreal distance = 0.0;
switch (type())
{
case Camera::PERSPECTIVE :
{
const qreal distX = (pointX-newCenter).norm() / sin(horizontalFieldOfView()/2.0);
const qreal distY = (pointY-newCenter).norm() / sin(fieldOfView()/2.0);
distance = qMax(distX, distY);
break;
}
case Camera::ORTHOGRAPHIC :
{
const qreal dist = ((newCenter-pivotPoint()) * vd);
//#CONNECTION# getOrthoWidthHeight
const qreal distX = (pointX-newCenter).norm() / orthoCoef_ / ((aspectRatio() < 1.0) ? 1.0 : aspectRatio());
const qreal distY = (pointY-newCenter).norm() / orthoCoef_ / ((aspectRatio() < 1.0) ? 1.0/aspectRatio() : 1.0);
distance = dist + qMax(distX, distY);
break;
}
}
Vec newPos(newCenter - distance * vd);
frame()->setPositionWithConstraint(newPos);
}
/*! Rotates the Camera so that its upVector() becomes \p up (defined in the world coordinate
system).
The Camera is rotated around an axis orthogonal to \p up and to the current upVector() direction.
Use this method in order to define the Camera horizontal plane.
When \p noMove is set to \c false, the orientation modification is compensated by a translation, so
that the pivotPoint() stays projected at the same position on screen. This is especially
useful when the Camera is used as an observer of the scene (default mouse binding).
When \p noMove is \c true (default), the Camera position() is left unchanged, which is an intuitive
behavior when the Camera is in a walkthrough fly mode (see the QGLViewer::MOVE_FORWARD and
QGLViewer::MOVE_BACKWARD QGLViewer::MouseAction).
The frame()'s ManipulatedCameraFrame::sceneUpVector() is set accordingly.
See also setViewDirection(), lookAt() and setOrientation(). */
void Camera::setUpVector(const Vec& up, bool noMove)
{
Quaternion q(Vec(0.0, 1.0, 0.0), frame()->transformOf(up));
if (!noMove)
frame()->setPosition(pivotPoint() - (frame()->orientation()*q).rotate(frame()->coordinatesOf(pivotPoint())));
frame()->rotate(q);
// Useful in fly mode to keep the horizontal direction.
frame()->updateSceneUpVector();
}
/*! Sets the orientation() of the Camera using polar coordinates.
\p theta rotates the Camera around its Y axis, and \e then \p phi rotates it around its X axis.
The polar coordinates are defined in the world coordinates system: \p theta = \p phi = 0 means
that the Camera is directed towards the world Z axis. Both angles are expressed in radians.
See also setUpVector(). The position() of the Camera is unchanged, you may want to call showEntireScene()
after this method to move the Camera.
This method can be useful to create Quicktime VR panoramic sequences, see the
QGLViewer::saveSnapshot() documentation for details. */
void Camera::setOrientation(qreal theta, qreal phi)
{
Vec axis(0.0, 1.0, 0.0);
const Quaternion rot1(axis, theta);
axis = Vec(-cos(theta), 0.0, sin(theta));
const Quaternion rot2(axis, phi);
setOrientation(rot1 * rot2);
}
/*! Sets the Camera orientation(), defined in the world coordinate system. */
void Camera::setOrientation(const Quaternion& q)
{
frame()->setOrientation(q);
frame()->updateSceneUpVector();
}
/*! Rotates the Camera so that its viewDirection() is \p direction (defined in the world coordinate
system).
The Camera position() is not modified. The Camera is rotated so that the horizon (defined by its
upVector()) is preserved. See also lookAt() and setUpVector(). */
void Camera::setViewDirection(const Vec& direction)
{
if (direction.squaredNorm() < 1E-10)
return;
Vec xAxis = direction ^ upVector();
if (xAxis.squaredNorm() < 1E-10)
{
// target is aligned with upVector, this means a rotation around X axis
// X axis is then unchanged, let's keep it !
xAxis = frame()->inverseTransformOf(Vec(1.0, 0.0, 0.0));
}
Quaternion q;
q.setFromRotatedBasis(xAxis, xAxis^direction, -direction);
frame()->setOrientationWithConstraint(q);
}
// Compute a 3 by 3 determinant.
static qreal det(qreal m00,qreal m01,qreal m02,
qreal m10,qreal m11,qreal m12,
qreal m20,qreal m21,qreal m22)
{
return m00*m11*m22 + m01*m12*m20 + m02*m10*m21 - m20*m11*m02 - m10*m01*m22 - m00*m21*m12;
}
// Computes the index of element [i][j] in a \c qreal matrix[3][4].
static inline unsigned int ind(unsigned int i, unsigned int j)
{
return (i*4+j);
}
/*! Returns the Camera position (the eye), defined in the world coordinate system.
Use setPosition() to set the Camera position. Other convenient methods are showEntireScene() or
fitSphere(). Actually returns \c frame()->position().
This position corresponds to the projection center of a Camera::PERSPECTIVE Camera. It is not
located in the image plane, which is at a zNear() distance ahead. */
Vec Camera::position() const { return frame()->position(); }
/*! Returns the normalized up vector of the Camera, defined in the world coordinate system.
Set using setUpVector() or setOrientation(). It is orthogonal to viewDirection() and to
rightVector().
It corresponds to the Y axis of the associated frame() (actually returns
frame()->inverseTransformOf(Vec(0.0, 1.0, 0.0)) ). */
Vec Camera::upVector() const
{
return frame()->inverseTransformOf(Vec(0.0, 1.0, 0.0));
}
/*! Returns the normalized view direction of the Camera, defined in the world coordinate system.
Change this value using setViewDirection(), lookAt() or setOrientation(). It is orthogonal to
upVector() and to rightVector().
This corresponds to the negative Z axis of the frame() ( frame()->inverseTransformOf(Vec(0.0,
0.0, -1.0)) ). */
Vec Camera::viewDirection() const { return frame()->inverseTransformOf(Vec(0.0, 0.0, -1.0)); }
/*! Returns the normalized right vector of the Camera, defined in the world coordinate system.
This vector lies in the Camera horizontal plane, directed along the X axis (orthogonal to
upVector() and to viewDirection()). Set using setUpVector(), lookAt() or setOrientation().
Simply returns frame()->inverseTransformOf(Vec(1.0, 0.0, 0.0)). */
Vec Camera::rightVector() const
{
return frame()->inverseTransformOf(Vec(1.0, 0.0, 0.0));
}
/*! Returns the Camera orientation, defined in the world coordinate system.
Actually returns \c frame()->orientation(). Use setOrientation(), setUpVector() or lookAt() to
set the Camera orientation. */
Quaternion Camera::orientation() const { return frame()->orientation(); }
/*! Sets the Camera position() (the eye), defined in the world coordinate system. */
void Camera::setPosition(const Vec& pos) { frame()->setPosition(pos); }
/*! Returns the Camera frame coordinates of a point \p src defined in world coordinates.
worldCoordinatesOf() performs the inverse transformation.
Note that the point coordinates are simply converted in a different coordinate system. They are
not projected on screen. Use projectedCoordinatesOf() for that. */
Vec Camera::cameraCoordinatesOf(const Vec& src) const { return frame()->coordinatesOf(src); }
/*! Returns the world coordinates of the point whose position \p src is defined in the Camera
coordinate system.
cameraCoordinatesOf() performs the inverse transformation. */
Vec Camera::worldCoordinatesOf(const Vec& src) const { return frame()->inverseCoordinatesOf(src); }
/*! Returns the fly speed of the Camera.
Simply returns frame()->flySpeed(). See the ManipulatedCameraFrame::flySpeed() documentation.
This value is only meaningful when the MouseAction bindings is QGLViewer::MOVE_FORWARD or
QGLViewer::MOVE_BACKWARD.
Set to 1% of the sceneRadius() by setSceneRadius(). See also setFlySpeed(). */
qreal Camera::flySpeed() const { return frame()->flySpeed(); }
/*! Sets the Camera flySpeed().
\attention This value is modified by setSceneRadius(). */
void Camera::setFlySpeed(qreal speed) { frame()->setFlySpeed(speed); }
/*! The point the Camera pivots around with the QGLViewer::ROTATE mouse binding. Defined in world coordinate system.
Default value is the sceneCenter().
\attention setSceneCenter() changes this value. */
Vec Camera::pivotPoint() const { return frame()->pivotPoint(); }
/*! Sets the Camera's position() and orientation() from an OpenGL ModelView matrix.
This enables a Camera initialisation from an other OpenGL application. \p modelView is a 16 GLdouble
vector representing a valid OpenGL ModelView matrix, such as one can get using:
\code
GLdouble mvm[16];
glGetDoublev(GL_MODELVIEW_MATRIX, mvm);
myCamera->setFromModelViewMatrix(mvm);
\endcode
After this method has been called, getModelViewMatrix() returns a matrix equivalent to \p
modelView.
Only the orientation() and position() of the Camera are modified.
\note If you defined your matrix as \c GLdouble \c mvm[4][4], pass \c &(mvm[0][0]) as a
parameter. */
void Camera::setFromModelViewMatrix(const GLdouble* const modelViewMatrix)
{
// Get upper left (rotation) matrix
qreal upperLeft[3][3];
for (int i=0; i<3; ++i)
for (int j=0; j<3; ++j)
upperLeft[i][j] = modelViewMatrix[i*4+j];
// Transform upperLeft into the associated Quaternion
Quaternion q;
q.setFromRotationMatrix(upperLeft);
setOrientation(q);
setPosition(-q.rotate(Vec(modelViewMatrix[12], modelViewMatrix[13], modelViewMatrix[14])));
}
/*! Defines the Camera position(), orientation() and fieldOfView() from a projection matrix.
\p matrix has to be given in the format used by vision algorithm. It has 3 lines and 4 columns. It
transforms a point from the world homogeneous coordinate system (4 coordinates: \c sx, \c sy, \c sz
and \c s) into a point in the screen homogeneous coordinate system (3 coordinates: \c sx, \c sy,
and \c s, where \c x and \c y are the pixel coordinates on the screen).
Its three lines correspond to the homogeneous coordinates of the normals to the planes x=0, y=0 and
z=0, defined in the Camera coordinate system.
The elements of the matrix are ordered in line major order: you can call \c
setFromProjectionMatrix(&(matrix[0][0])) if you defined your matrix as a \c qreal \c matrix[3][4].
\attention Passing the result of getProjectionMatrix() or getModelViewMatrix() to this method is
not possible (purposefully incompatible matrix dimensions). \p matrix is more likely to be the
product of these two matrices, without the last line.
Use setFromModelViewMatrix() to set position() and orientation() from a \c GL_MODELVIEW matrix.
fieldOfView() can also be retrieved from a \e perspective \c GL_PROJECTION matrix using 2.0 *
atan(1.0/projectionMatrix[5]).
This code was written by Sylvain Paris. */
void Camera::setFromProjectionMatrix(const qreal matrix[12])
{
// The 3 lines of the matrix are the normals to the planes x=0, y=0, z=0
// in the camera CS. As we normalize them, we do not need the 4th coordinate.
Vec line_0(matrix[ind(0,0)],matrix[ind(0,1)],matrix[ind(0,2)]);
Vec line_1(matrix[ind(1,0)],matrix[ind(1,1)],matrix[ind(1,2)]);
Vec line_2(matrix[ind(2,0)],matrix[ind(2,1)],matrix[ind(2,2)]);
line_0.normalize();
line_1.normalize();
line_2.normalize();
// The camera position is at (0,0,0) in the camera CS so it is the
// intersection of the 3 planes. It can be seen as the kernel
// of the 3x4 projection matrix. We calculate it through 4 dimensional
// vectorial product. We go directly into 3D that is to say we directly
// divide the first 3 coordinates by the 4th one.
// We derive the 4 dimensional vectorial product formula from the
// computation of a 4x4 determinant that is developped according to
// its 4th column. This implies some 3x3 determinants.
const Vec cam_pos = Vec(det(matrix[ind(0,1)],matrix[ind(0,2)],matrix[ind(0,3)],
matrix[ind(1,1)],matrix[ind(1,2)],matrix[ind(1,3)],
matrix[ind(2,1)],matrix[ind(2,2)],matrix[ind(2,3)]),
-det(matrix[ind(0,0)],matrix[ind(0,2)],matrix[ind(0,3)],
matrix[ind(1,0)],matrix[ind(1,2)],matrix[ind(1,3)],
matrix[ind(2,0)],matrix[ind(2,2)],matrix[ind(2,3)]),
det(matrix[ind(0,0)],matrix[ind(0,1)],matrix[ind(0,3)],
matrix[ind(1,0)],matrix[ind(1,1)],matrix[ind(1,3)],
matrix[ind(2,0)],matrix[ind(2,1)],matrix[ind(2,3)])) /
(-det(matrix[ind(0,0)],matrix[ind(0,1)],matrix[ind(0,2)],
matrix[ind(1,0)],matrix[ind(1,1)],matrix[ind(1,2)],
matrix[ind(2,0)],matrix[ind(2,1)],matrix[ind(2,2)]));
// We compute the rotation matrix column by column.
// GL Z axis is front facing.
Vec column_2 = -line_2;
// X-axis is almost like line_0 but should be orthogonal to the Z axis.
Vec column_0 = ((column_2^line_0)^column_2);
column_0.normalize();
// Y-axis is almost like line_1 but should be orthogonal to the Z axis.
// Moreover line_1 is downward oriented as the screen CS.
Vec column_1 = -((column_2^line_1)^column_2);
column_1.normalize();
qreal rot[3][3];
rot[0][0] = column_0[0];
rot[1][0] = column_0[1];
rot[2][0] = column_0[2];
rot[0][1] = column_1[0];
rot[1][1] = column_1[1];
rot[2][1] = column_1[2];
rot[0][2] = column_2[0];
rot[1][2] = column_2[1];
rot[2][2] = column_2[2];
// We compute the field of view
// line_1^column_0 -> vector of intersection line between
// y_screen=0 and x_camera=0 plane.
// column_2*(...) -> cos of the angle between Z vector et y_screen=0 plane
// * 2 -> field of view = 2 * half angle
// We need some intermediate values.
Vec dummy = line_1^column_0;
dummy.normalize();
qreal fov = acos(column_2*dummy) * 2.0;
// We set the camera.
Quaternion q;
q.setFromRotationMatrix(rot);
setOrientation(q);
setPosition(cam_pos);
setFieldOfView(fov);
}
/*
// persp : projectionMatrix_[0] = f/aspectRatio();
void Camera::setFromProjectionMatrix(const GLdouble* projectionMatrix)
{
QString message;
if ((fabs(projectionMatrix[1]) > 1E-3) ||
(fabs(projectionMatrix[2]) > 1E-3) ||
(fabs(projectionMatrix[3]) > 1E-3) ||
(fabs(projectionMatrix[4]) > 1E-3) ||
(fabs(projectionMatrix[6]) > 1E-3) ||
(fabs(projectionMatrix[7]) > 1E-3) ||
(fabs(projectionMatrix[8]) > 1E-3) ||
(fabs(projectionMatrix[9]) > 1E-3))
message = "Non null coefficient in projection matrix - Aborting";
else
if ((fabs(projectionMatrix[11]+1.0) < 1E-5) && (fabs(projectionMatrix[15]) < 1E-5))
{
if (projectionMatrix[5] < 1E-4)
message="Negative field of view in Camera::setFromProjectionMatrix";
else
setType(Camera::PERSPECTIVE);
}
else
if ((fabs(projectionMatrix[11]) < 1E-5) && (fabs(projectionMatrix[15]-1.0) < 1E-5))
setType(Camera::ORTHOGRAPHIC);
else
message = "Unable to determine camera type in setFromProjectionMatrix - Aborting";
if (!message.isEmpty())
{
qWarning(message);
return;
}
switch (type())
{
case Camera::PERSPECTIVE:
{
setFieldOfView(2.0 * atan(1.0/projectionMatrix[5]));
const qreal far = projectionMatrix[14] / (2.0 * (1.0 + projectionMatrix[10]));
const qreal near = (projectionMatrix[10]+1.0) / (projectionMatrix[10]-1.0) * far;
setSceneRadius((far-near)/2.0);
setSceneCenter(position() + (near + sceneRadius())*viewDirection());
break;
}
case Camera::ORTHOGRAPHIC:
{
GLdouble w, h;
getOrthoWidthHeight(w,h);
projectionMatrix_[0] = 1.0/w;
projectionMatrix_[5] = 1.0/h;
projectionMatrix_[10] = -2.0/(ZFar - ZNear);
projectionMatrix_[11] = 0.0;
projectionMatrix_[14] = -(ZFar + ZNear)/(ZFar - ZNear);
projectionMatrix_[15] = 1.0;
// same as glOrtho( -w, w, -h, h, zNear(), zFar() );
break;
}
}
}
*/
///////////////////////// Camera to world transform ///////////////////////
/*! Same as cameraCoordinatesOf(), but with \c qreal[3] parameters (\p src and \p res may be identical pointers). */
void Camera::getCameraCoordinatesOf(const qreal src[3], qreal res[3]) const
{
Vec r = cameraCoordinatesOf(Vec(src));
for (int i=0; i<3; ++i)
res[i] = r[i];
}
/*! Same as worldCoordinatesOf(), but with \c qreal[3] parameters (\p src and \p res may be identical pointers). */
void Camera::getWorldCoordinatesOf(const qreal src[3], qreal res[3]) const
{
Vec r = worldCoordinatesOf(Vec(src));
for (int i=0; i<3; ++i)
res[i] = r[i];
}
/*! Fills \p viewport with the Camera OpenGL viewport.
This method is mainly used in conjunction with \c gluProject, which requires such a viewport.
Returned values are (0, screenHeight(), screenWidth(), - screenHeight()), so that the origin is
located in the \e upper left corner of the window (Qt style coordinate system). */
void Camera::getViewport(GLint viewport[4]) const
{
viewport[0] = 0;
viewport[1] = screenHeight();
viewport[2] = screenWidth();
viewport[3] = -screenHeight();
}
/*! Returns the screen projected coordinates of a point \p src defined in the \p frame coordinate
system.
When \p frame in \c NULL (default), \p src is expressed in the world coordinate system.
The x and y coordinates of the returned Vec are expressed in pixel, (0,0) being the \e upper left
corner of the window. The z coordinate ranges between 0.0 (near plane) and 1.0 (excluded, far
plane). See the \c gluProject man page for details.
unprojectedCoordinatesOf() performs the inverse transformation.
See the <a href="../examples/screenCoordSystem.html">screenCoordSystem example</a>.
This method only uses the intrinsic Camera parameters (see getModelViewMatrix(),
getProjectionMatrix() and getViewport()) and is completely independent of the OpenGL \c
GL_MODELVIEW, \c GL_PROJECTION and viewport matrices. You can hence define a virtual Camera and use
this method to compute projections out of a classical rendering context.
\attention However, if your Camera is not attached to a QGLViewer (used for offscreen computations
for instance), make sure the Camera matrices are updated before calling this method. Call
computeModelViewMatrix() and computeProjectionMatrix() to do so.
If you call this method several times with no change in the matrices, consider precomputing the
projection times modelview matrix to save computation time if required (\c P x \c M in the \c
gluProject man page).
Here is the code corresponding to what this method does (kindly submitted by Robert W. Kuhn) :
\code
Vec project(Vec point)
{
GLint Viewport[4];
GLdouble Projection[16], Modelview[16];
GLdouble matrix[16];
// Precomputation begin
glGetIntegerv(GL_VIEWPORT , Viewport);
glGetDoublev (GL_MODELVIEW_MATRIX , Modelview);
glGetDoublev (GL_PROJECTION_MATRIX, Projection);
for (unsigned short m=0; m<4; ++m)
{
for (unsigned short l=0; l<4; ++l)
{
qreal sum = 0.0;
for (unsigned short k=0; k<4; ++k)
sum += Projection[l+4*k]*Modelview[k+4*m];
matrix[l+4*m] = sum;
}
}
// Precomputation end
GLdouble v[4], vs[4];
v[0]=point[0]; v[1]=point[1]; v[2]=point[2]; v[3]=1.0;
vs[0]=matrix[0 ]*v[0] + matrix[4 ]*v[1] + matrix[8 ]*v[2] + matrix[12 ]*v[3];
vs[1]=matrix[1 ]*v[0] + matrix[5 ]*v[1] + matrix[9 ]*v[2] + matrix[13 ]*v[3];
vs[2]=matrix[2 ]*v[0] + matrix[6 ]*v[1] + matrix[10]*v[2] + matrix[14 ]*v[3];
vs[3]=matrix[3 ]*v[0] + matrix[7 ]*v[1] + matrix[11]*v[2] + matrix[15 ]*v[3];
vs[0] /= vs[3];
vs[1] /= vs[3];
vs[2] /= vs[3];
vs[0] = vs[0] * 0.5 + 0.5;
vs[1] = vs[1] * 0.5 + 0.5;
vs[2] = vs[2] * 0.5 + 0.5;
vs[0] = vs[0] * Viewport[2] + Viewport[0];
vs[1] = vs[1] * Viewport[3] + Viewport[1];
return Vec(vs[0], Viewport[3]-vs[1], vs[2]);
}
\endcode
*/
Vec Camera::projectedCoordinatesOf(const Vec& src, const Frame* frame) const
{
GLdouble x,y,z;
static GLint viewport[4];
getViewport(viewport);
if (frame)
{
const Vec tmp = frame->inverseCoordinatesOf(src);
gluProject(tmp.x,tmp.y,tmp.z, modelViewMatrix_, projectionMatrix_, viewport, &x,&y,&z);
}
else
gluProject(src.x,src.y,src.z, modelViewMatrix_, projectionMatrix_, viewport, &x,&y,&z);
return Vec(x,y,z);
}
/*! Returns the world unprojected coordinates of a point \p src defined in the screen coordinate
system.
The \p src.x and \p src.y input values are expressed in pixels, (0,0) being the \e upper left corner
of the window. \p src.z is a depth value ranging in [0..1[ (respectively corresponding to the near
and far planes). Note that src.z is \e not a linear interpolation between zNear and zFar.
/code
src.z = zFar() / (zFar() - zNear()) * (1.0 - zNear() / z);
/endcode
Where z is the distance from the point you project to the camera, along the viewDirection().
See the \c gluUnProject man page for details.
The result is expressed in the \p frame coordinate system. When \p frame is \c NULL (default), the
result is expressed in the world coordinates system. The possible \p frame Frame::referenceFrame()
are taken into account.
projectedCoordinatesOf() performs the inverse transformation.
This method only uses the intrinsic Camera parameters (see getModelViewMatrix(),
getProjectionMatrix() and getViewport()) and is completely independent of the OpenGL \c
GL_MODELVIEW, \c GL_PROJECTION and viewport matrices. You can hence define a virtual Camera and use
this method to compute un-projections out of a classical rendering context.
\attention However, if your Camera is not attached to a QGLViewer (used for offscreen computations
for instance), make sure the Camera matrices are updated before calling this method (use
computeModelViewMatrix(), computeProjectionMatrix()). See also setScreenWidthAndHeight().
This method is not computationally optimized. If you call it several times with no change in the
matrices, you should buffer the entire inverse projection matrix (modelview, projection and then
viewport) to speed-up the queries. See the \c gluUnProject man page for details. */
Vec Camera::unprojectedCoordinatesOf(const Vec& src, const Frame* frame) const
{
GLdouble x,y,z;
static GLint viewport[4];
getViewport(viewport);
gluUnProject(src.x,src.y,src.z, modelViewMatrix_, projectionMatrix_, viewport, &x,&y,&z);
if (frame)
return frame->coordinatesOf(Vec(x,y,z));
else
return Vec(x,y,z);
}
/*! Same as projectedCoordinatesOf(), but with \c qreal parameters (\p src and \p res can be identical pointers). */
void Camera::getProjectedCoordinatesOf(const qreal src[3], qreal res[3], const Frame* frame) const
{
Vec r = projectedCoordinatesOf(Vec(src), frame);
for (int i=0; i<3; ++i)
res[i] = r[i];
}
/*! Same as unprojectedCoordinatesOf(), but with \c qreal parameters (\p src and \p res can be identical pointers). */
void Camera::getUnprojectedCoordinatesOf(const qreal src[3], qreal res[3], const Frame* frame) const
{
Vec r = unprojectedCoordinatesOf(Vec(src), frame);
for (int i=0; i<3; ++i)
res[i] = r[i];
}
///////////////////////////////////// KFI /////////////////////////////////////////
/*! Returns the KeyFrameInterpolator that defines the Camera path number \p i.
If path \p i is not defined for this index, the method returns a \c NULL pointer. */
KeyFrameInterpolator* Camera::keyFrameInterpolator(unsigned int i) const
{
if (kfi_.contains(i))
return kfi_[i];
else
return NULL;
}
/*! Sets the KeyFrameInterpolator that defines the Camera path of index \p i.
The previous keyFrameInterpolator() is lost and should be deleted by the calling method if
needed.
The KeyFrameInterpolator::interpolated() signal of \p kfi probably needs to be connected to the
Camera's associated QGLViewer::update() slot, so that when the Camera position is interpolated
using \p kfi, every interpolation step updates the display:
\code
myViewer.camera()->deletePath(3);
myViewer.camera()->setKeyFrameInterpolator(3, myKeyFrameInterpolator);
connect(myKeyFrameInterpolator, SIGNAL(interpolated()), myViewer, SLOT(update());
\endcode
\note These connections are done automatically when a Camera is attached to a QGLViewer, or when a
new KeyFrameInterpolator is defined using the QGLViewer::addKeyFrameKeyboardModifiers() and
QGLViewer::pathKey() (default is Alt+F[1-12]). See the <a href="../keyboard.html">keyboard page</a>
for details. */
void Camera::setKeyFrameInterpolator(unsigned int i, KeyFrameInterpolator* const kfi)
{
if (kfi)
kfi_[i] = kfi;
else
kfi_.remove(i);
}
/*! Adds the current Camera position() and orientation() as a keyFrame to the path number \p i.
This method can also be used if you simply want to save a Camera point of view (a path made of a
single keyFrame). Use playPath() to make the Camera play the keyFrame path (resp. restore
the point of view). Use deletePath() to clear the path.
The default keyboard shortcut for this method is Alt+F[1-12]. Set QGLViewer::pathKey() and
QGLViewer::addKeyFrameKeyboardModifiers().
If you use directly this method and the keyFrameInterpolator(i) does not exist, a new one is
created. Its KeyFrameInterpolator::interpolated() signal should then be connected to the
QGLViewer::update() slot (see setKeyFrameInterpolator()). */
void Camera::addKeyFrameToPath(unsigned int i)
{
if (!kfi_.contains(i))
setKeyFrameInterpolator(i, new KeyFrameInterpolator(frame()));
kfi_[i]->addKeyFrame(*(frame()));
}
/*! Makes the Camera follow the path of keyFrameInterpolator() number \p i.
If the interpolation is started, it stops it instead.
This method silently ignores undefined (empty) paths (see keyFrameInterpolator()).
The default keyboard shortcut for this method is F[1-12]. Set QGLViewer::pathKey() and
QGLViewer::playPathKeyboardModifiers(). */
void Camera::playPath(unsigned int i)
{
if (kfi_.contains(i)) {
if (kfi_[i]->interpolationIsStarted())
kfi_[i]->stopInterpolation();
else
kfi_[i]->startInterpolation();
}
}
/*! Resets the path of the keyFrameInterpolator() number \p i.
If this path is \e not being played (see playPath() and
KeyFrameInterpolator::interpolationIsStarted()), resets it to its starting position (see
KeyFrameInterpolator::resetInterpolation()). If the path is played, simply stops interpolation. */
void Camera::resetPath(unsigned int i)
{
if (kfi_.contains(i)) {
if ((kfi_[i]->interpolationIsStarted()))
kfi_[i]->stopInterpolation();
else
{
kfi_[i]->resetInterpolation();
kfi_[i]->interpolateAtTime(kfi_[i]->interpolationTime());
}
}
}
/*! Deletes the keyFrameInterpolator() of index \p i.
Disconnect the keyFrameInterpolator() KeyFrameInterpolator::interpolated() signal before deleting the
keyFrameInterpolator() if needed:
\code
disconnect(camera()->keyFrameInterpolator(i), SIGNAL(interpolated()), this, SLOT(update()));
camera()->deletePath(i);
\endcode */
void Camera::deletePath(unsigned int i)
{
if (kfi_.contains(i))
{
kfi_[i]->stopInterpolation();
delete kfi_[i];
kfi_.remove(i);
}
}
/*! Draws all the Camera paths defined by the keyFrameInterpolator().
Simply calls KeyFrameInterpolator::drawPath() for all the defined paths. The path color is the
current \c glColor().
\attention The OpenGL state is modified by this method: see KeyFrameInterpolator::drawPath(). */
void Camera::drawAllPaths()
{
for (QMap<unsigned int, KeyFrameInterpolator*>::ConstIterator it = kfi_.begin(), end=kfi_.end(); it != end; ++it)
(it.value())->drawPath(3, 5, sceneRadius());
}
////////////////////////////////////////////////////////////////////////////////
/*! Returns an XML \c QDomElement that represents the Camera.
\p name is the name of the QDomElement tag. \p doc is the \c QDomDocument factory used to create
QDomElement.
Concatenates the Camera parameters, the ManipulatedCameraFrame::domElement() and the paths'
KeyFrameInterpolator::domElement().
Use initFromDOMElement() to restore the Camera state from the resulting \c QDomElement.
If you want to save the Camera state in a file, use:
\code
QDomDocument document("myCamera");
doc.appendChild( myCamera->domElement("Camera", document) );
QFile f("myCamera.xml");
if (f.open(IO_WriteOnly))
{
QTextStream out(&f);
document.save(out, 2);
}
\endcode
Note that the QGLViewer::camera() is automatically saved by QGLViewer::saveStateToFile() when a
QGLViewer is closed. Use QGLViewer::restoreStateFromFile() to restore it back. */
QDomElement Camera::domElement(const QString& name, QDomDocument& document) const
{
QDomElement de = document.createElement(name);
QDomElement paramNode = document.createElement("Parameters");
paramNode.setAttribute("fieldOfView", QString::number(fieldOfView()));
paramNode.setAttribute("zNearCoefficient", QString::number(zNearCoefficient()));
paramNode.setAttribute("zClippingCoefficient", QString::number(zClippingCoefficient()));
paramNode.setAttribute("orthoCoef", QString::number(orthoCoef_));
paramNode.setAttribute("sceneRadius", QString::number(sceneRadius()));
paramNode.appendChild(sceneCenter().domElement("SceneCenter", document));
switch (type())
{
case Camera::PERSPECTIVE : paramNode.setAttribute("Type", "PERSPECTIVE"); break;
case Camera::ORTHOGRAPHIC : paramNode.setAttribute("Type", "ORTHOGRAPHIC"); break;
}
de.appendChild(paramNode);
QDomElement stereoNode = document.createElement("Stereo");
stereoNode.setAttribute("IODist", QString::number(IODistance()));
stereoNode.setAttribute("focusDistance", QString::number(focusDistance()));
stereoNode.setAttribute("physScreenWidth", QString::number(physicalScreenWidth()));
de.appendChild(stereoNode);
de.appendChild(frame()->domElement("ManipulatedCameraFrame", document));
// KeyFrame paths
for (QMap<unsigned int, KeyFrameInterpolator*>::ConstIterator it = kfi_.begin(), end=kfi_.end(); it != end; ++it)
{
QDomElement kfNode = (it.value())->domElement("KeyFrameInterpolator", document);
kfNode.setAttribute("index", QString::number(it.key()));
de.appendChild(kfNode);
}
return de;
}
/*! Restores the Camera state from a \c QDomElement created by domElement().
Use the following code to retrieve a Camera state from a file created using domElement():
\code
// Load DOM from file
QDomDocument document;
QFile f("myCamera.xml");
if (f.open(IO_ReadOnly))
{
document.setContent(&f);
f.close();
}
// Parse the DOM tree
QDomElement main = document.documentElement();
myCamera->initFromDOMElement(main);
\endcode
The frame() pointer is not modified by this method. The frame() state is however modified.
\attention The original keyFrameInterpolator() are deleted and should be copied first if they are shared. */
void Camera::initFromDOMElement(const QDomElement& element)
{
QDomElement child=element.firstChild().toElement();
QMutableMapIterator<unsigned int, KeyFrameInterpolator*> it(kfi_);
while (it.hasNext()) {
it.next();
deletePath(it.key());
}
while (!child.isNull())
{
if (child.tagName() == "Parameters")
{
// #CONNECTION# Default values set in constructor
setFieldOfView(DomUtils::qrealFromDom(child, "fieldOfView", M_PI/4.0));
setZNearCoefficient(DomUtils::qrealFromDom(child, "zNearCoefficient", 0.005));
setZClippingCoefficient(DomUtils::qrealFromDom(child, "zClippingCoefficient", sqrt(3.0)));
orthoCoef_ = DomUtils::qrealFromDom(child, "orthoCoef", tan(fieldOfView()/2.0));
setSceneRadius(DomUtils::qrealFromDom(child, "sceneRadius", sceneRadius()));
setType(PERSPECTIVE);
QString type = child.attribute("Type", "PERSPECTIVE");
if (type == "PERSPECTIVE") setType(Camera::PERSPECTIVE);
if (type == "ORTHOGRAPHIC") setType(Camera::ORTHOGRAPHIC);
QDomElement child2=child.firstChild().toElement();
while (!child2.isNull())
{
/* Although the scene does not change when a camera is loaded, restore the saved center and radius values.
Mainly useful when a the viewer is restored on startup, with possible additional cameras. */
if (child2.tagName() == "SceneCenter")
setSceneCenter(Vec(child2));
child2 = child2.nextSibling().toElement();
}
}
if (child.tagName() == "ManipulatedCameraFrame")
frame()->initFromDOMElement(child);
if (child.tagName() == "Stereo")
{
setIODistance(DomUtils::qrealFromDom(child, "IODist", 0.062));
setFocusDistance(DomUtils::qrealFromDom(child, "focusDistance", focusDistance()));
setPhysicalScreenWidth(DomUtils::qrealFromDom(child, "physScreenWidth", 0.5));
}
if (child.tagName() == "KeyFrameInterpolator")
{
unsigned int index = DomUtils::uintFromDom(child, "index", 0);
setKeyFrameInterpolator(index, new KeyFrameInterpolator(frame()));
if (keyFrameInterpolator(index))
keyFrameInterpolator(index)->initFromDOMElement(child);
}
child = child.nextSibling().toElement();
}
}
/*! Gives the coefficients of a 3D half-line passing through the Camera eye and pixel (x,y).
The origin of the half line (eye position) is stored in \p orig, while \p dir contains the properly
oriented and normalized direction of the half line.
\p x and \p y are expressed in Qt format (origin in the upper left corner). Use screenHeight() - y
to convert to OpenGL units.
This method is useful for analytical intersection in a selection method.
See the <a href="../examples/select.html">select example</a> for an illustration. */
void Camera::convertClickToLine(const QPoint& pixel, Vec& orig, Vec& dir) const
{
switch (type())
{
case Camera::PERSPECTIVE:
orig = position();
dir = Vec( ((2.0 * pixel.x() / screenWidth()) - 1.0) * tan(fieldOfView()/2.0) * aspectRatio(),
((2.0 * (screenHeight()-pixel.y()) / screenHeight()) - 1.0) * tan(fieldOfView()/2.0),
-1.0 );
dir = worldCoordinatesOf(dir) - orig;
dir.normalize();
break;
case Camera::ORTHOGRAPHIC:
{
GLdouble w,h;
getOrthoWidthHeight(w,h);
orig = Vec((2.0 * pixel.x() / screenWidth() - 1.0)*w, -(2.0 * pixel.y() / screenHeight() - 1.0)*h, 0.0);
orig = worldCoordinatesOf(orig);
dir = viewDirection();
break;
}
}
}
#ifndef DOXYGEN
/*! This method has been deprecated in libQGLViewer version 2.2.0 */
void Camera::drawCamera(qreal, qreal, qreal)
{
qWarning("drawCamera is deprecated. Use Camera::draw() instead.");
}
#endif
/*! Draws a representation of the Camera in the 3D world.
The near and far planes are drawn as quads, the frustum is drawn using lines and the camera up
vector is represented by an arrow to disambiguate the drawing. See the
<a href="../examples/standardCamera.html">standardCamera example</a> for an illustration.
Note that the current \c glColor and \c glPolygonMode are used to draw the near and far planes. See
the <a href="../examples/frustumCulling.html">frustumCulling example</a> for an example of
semi-transparent plane drawing. Similarly, the current \c glLineWidth and \c glColor is used to draw
the frustum outline.
When \p drawFarPlane is \c false, only the near plane is drawn. \p scale can be used to scale the
drawing: a value of 1.0 (default) will draw the Camera's frustum at its actual size.
This method assumes that the \c glMatrixMode is \c GL_MODELVIEW and that the current ModelView
matrix corresponds to the world coordinate system (as it is at the beginning of QGLViewer::draw()).
The Camera is then correctly positioned and orientated.
\note The drawing of a QGLViewer's own QGLViewer::camera() should not be visible, but may create
artefacts due to numerical imprecisions. */
void Camera::draw(bool drawFarPlane, qreal scale) const
{
glPushMatrix();
glMultMatrixd(frame()->worldMatrix());
// 0 is the upper left coordinates of the near corner, 1 for the far one
Vec points[2];
points[0].z = scale * zNear();
points[1].z = scale * zFar();
switch (type())
{
case Camera::PERSPECTIVE:
{
points[0].y = points[0].z * tan(fieldOfView()/2.0);
points[0].x = points[0].y * aspectRatio();
const qreal ratio = points[1].z / points[0].z;
points[1].y = ratio * points[0].y;
points[1].x = ratio * points[0].x;
break;
}
case Camera::ORTHOGRAPHIC:
{
GLdouble hw, hh;
getOrthoWidthHeight(hw, hh);
points[0].x = points[1].x = scale * qreal(hw);
points[0].y = points[1].y = scale * qreal(hh);
break;
}
}
const int farIndex = drawFarPlane?1:0;
// Near and (optionally) far plane(s)
glBegin(GL_QUADS);
for (int i=farIndex; i>=0; --i)
{
glNormal3d(0.0f, 0.0f, (i==0)?1.0f:-1.0f);
glVertex3d( points[i].x, points[i].y, -points[i].z);
glVertex3d(-points[i].x, points[i].y, -points[i].z);
glVertex3d(-points[i].x, -points[i].y, -points[i].z);
glVertex3d( points[i].x, -points[i].y, -points[i].z);
}
glEnd();
// Up arrow
const qreal arrowHeight = 1.5 * points[0].y;
const qreal baseHeight = 1.2 * points[0].y;
const qreal arrowHalfWidth = 0.5 * points[0].x;
const qreal baseHalfWidth = 0.3 * points[0].x;
glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
// Base
glBegin(GL_QUADS);
glVertex3d(-baseHalfWidth, points[0].y, -points[0].z);
glVertex3d( baseHalfWidth, points[0].y, -points[0].z);
glVertex3d( baseHalfWidth, baseHeight, -points[0].z);
glVertex3d(-baseHalfWidth, baseHeight, -points[0].z);
glEnd();
// Arrow
glBegin(GL_TRIANGLES);
glVertex3d( 0.0, arrowHeight, -points[0].z);
glVertex3d(-arrowHalfWidth, baseHeight, -points[0].z);
glVertex3d( arrowHalfWidth, baseHeight, -points[0].z);
glEnd();
// Frustum lines
switch (type())
{
case Camera::PERSPECTIVE :
glBegin(GL_LINES);
glVertex3d(0.0, 0.0, 0.0);
glVertex3d( points[farIndex].x, points[farIndex].y, -points[farIndex].z);
glVertex3d(0.0, 0.0, 0.0);
glVertex3d(-points[farIndex].x, points[farIndex].y, -points[farIndex].z);
glVertex3d(0.0, 0.0, 0.0);
glVertex3d(-points[farIndex].x, -points[farIndex].y, -points[farIndex].z);
glVertex3d(0.0, 0.0, 0.0);
glVertex3d( points[farIndex].x, -points[farIndex].y, -points[farIndex].z);
glEnd();
break;
case Camera::ORTHOGRAPHIC :
if (drawFarPlane)
{
glBegin(GL_LINES);
glVertex3d( points[0].x, points[0].y, -points[0].z);
glVertex3d( points[1].x, points[1].y, -points[1].z);
glVertex3d(-points[0].x, points[0].y, -points[0].z);
glVertex3d(-points[1].x, points[1].y, -points[1].z);
glVertex3d(-points[0].x, -points[0].y, -points[0].z);
glVertex3d(-points[1].x, -points[1].y, -points[1].z);
glVertex3d( points[0].x, -points[0].y, -points[0].z);
glVertex3d( points[1].x, -points[1].y, -points[1].z);
glEnd();
}
}
glPopMatrix();
}
/*! Returns the 6 plane equations of the Camera frustum.
The six 4-component vectors of \p coef respectively correspond to the left, right, near, far, top
and bottom Camera frustum planes. Each vector holds a plane equation of the form:
\code
a*x + b*y + c*z + d = 0
\endcode
where \c a, \c b, \c c and \c d are the 4 components of each vector, in that order.
See the <a href="../examples/frustumCulling.html">frustumCulling example</a> for an application.
This format is compatible with the \c glClipPlane() function. One camera frustum plane can hence be
applied in an other viewer to visualize the culling results:
\code
// Retrieve plane equations
GLdouble coef[6][4];
mainViewer->camera()->getFrustumPlanesCoefficients(coef);
// These two additional clipping planes (which must have been enabled)
// will reproduce the mainViewer's near and far clipping.
glClipPlane(GL_CLIP_PLANE0, coef[2]);
glClipPlane(GL_CLIP_PLANE1, coef[3]);
\endcode */
void Camera::getFrustumPlanesCoefficients(GLdouble coef[6][4]) const
{
// Computed once and for all
const Vec pos = position();
const Vec viewDir = viewDirection();
const Vec up = upVector();
const Vec right = rightVector();
const qreal posViewDir = pos * viewDir;
static Vec normal[6];
static GLdouble dist[6];
switch (type())
{
case Camera::PERSPECTIVE :
{
const qreal hhfov = horizontalFieldOfView() / 2.0;
const qreal chhfov = cos(hhfov);
const qreal shhfov = sin(hhfov);
normal[0] = - shhfov * viewDir;
normal[1] = normal[0] + chhfov * right;
normal[0] = normal[0] - chhfov * right;
normal[2] = -viewDir;
normal[3] = viewDir;
const qreal hfov = fieldOfView() / 2.0;
const qreal chfov = cos(hfov);
const qreal shfov = sin(hfov);
normal[4] = - shfov * viewDir;
normal[5] = normal[4] - chfov * up;
normal[4] = normal[4] + chfov * up;
for (int i=0; i<2; ++i)
dist[i] = pos * normal[i];
for (int j=4; j<6; ++j)
dist[j] = pos * normal[j];
// Natural equations are:
// dist[0,1,4,5] = pos * normal[0,1,4,5];
// dist[2] = (pos + zNear() * viewDir) * normal[2];
// dist[3] = (pos + zFar() * viewDir) * normal[3];
// 2 times less computations using expanded/merged equations. Dir vectors are normalized.
const qreal posRightCosHH = chhfov * pos * right;
dist[0] = -shhfov * posViewDir;
dist[1] = dist[0] + posRightCosHH;
dist[0] = dist[0] - posRightCosHH;
const qreal posUpCosH = chfov * pos * up;
dist[4] = - shfov * posViewDir;
dist[5] = dist[4] - posUpCosH;
dist[4] = dist[4] + posUpCosH;
break;
}
case Camera::ORTHOGRAPHIC :
normal[0] = -right;
normal[1] = right;
normal[4] = up;
normal[5] = -up;
GLdouble hw, hh;
getOrthoWidthHeight(hw, hh);
dist[0] = (pos - hw * right) * normal[0];
dist[1] = (pos + hw * right) * normal[1];
dist[4] = (pos + hh * up) * normal[4];
dist[5] = (pos - hh * up) * normal[5];
break;
}
// Front and far planes are identical for both camera types.
normal[2] = -viewDir;
normal[3] = viewDir;
dist[2] = -posViewDir - zNear();
dist[3] = posViewDir + zFar();
for (int i=0; i<6; ++i)
{
coef[i][0] = GLdouble(normal[i].x);
coef[i][1] = GLdouble(normal[i].y);
coef[i][2] = GLdouble(normal[i].z);
coef[i][3] = dist[i];
}
}
void Camera::onFrameModified() {
projectionMatrixIsUpToDate_ = false;
modelViewMatrixIsUpToDate_ = false;
}
|