123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249 |
- #include <unistd.h>
- #include <cstdio>
- #include "x.hpp"
- #include "options.hpp"
-
- int main( int argc, char** argv ) {
- int err = options->parseOptions( argc, argv );
- if ( err ) {
- return err;
- }
- int state = 0;
- bool running = true;
- slop::Rectangle* selection = NULL;
- slop::Rectangle* windowselection = NULL;
- Window window = None;
- std::string xdisplay = options->m_xdisplay;
- int padding = options->m_padding;
- int borderSize = options->m_borderSize;
- int tolerance = options->m_tolerance;
- float r = options->m_red;
- float g = options->m_green;
- float b = options->m_blue;
- bool keyboard = options->m_keyboard;
- timespec start, time;
- int cx = 0;
- int cy = 0;
- int cxoffset = 0;
- int cyoffset = 0;
- int woffset = 0;
- int hoffset = 0;
-
- // First we set up the x interface and grab the mouse,
- // if we fail for either we exit immediately.
- err = xengine->init( xdisplay.c_str() );
- if ( err ) {
- printf( "X=0\n" );
- printf( "Y=0\n" );
- printf( "W=0\n" );
- printf( "H=0\n" );
- return err;
- }
- err = xengine->grabCursor( slop::Cross );
- if ( err ) {
- printf( "X=0\n" );
- printf( "Y=0\n" );
- printf( "W=0\n" );
- printf( "H=0\n" );
- return err;
- }
- if ( keyboard ) {
- err = xengine->grabKeyboard();
- if ( err ) {
- printf( "X=0\n" );
- printf( "Y=0\n" );
- printf( "W=0\n" );
- printf( "H=0\n" );
- return err;
- }
- }
- clock_gettime( CLOCK_REALTIME, &start );
- while ( running ) {
- clock_gettime( CLOCK_REALTIME, &time );
- // "ticking" the xengine makes it process all queued events.
- xengine->tick();
- // If the user presses any key on the keyboard, exit the application.
- // Make sure at least options->m_gracetime has passed before allowing canceling
- double timei = double( time.tv_sec*1000000000L + time.tv_nsec )/1000000000.f;
- double starti = double( start.tv_sec*1000000000L + start.tv_nsec )/1000000000.f;
- if ( timei - starti > options->m_gracetime ) {
- if ( xengine->m_keypressed ) {
- printf( "X=0\n" );
- printf( "Y=0\n" );
- printf( "W=0\n" );
- printf( "H=0\n" );
- fprintf( stderr, "User pressed key. Canceled selection.\n" );
- state = -1;
- running = false;
- }
- } else {
- xengine->m_keypressed = false;
- }
- if ( xengine->mouseDown( 3 ) ) {
- printf( "X=0\n" );
- printf( "Y=0\n" );
- printf( "W=0\n" );
- printf( "H=0\n" );
- fprintf( stderr, "User right-clicked. Canceled selection.\n" );
- state = -1;
- running = false;
- }
- // Our adorable little state manager will handle what state we're in.
- switch ( state ) {
- default: {
- break;
- }
- case 0: {
- // If xengine has found a window we're hovering over (or if it changed)
- // create a rectangle around it so the user knows he/she can click on it.
- // --but only if the user wants us to
- if ( window != xengine->m_hoverXWindow && tolerance > 0 ) {
- // Make sure to delete the old selection rectangle.
- if ( windowselection ) {
- xengine->removeRect( windowselection ); // removeRect also dealloc's the rectangle for us.
- }
- slop::WindowRectangle t = xengine->m_hoverWindow;
- windowselection = new slop::Rectangle( t.m_x - t.m_border,
- t.m_y - t.m_border,
- t.m_width + t.m_border * 2,
- t.m_height + t.m_border * 2,
- borderSize, padding,
- r, g, b );
- xengine->addRect( windowselection );
- window = xengine->m_hoverXWindow;
- }
- // If the user clicked, remove the old selection rectangle and then
- // move on to the next state.
- if ( xengine->mouseDown( 1 ) ) {
- if ( windowselection ) {
- xengine->removeRect( windowselection );
- }
- state++;
- }
- break;
- }
- case 1: {
- // Set the mouse position of where we clicked, used so that click tolerance doesn't affect the rectangle's position.
- cx = xengine->m_mousex;
- cy = xengine->m_mousey;
- state++;
- break;
- }
- case 2: {
- // If the user has let go of the mouse button, we'll just
- // continue to the next state.
- if ( !xengine->mouseDown( 1 ) ) {
- state++;
- break;
- }
- // Check to make sure the user actually wants to drag for a selection before creating a rectangle.
- int w = xengine->m_mousex - cx;
- int h = xengine->m_mousey - cy;
- if ( ( std::abs( w ) >= tolerance || std::abs( h ) >= tolerance ) && !selection ) {
- selection = new slop::Rectangle( cx, cy, 0, 0, borderSize, padding, r, g, b );
- xengine->addRect( selection );
- } else if ( !selection ) {
- continue;
- }
- // We also detect which way the user is pulling and set the mouse icon accordingly.
- // and offset the rectangle to be accurate, this is because the mouse actually selects a pixel up and to the left.
- bool x = selection->m_flippedx;
- bool y = selection->m_flippedy;
- if ( !x && !y ) {
- cxoffset = 0;
- cyoffset = 0;
- woffset = 1;
- hoffset = 1;
- xengine->setCursor( slop::LowerRightCorner );
- } else if ( x && !y ) {
- cxoffset = 1;
- cyoffset = 0;
- woffset = -1;
- hoffset = 1;
- xengine->setCursor( slop::LowerLeftCorner );
- } else if ( !x && y ) {
- cxoffset = 0;
- cyoffset = 1;
- woffset = 1;
- hoffset = -1;
- xengine->setCursor( slop::UpperRightCorner );
- } else {
- cxoffset = 1;
- cyoffset = 1;
- woffset = -1;
- hoffset = -1;
- xengine->setCursor( slop::UpperLeftCorner );
- }
- // Set the selection rectangle's dimensions to mouse movement.
- // We use the function setDim since rectangles can't have negative widths,
- // and because the rectangles have borders and padding to worry about.
- selection->setPos( cx + cxoffset, cy + cyoffset );
- selection->setDim( w + woffset, h + hoffset );
- break;
- }
- case 3: {
- int x, y, w, h;
- // Exit the utility after this state runs once.
- running = false;
- if ( selection ) {
- // We pull the dimensions and positions from the selection rectangle.
- // The selection rectangle automatically converts the positions and
- // dimensions to absolute coordinates when we set them earilier.
- x = selection->m_x+selection->m_xoffset;
- y = selection->m_y+selection->m_yoffset;
- w = selection->m_width;
- h = selection->m_height;
- // Delete the rectangle.
- xengine->removeRect( selection );
- // if we're not hovering over a window, or our selection is larger than our tolerance
- // just print the selection.
- if ( w >= tolerance || h >= tolerance || xengine->m_hoverXWindow == None ) {
- printf( "X=%i\n", x );
- printf( "Y=%i\n", y );
- printf( "W=%i\n", w );
- printf( "H=%i\n", h );
- break;
- }
- }
- // Otherwise lets grab the window's dimensions and use those (with padding).
- // --but only if the user lets us, if the user doesn't just select a single pixel there.
- if ( tolerance > 0 ) {
- slop::WindowRectangle t = xengine->m_hoverWindow;
- x = t.m_x - padding - t.m_border;
- y = t.m_y - padding - t.m_border;
- w = t.m_width + t.m_border * 2 + padding*2;
- h = t.m_height + t.m_border * 2 + padding*2;
- } else {
- x = cx;
- y = cy;
- w = 1;
- h = 1;
- }
- printf( "X=%i\n", x );
- printf( "Y=%i\n", y );
- printf( "W=%i\n", w );
- printf( "H=%i\n", h );
- break;
- }
- }
- // No need to max out CPU
- // FIXME: This could be adjusted to measure how much time has passed,
- // we may very well need to max out the CPU if someone has a really- really
- // bad computer.
- usleep( 1000 );
- }
- xengine->releaseCursor();
- xengine->releaseKeyboard();
- // Try to process any last-second requests.
- xengine->tick();
- // Clean up global classes.
- delete xengine;
- delete options;
- // Wait to make sure X11 cleans up our window before we end.
- usleep( 100000 );
- // If we canceled the selection, return error.
- if ( state == -1 ) {
- return 1;
- }
- return 0;
- }
|