diff options
| -rw-r--r-- | tracker-tobii-eyex/head-tracking.cpp | 89 | ||||
| -rw-r--r-- | tracker-tobii-eyex/lang/nl_NL.ts | 45 | ||||
| -rw-r--r-- | tracker-tobii-eyex/lang/ru_RU.ts | 45 | ||||
| -rw-r--r-- | tracker-tobii-eyex/lang/stub.ts | 45 | ||||
| -rw-r--r-- | tracker-tobii-eyex/lang/zh_CN.ts | 45 | ||||
| -rw-r--r-- | tracker-tobii-eyex/tobii-eyex-dialog.cpp | 5 | ||||
| -rw-r--r-- | tracker-tobii-eyex/tobii-eyex-dialog.ui | 449 | ||||
| -rw-r--r-- | tracker-tobii-eyex/tobii-eyex.cpp | 106 | ||||
| -rw-r--r-- | tracker-tobii-eyex/tobii-eyex.hpp | 57 | ||||
| -rw-r--r-- | tracker-tobii-eyex/tobii-settings.hpp | 11 | 
10 files changed, 150 insertions, 747 deletions
| diff --git a/tracker-tobii-eyex/head-tracking.cpp b/tracker-tobii-eyex/head-tracking.cpp new file mode 100644 index 00000000..8973293d --- /dev/null +++ b/tracker-tobii-eyex/head-tracking.cpp @@ -0,0 +1,89 @@ +#include "tobii-eyex.hpp" +#include "compat/math.hpp" +#include "compat/math-imports.hpp" + +real tobii_eyex_tracker::gain(real x) +{ +    // simple sigmoid +    x = clamp(x * 12 - 6, -6, 6); +    x = 1 / (1 + exp(-x)); +    x = x * 2 - 1; +    return clamp(x, -1, 1); +} + +void tobii_eyex_tracker::data(double* data) +{ +    real px, py, max_x, max_y; +    bool fresh; + +    { +        QMutexLocker l(&global_state_mtx); + +        if (!dev_state.is_valid()) +            return; + +        px = dev_state.px; +        py = dev_state.py; +        max_x = dev_state.display_res_x - 1; +        max_y = dev_state.display_res_y - 1; + +        fresh = dev_state.fresh; +        dev_state.fresh = false; +    } + +    if (fresh) +    { +        real x = (2*px - max_x)/max_x; // (-1)->1 +        real y = (2*py - max_y)/max_y; // idem + +        data[TX] = x * 50; +        data[TY] = y * -50; + +        const double dt = t.elapsed_seconds(); +        t.start(); + +        const double max_yaw = *s.acc_max_yaw, max_pitch = *s.acc_max_pitch; +        const double dz_ = *s.acc_dz; // closed set of 0->x, some arbitrary x < 1 + +        for (auto* k_ : { &x, &y }) +        { +            real& k = *k_; + +            if (std::fabs(k) < dz_) +                k = 0; +            else +            { +                // input has reduced dynamic range +                k -= copysign(dz_, k); +                k *= 1 + dz_; +            } +        } + +        const double c = *s.acc_speed; + +        // XXX check this for stability -sh 20180709 +        const double yaw_delta = gain(x) * c * dt; +        const double pitch_delta = gain(y) * c * dt; + +        yaw += yaw_delta; +        pitch += pitch_delta; + +        yaw = clamp(yaw, -max_yaw, max_yaw); +        pitch = clamp(pitch, -max_pitch, max_pitch); +    } + +    if (do_center) +    { +        do_center = false; +        yaw = 0; +        pitch = 0; +    } + +    data[Yaw] = yaw; +    data[Pitch] = pitch; +    data[Roll] = 0; +    data[TZ] = 0; // XXX TODO + +    // tan(x) in 0->.7 is almost linear. we don't need to adjust. +    // .7 is 40 degrees which is already quite a lot from the monitor. +} diff --git a/tracker-tobii-eyex/lang/nl_NL.ts b/tracker-tobii-eyex/lang/nl_NL.ts index 35c7f98a..47076f04 100644 --- a/tracker-tobii-eyex/lang/nl_NL.ts +++ b/tracker-tobii-eyex/lang/nl_NL.ts @@ -7,51 +7,6 @@          <source>Tracker options</source>          <translation type="unfinished"></translation>      </message> -    <message> -        <source>Tracking settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode shifts the view toward a target that may be offscreen then fixes upon it. -On the other hand, the snap mode allows for a quick glance outside the field of vision.</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Screen edge length</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max yaw</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max pitch</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Position output</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Enabled</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Snap mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Tracking mode</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Speed</source> -        <translation type="unfinished"></translation> -    </message>  </context>  <context>      <name>tobii_eyex_tracker</name> diff --git a/tracker-tobii-eyex/lang/ru_RU.ts b/tracker-tobii-eyex/lang/ru_RU.ts index 806c23b8..65ec8ca5 100644 --- a/tracker-tobii-eyex/lang/ru_RU.ts +++ b/tracker-tobii-eyex/lang/ru_RU.ts @@ -7,51 +7,6 @@          <source>Tracker options</source>          <translation type="unfinished"></translation>      </message> -    <message> -        <source>Tracking settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode shifts the view toward a target that may be offscreen then fixes upon it. -On the other hand, the snap mode allows for a quick glance outside the field of vision.</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Screen edge length</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max yaw</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max pitch</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Position output</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Enabled</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Snap mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Tracking mode</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Speed</source> -        <translation type="unfinished"></translation> -    </message>  </context>  <context>      <name>tobii_eyex_tracker</name> diff --git a/tracker-tobii-eyex/lang/stub.ts b/tracker-tobii-eyex/lang/stub.ts index 26acccaa..6e4c7d1a 100644 --- a/tracker-tobii-eyex/lang/stub.ts +++ b/tracker-tobii-eyex/lang/stub.ts @@ -7,51 +7,6 @@          <source>Tracker options</source>          <translation type="unfinished"></translation>      </message> -    <message> -        <source>Tracking settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode shifts the view toward a target that may be offscreen then fixes upon it. -On the other hand, the snap mode allows for a quick glance outside the field of vision.</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Screen edge length</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max yaw</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max pitch</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Position output</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Enabled</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Snap mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Tracking mode</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Speed</source> -        <translation type="unfinished"></translation> -    </message>  </context>  <context>      <name>tobii_eyex_tracker</name> diff --git a/tracker-tobii-eyex/lang/zh_CN.ts b/tracker-tobii-eyex/lang/zh_CN.ts index b7896059..6e4c7d1a 100644 --- a/tracker-tobii-eyex/lang/zh_CN.ts +++ b/tracker-tobii-eyex/lang/zh_CN.ts @@ -7,51 +7,6 @@          <source>Tracker options</source>          <translation type="unfinished"></translation>      </message> -    <message> -        <source>Tracking settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode shifts the view toward a target that may be offscreen then fixes upon it. -On the other hand, the snap mode allows for a quick glance outside the field of vision.</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Tracking mode</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Accumulative mode settings</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Screen edge length</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Speed</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max yaw</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Max pitch</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Position output</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Enabled</source> -        <translation type="unfinished"></translation> -    </message> -    <message> -        <source>Snap mode settings</source> -        <translation type="unfinished"></translation> -    </message>  </context>  <context>      <name>tobii_eyex_tracker</name> diff --git a/tracker-tobii-eyex/tobii-eyex-dialog.cpp b/tracker-tobii-eyex/tobii-eyex-dialog.cpp index 807542e1..6ab916be 100644 --- a/tracker-tobii-eyex/tobii-eyex-dialog.cpp +++ b/tracker-tobii-eyex/tobii-eyex-dialog.cpp @@ -7,10 +7,7 @@ tobii_eyex_dialog::tobii_eyex_dialog()      connect(ui.buttonBox, &QDialogButtonBox::accepted, this, &tobii_eyex_dialog::do_ok);      connect(ui.buttonBox, &QDialogButtonBox::rejected, this, &tobii_eyex_dialog::do_cancel); -    ui.tracking_mode->addItem("Snap", tobii_snap); -    ui.tracking_mode->addItem("Accumulative", tobii_acc); - -    tie_setting(s.mode, ui.tracking_mode); +    //tie_setting(s.mode, ui.tracking_mode);  }  void tobii_eyex_dialog::do_ok() diff --git a/tracker-tobii-eyex/tobii-eyex-dialog.ui b/tracker-tobii-eyex/tobii-eyex-dialog.ui index a6aecafb..37451f06 100644 --- a/tracker-tobii-eyex/tobii-eyex-dialog.ui +++ b/tracker-tobii-eyex/tobii-eyex-dialog.ui @@ -6,8 +6,8 @@     <rect>      <x>0</x>      <y>0</y> -    <width>659</width> -    <height>471</height> +    <width>184</width> +    <height>35</height>     </rect>    </property>    <property name="windowTitle"> @@ -22,447 +22,6 @@      <number>4</number>     </property>     <item row="0" column="0"> -    <widget class="QGroupBox" name="groupBox"> -     <property name="sizePolicy"> -      <sizepolicy hsizetype="Minimum" vsizetype="Maximum"> -       <horstretch>0</horstretch> -       <verstretch>0</verstretch> -      </sizepolicy> -     </property> -     <property name="title"> -      <string>Tracking settings</string> -     </property> -     <layout class="QVBoxLayout" name="verticalLayout"> -      <item> -       <widget class="QLabel" name="label_2"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="minimumSize"> -         <size> -          <width>617</width> -          <height>0</height> -         </size> -        </property> -        <property name="text"> -         <string>Accumulative mode shifts the view toward a target that may be offscreen then fixes upon it. -On the other hand, the snap mode allows for a quick glance outside the field of vision.</string> -        </property> -        <property name="textFormat"> -         <enum>Qt::PlainText</enum> -        </property> -        <property name="wordWrap"> -         <bool>true</bool> -        </property> -       </widget> -      </item> -      <item> -       <widget class="QFrame" name="frame"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Preferred" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="frameShape"> -         <enum>QFrame::NoFrame</enum> -        </property> -        <property name="frameShadow"> -         <enum>QFrame::Raised</enum> -        </property> -        <layout class="QGridLayout" name="gridLayout"> -         <property name="leftMargin"> -          <number>4</number> -         </property> -         <property name="topMargin"> -          <number>4</number> -         </property> -         <property name="rightMargin"> -          <number>4</number> -         </property> -         <property name="bottomMargin"> -          <number>4</number> -         </property> -         <property name="horizontalSpacing"> -          <number>9</number> -         </property> -         <property name="verticalSpacing"> -          <number>4</number> -         </property> -         <item row="0" column="0"> -          <widget class="QLabel" name="label"> -           <property name="text"> -            <string>Tracking mode</string> -           </property> -          </widget> -         </item> -         <item row="0" column="1"> -          <widget class="QComboBox" name="tracking_mode"> -           <property name="sizePolicy"> -            <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -             <horstretch>0</horstretch> -             <verstretch>0</verstretch> -            </sizepolicy> -           </property> -          </widget> -         </item> -         <item row="0" column="2"> -          <spacer name="horizontalSpacer"> -           <property name="orientation"> -            <enum>Qt::Horizontal</enum> -           </property> -           <property name="sizeHint" stdset="0"> -            <size> -             <width>10</width> -             <height>0</height> -            </size> -           </property> -          </spacer> -         </item> -        </layout> -       </widget> -      </item> -     </layout> -    </widget> -   </item> -   <item row="1" column="0"> -    <widget class="QGroupBox" name="groupBox_2"> -     <property name="sizePolicy"> -      <sizepolicy hsizetype="Minimum" vsizetype="Maximum"> -       <horstretch>0</horstretch> -       <verstretch>0</verstretch> -      </sizepolicy> -     </property> -     <property name="title"> -      <string>Accumulative mode settings</string> -     </property> -     <layout class="QGridLayout" name="gridLayout_2"> -      <item row="0" column="0"> -       <widget class="QLabel" name="label_3"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Screen edge length</string> -        </property> -       </widget> -      </item> -      <item row="0" column="1"> -       <widget class="QSlider" name="acc_inv_deadzone_slider"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Maximum"> -          <horstretch>9</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="minimum"> -         <number>1</number> -        </property> -        <property name="maximum"> -         <number>20</number> -        </property> -        <property name="pageStep"> -         <number>1</number> -        </property> -        <property name="orientation"> -         <enum>Qt::Horizontal</enum> -        </property> -       </widget> -      </item> -      <item row="0" column="2"> -       <widget class="QLabel" name="acc_inv_deadzone_label"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Preferred"> -          <horstretch>10</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string/> -        </property> -       </widget> -      </item> -      <item row="1" column="0"> -       <widget class="QLabel" name="label_4"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Speed</string> -        </property> -       </widget> -      </item> -      <item row="2" column="0"> -       <widget class="QLabel" name="label_6"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Max yaw</string> -        </property> -       </widget> -      </item> -      <item row="1" column="1"> -       <widget class="QSlider" name="acc_speed_slider"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Maximum"> -          <horstretch>9</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="maximum"> -         <number>100</number> -        </property> -        <property name="pageStep"> -         <number>1</number> -        </property> -        <property name="orientation"> -         <enum>Qt::Horizontal</enum> -        </property> -       </widget> -      </item> -      <item row="3" column="0"> -       <widget class="QLabel" name="label_7"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Max pitch</string> -        </property> -       </widget> -      </item> -      <item row="2" column="1"> -       <widget class="QComboBox" name="acc_max_yaw"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -       </widget> -      </item> -      <item row="3" column="1"> -       <widget class="QComboBox" name="acc_max_pitch"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -       </widget> -      </item> -      <item row="4" column="0"> -       <widget class="QLabel" name="label_11"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Position output</string> -        </property> -       </widget> -      </item> -      <item row="4" column="1"> -       <widget class="QCheckBox" name="acc_pos_output"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Enabled</string> -        </property> -       </widget> -      </item> -     </layout> -    </widget> -   </item> -   <item row="2" column="0"> -    <widget class="QGroupBox" name="groupBox_3"> -     <property name="sizePolicy"> -      <sizepolicy hsizetype="Minimum" vsizetype="Maximum"> -       <horstretch>0</horstretch> -       <verstretch>0</verstretch> -      </sizepolicy> -     </property> -     <property name="title"> -      <string>Snap mode settings</string> -     </property> -     <layout class="QGridLayout" name="gridLayout_3"> -      <item row="2" column="1" rowspan="2"> -       <widget class="QComboBox" name="snap_max_yaw"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -       </widget> -      </item> -      <item row="3" column="0" rowspan="2"> -       <widget class="QLabel" name="label_10"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Max pitch</string> -        </property> -       </widget> -      </item> -      <item row="2" column="0"> -       <widget class="QLabel" name="label_9"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Max yaw</string> -        </property> -       </widget> -      </item> -      <item row="1" column="0"> -       <widget class="QLabel" name="label_8"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Speed</string> -        </property> -       </widget> -      </item> -      <item row="0" column="1"> -       <widget class="QSlider" name="snap_inv_deazone_slider"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Maximum"> -          <horstretch>8</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="minimum"> -         <number>1</number> -        </property> -        <property name="maximum"> -         <number>20</number> -        </property> -        <property name="pageStep"> -         <number>1</number> -        </property> -        <property name="orientation"> -         <enum>Qt::Horizontal</enum> -        </property> -       </widget> -      </item> -      <item row="4" column="1"> -       <widget class="QComboBox" name="snap_max_pitch"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -       </widget> -      </item> -      <item row="0" column="2"> -       <widget class="QLabel" name="snap_inv_deadzone_label"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Preferred" vsizetype="Preferred"> -          <horstretch>10</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string/> -        </property> -       </widget> -      </item> -      <item row="1" column="1"> -       <widget class="QSlider" name="snap_speed_slider"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Expanding" vsizetype="Maximum"> -          <horstretch>8</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="maximum"> -         <number>100</number> -        </property> -        <property name="pageStep"> -         <number>1</number> -        </property> -        <property name="orientation"> -         <enum>Qt::Horizontal</enum> -        </property> -       </widget> -      </item> -      <item row="0" column="0"> -       <widget class="QLabel" name="label_5"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Screen edge length</string> -        </property> -       </widget> -      </item> -      <item row="5" column="1"> -       <widget class="QCheckBox" name="snap_pos_output"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Enabled</string> -        </property> -       </widget> -      </item> -      <item row="5" column="0"> -       <widget class="QLabel" name="label_12"> -        <property name="sizePolicy"> -         <sizepolicy hsizetype="Maximum" vsizetype="Maximum"> -          <horstretch>0</horstretch> -          <verstretch>0</verstretch> -         </sizepolicy> -        </property> -        <property name="text"> -         <string>Position output</string> -        </property> -       </widget> -      </item> -     </layout> -    </widget> -   </item> -   <item row="3" column="0">      <widget class="QDialogButtonBox" name="buttonBox">       <property name="sizePolicy">        <sizepolicy hsizetype="Minimum" vsizetype="Maximum"> @@ -479,6 +38,10 @@ On the other hand, the snap mode allows for a quick glance outside the field of   </widget>   <resources>    <include location="tobii-eyex-res.qrc"/> +  <include location="tobii-eyex-res.qrc"/> +  <include location="tobii-eyex-res.qrc"/> +  <include location="tobii-eyex-res.qrc"/> +  <include location="tobii-eyex-res.qrc"/>   </resources>   <connections/>  </ui> diff --git a/tracker-tobii-eyex/tobii-eyex.cpp b/tracker-tobii-eyex/tobii-eyex.cpp index 8f2edd62..af0d0c05 100644 --- a/tracker-tobii-eyex/tobii-eyex.cpp +++ b/tracker-tobii-eyex/tobii-eyex.cpp @@ -1,6 +1,7 @@  #include "tobii-eyex.hpp"  #include "compat/math-imports.hpp" +#include <tuple>  #include <cstdlib>  #include <cstdio> @@ -32,17 +33,7 @@ static inline tobii_eyex_tracker& to_self(TX_USERPARAM param)      return *reinterpret_cast<tobii_eyex_tracker*>(param);  } -tobii_eyex_tracker::tobii_eyex_tracker() : -    dev_ctx(TX_EMPTY_HANDLE), -    conn_state_changed_ticket(TX_INVALID_TICKET), -    event_handler_ticket(TX_INVALID_TICKET), -    state_snapshot(TX_EMPTY_HANDLE), -    display_state(TX_EMPTY_HANDLE), -    yaw(0), -    pitch(0), -    do_center(false) -{ -} +tobii_eyex_tracker::tobii_eyex_tracker() = default;  void tobii_eyex_tracker::call_tx_deinit()  { @@ -54,12 +45,12 @@ tobii_eyex_tracker::~tobii_eyex_tracker()  {      dbg_verbose("dtor"); -    (void) txDisableConnection(dev_ctx); +    (void) txDisableConnection(ctx);      (void) txReleaseObject(&state_snapshot);      bool status = true; -    status &= txShutdownContext(dev_ctx, TX_CLEANUPTIMEOUT_FORCEIMMEDIATE, TX_FALSE) == TX_RESULT_OK; -    status &= txReleaseContext(&dev_ctx) == TX_RESULT_OK; +    status &= txShutdownContext(ctx, TX_CLEANUPTIMEOUT_FORCEIMMEDIATE, TX_FALSE) == TX_RESULT_OK; +    status &= txReleaseContext(&ctx) == TX_RESULT_OK;      // the API cleanup function needs to be called exactly once over image lifetime.      // client software communicates with a service and a desktop program. @@ -129,7 +120,7 @@ void tobii_eyex_tracker::snapshot_committed_handler(TX_CONSTHANDLE async_data_ha          dbg_notice("snapshot bad result code") << result;  } -void tobii_eyex_tracker::connection_state_change_handler(TX_CONNECTIONSTATE state, TX_USERPARAM param) +void tobii_eyex_tracker::state_change_handler(TX_CONNECTIONSTATE state, TX_USERPARAM param)  {      tobii_eyex_tracker& self = to_self(param); @@ -142,7 +133,7 @@ void tobii_eyex_tracker::connection_state_change_handler(TX_CONNECTIONSTATE stat              dbg_notice("connected but failed to initialize data stream");          else          { -            txGetStateAsync(self.dev_ctx, TX_STATEPATH_EYETRACKINGSCREENBOUNDS, display_state_handler, param); +            txGetStateAsync(self.ctx, TX_STATEPATH_EYETRACKINGSCREENBOUNDS, display_state_handler, param);              dbg_notice("connected, data stream ok");          }      } @@ -222,11 +213,11 @@ module_status tobii_eyex_tracker::start_tracker(QFrame*)      bool status = true;      status &= txInitializeEyeX(TX_EYEXCOMPONENTOVERRIDEFLAG_NONE, nullptr, nullptr, nullptr, nullptr) == TX_RESULT_OK; -    status &= txCreateContext(&dev_ctx, TX_FALSE) == TX_RESULT_OK; -    status &= register_state_snapshot(dev_ctx, &state_snapshot); -    status &= txRegisterConnectionStateChangedHandler(dev_ctx, &conn_state_changed_ticket, connection_state_change_handler, reinterpret_cast<TX_USERPARAM>(this)) == TX_RESULT_OK; -    status &= txRegisterEventHandler(dev_ctx, &event_handler_ticket, event_handler, reinterpret_cast<TX_USERPARAM>(this)) == TX_RESULT_OK; -    status &= txEnableConnection(dev_ctx) == TX_RESULT_OK; +    status &= txCreateContext(&ctx, TX_FALSE) == TX_RESULT_OK; +    status &= register_state_snapshot(ctx, &state_snapshot); +    status &= txRegisterConnectionStateChangedHandler(ctx, &state_changed_ticket, state_change_handler, (TX_USERPARAM)this) == TX_RESULT_OK; +    status &= txRegisterEventHandler(ctx, &event_handler_ticket, event_handler, (TX_USERPARAM)this) == TX_RESULT_OK; +    status &= txEnableConnection(ctx) == TX_RESULT_OK;      if (!status)          return error(tr("Connection can't be established. device missing?")); @@ -234,78 +225,15 @@ module_status tobii_eyex_tracker::start_tracker(QFrame*)          return status_ok();  } -tobii_eyex_tracker::num tobii_eyex_tracker::gain(num x) +bool tobii_eyex_tracker::center()  { -    return 1; +    do_center = true; +    return true;  } -static inline double signum(double x) -{ -    return !(x < 0) - (x < 0); -} - -void tobii_eyex_tracker::data(double* data) -{ -    TX_REAL px, py, dw, dh, x_, y_; -    bool fresh; - -    { -        QMutexLocker l(&global_state_mtx); - -        if (!dev_state.is_valid()) -            return; - -        px = dev_state.px; -        py = dev_state.py; -        dw = dev_state.display_res_x; -        dh = dev_state.display_res_y; - -        fresh = dev_state.fresh; -        dev_state.fresh = false; -    } - -    x_ = (px-dw/2.) / (dw/2.); -    y_ = (py-dh/2.) / (dh/2.); +settings::settings() : opts("tobii-eyex") {} -    data[TX] = x_ * 50; -    data[TY] = y_ * -50; - -    if (fresh) -    { -        const double dt = t.elapsed_seconds(); -        t.start(); - -        using std::fabs; - -        constexpr double max_yaw = 45, max_pitch = 30; -        constexpr double c_yaw = 3; -        constexpr double c_pitch = c_yaw * max_pitch / max_yaw; - -        const double yaw_delta = gain(fabs(x_)) * signum(x_) * c_yaw * dt; -        const double pitch_delta = gain(fabs(y_)) * signum(y_) * c_pitch * dt; - -        yaw += yaw_delta; -        pitch += pitch_delta; - -        yaw = clamp(yaw, -max_yaw, max_yaw); -        pitch = clamp(pitch, -max_pitch, max_pitch); -    } - -    if (do_center) -    { -        do_center = false; -        yaw = 0; -        pitch = 0; -    } - -    data[Yaw] = yaw; -    data[Pitch] = pitch; -    data[Roll] = 0; -    data[TZ] = 0; // XXX TODO - -    // tan(x) in 0->.7 is almost linear. we don't need to adjust. -    // .7 is 40 degrees which is already quite a lot from the monitor. -} +state::state() = default;  #include "tobii-eyex-dialog.hpp" diff --git a/tracker-tobii-eyex/tobii-eyex.hpp b/tracker-tobii-eyex/tobii-eyex.hpp index 84407930..def2ea2e 100644 --- a/tracker-tobii-eyex/tobii-eyex.hpp +++ b/tracker-tobii-eyex/tobii-eyex.hpp @@ -23,6 +23,20 @@ using namespace options;  #include <QObject>  #include <QMutex> +//using real = TX_REAL; +using real = double; + +struct state +{ +    real display_res_x = -1, display_res_y = -1; +    real px = -1, py = -1; +    real last_timestamp = -1; +    bool fresh = false; + +    state(); +    bool is_valid() const { return !(display_res_x < 0 || px < 0); } +}; +  class tobii_eyex_tracker : public TR, public ITracker  {      Q_OBJECT @@ -30,13 +44,9 @@ class tobii_eyex_tracker : public TR, public ITracker  public:      tobii_eyex_tracker();      ~tobii_eyex_tracker() override; -    module_status start_tracker(QFrame *) override; +    module_status start_tracker(QFrame*) override;      void data(double *data) override; -    bool center() override -    { -        do_center = true; -        return true; -    } +    bool center() override;  private:      static constexpr inline const char* const client_id = "opentrack-tobii-eyex"; @@ -44,47 +54,36 @@ private:      static bool register_state_snapshot(TX_CONTEXTHANDLE ctx, TX_HANDLE* state_snapshot_ptr);      static std::atomic_flag atexit_done; -    static void TX_CALLCONVENTION connection_state_change_handler(TX_CONNECTIONSTATE state, TX_USERPARAM param); +    static void TX_CALLCONVENTION state_change_handler(TX_CONNECTIONSTATE state, TX_USERPARAM param);      static void TX_CALLCONVENTION event_handler(TX_CONSTHANDLE async_data_handle, TX_USERPARAM param);      void gaze_data_handler(TX_HANDLE gaze_data_handle);      static void TX_CALLCONVENTION snapshot_committed_handler(TX_CONSTHANDLE async_data_handle, TX_USERPARAM param);      static void TX_CALLCONVENTION display_state_handler(TX_CONSTHANDLE async_data_handle, TX_USERPARAM param);      void process_display_state(TX_HANDLE display_state_handle); -    using num = double; +    real gain(real x); -    num gain(num x); +    state dev_state; +    real yaw = 0, pitch = 0; -    TX_CONTEXTHANDLE dev_ctx; -    TX_TICKET conn_state_changed_ticket; -    TX_TICKET event_handler_ticket; -    TX_HANDLE state_snapshot; -    TX_HANDLE display_state; +    TX_CONTEXTHANDLE ctx = TX_EMPTY_HANDLE; +    TX_TICKET state_changed_ticket = TX_INVALID_TICKET; +    TX_TICKET event_handler_ticket = TX_INVALID_TICKET; +    TX_HANDLE state_snapshot = TX_EMPTY_HANDLE; +    TX_HANDLE display_state = TX_EMPTY_HANDLE;      QMutex global_state_mtx;      settings s;      Timer t; -    struct state -    { -        TX_REAL display_res_x, display_res_y; -        TX_REAL px, py; -        TX_REAL last_timestamp; -        bool fresh; - -        state() : display_res_x(-1), display_res_y(-1), px(-1), py(-1), last_timestamp(0), fresh(false) {} -        bool is_valid() const { return !(display_res_x < 0 || px < 0); } -    } dev_state; - -    double yaw, pitch; -    std::atomic<bool> do_center; +    std::atomic<bool> do_center = false;  };  class tobii_eyex_metadata : public Metadata  {      Q_OBJECT -    QString name() { return QString("Tobii EyeX"); } -    QIcon icon() { return QIcon(":/images/tobii-eyex-logo.png"); } +    QString name() override { return QString("Tobii EyeX"); } +    QIcon icon() override { return QIcon(":/images/tobii-eyex-logo.png"); }  }; diff --git a/tracker-tobii-eyex/tobii-settings.hpp b/tracker-tobii-eyex/tobii-settings.hpp index 49b9cae8..6101a28f 100644 --- a/tracker-tobii-eyex/tobii-settings.hpp +++ b/tracker-tobii-eyex/tobii-settings.hpp @@ -21,6 +21,7 @@ enum max_pitch  struct settings final : public opts  { +#if 0      value<tobii_mode> mode { b, "mode", tobii_snap };      value<slider_value> snap_speed {b, "snap-speed", { .1, .05, 1 }}, @@ -28,9 +29,15 @@ struct settings final : public opts      value<slider_value> acc_speed {b, "acc-speed", { .1, .05, 1 }},                          acc_dz_len {b, "acc-screen-edge-length", { .1, .1, 1 }};      value<max_yaw> snap_yaw {b, "snap-max-yaw", y20}, -                   acc_yaw {b, "acc-max-yaw", y20};      value<max_pitch> snap_pitch {b, "snap-max-pitch", p15},                       acc_pitch {b, "acc-max-pitch", p15}; +                     acc_yaw {b, "acc-max-yaw", y20}; +#endif +    value<slider_value> acc_speed { b, "acc-max-speed-deg", { 3, 1, 10 } }; +    value<slider_value> acc_dz { b, "acc-deadzone", { .15, .0, .3 } }; -    settings() : opts("tobii-eyex") {} +    value<slider_value> acc_max_yaw { b, "acc-max-yaw", { 30, 15, 45} }, +                        acc_max_pitch { b, "acc-max-pitch", { 30, 15, 45 } }; + +    settings();  }; | 
