The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
#!/usr/bin/perl -w

# Copyright 2011, 2012 Kevin Ryde

# This file is part of X11-Protocol-Other.
#
# X11-Protocol-Other is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as published
# by the Free Software Foundation; either version 3, or (at your option) any
# later version.
#
# X11-Protocol-Other is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General
# Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with X11-Protocol-Other.  If not, see <http://www.gnu.org/licenses/>.

BEGIN { require 5 }
use strict;
use X11::Protocol;
use Test;

use lib 't';
use MyTestHelpers;
BEGIN { MyTestHelpers::nowarnings() }
END { MyTestHelpers::diag ("END"); }

# uncomment this to run the ### lines
#use Smart::Comments;

my $test_count = (tests => 61)[1];
plan tests => $test_count;

require X11::Protocol;
MyTestHelpers::diag ("X11::Protocol version ", X11::Protocol->VERSION);
MyTestHelpers::diag ("DISPLAY is ",
                     defined $ENV{'DISPLAY'} ? $ENV{'DISPLAY'} : 'undef');

my $display = $ENV{'DISPLAY'};
if (! defined $display) {
  foreach (1 .. $test_count) {
    skip ('No DISPLAY set', 1, 1);
  }
  exit 0;
}

# pass display arg so as not to get a "guess" warning
my $X;
if (! eval { $X = X11::Protocol->new ($display); }) {
  MyTestHelpers::diag ('Cannot connect to X server -- ',$@);
  foreach (1 .. $test_count) {
    skip ('Cannot connect to X server', 1, 1);
  }
  exit 0;
}
$X->QueryPointer($X->{'root'});  # sync

{
  my ($major_opcode, $first_event, $first_error)
    = $X->QueryExtension('XFIXES');
  if (! defined $major_opcode) {
    foreach (1 .. $test_count) {
      skip ('QueryExtension() no XFIXES on the server', 1, 1);
    }
    exit 0;
  }
  MyTestHelpers::diag ("XFIXES extension opcode=$major_opcode event=$first_event error=$first_error");
}

if (! $X->init_extension ('XFIXES')) {
  die "QueryExtension says XFIXES avaiable, but init_extension() failed";
}
$X->QueryPointer($X->root); # sync


my $other_X = X11::Protocol->new ($display);

my $other_window = $other_X->new_rsrc;
$other_X->CreateWindow ($other_window,
                        $X->root,         # parent
                        'InputOutput',
                        0,                # depth, from parent
                        'CopyFromParent', # visual
                        0,0,              # x,y
                        100,100,          # width,height
                        0,                # border
                        background_pixel => $X->white_pixel,
                       );
$other_X->QueryPointer($other_X->root); # sync



#------------------------------------------------------------------------------
# XFixesWindowRegionKind enum

{
  ok ($X->num('XFixesWindowRegionKind','Bounding'), 0);
  ok ($X->num('XFixesWindowRegionKind','Clip'),     1);

  ok ($X->num('XFixesWindowRegionKind',0), 0);
  ok ($X->num('XFixesWindowRegionKind',1), 1);

  ok ($X->interp('XFixesWindowRegionKind',0), 'Bounding');
  ok ($X->interp('XFixesWindowRegionKind',1), 'Clip');
}

#------------------------------------------------------------------------------
# XFixesSaveSetMode enum

{
  ok ($X->num('XFixesSaveSetMode','Insert'), 0);
  ok ($X->num('XFixesSaveSetMode','Delete'), 1);

  ok ($X->num('XFixesSaveSetMode',0), 0);
  ok ($X->num('XFixesSaveSetMode',1), 1);

  ok ($X->interp('XFixesSaveSetMode',0), 'Insert');
  ok ($X->interp('XFixesSaveSetMode',1), 'Delete');
}

#------------------------------------------------------------------------------
# XFixesSaveSetTarget enum

{
  ok ($X->num('XFixesSaveSetTarget','Nearest'), 0);
  ok ($X->num('XFixesSaveSetTarget','Root'), 1);

  ok ($X->num('XFixesSaveSetTarget',0), 0);
  ok ($X->num('XFixesSaveSetTarget',1), 1);

  ok ($X->interp('XFixesSaveSetTarget',0), 'Nearest');
  ok ($X->interp('XFixesSaveSetTarget',1), 'Root');
}

#------------------------------------------------------------------------------
# XFixesSaveSetMap enum

{
  ok ($X->num('XFixesSaveSetMap','Map'), 0);
  ok ($X->num('XFixesSaveSetMap','Unmap'), 1);

  ok ($X->num('XFixesSaveSetMap',0), 0);
  ok ($X->num('XFixesSaveSetMap',1), 1);

  ok ($X->interp('XFixesSaveSetMap',0), 'Map');
  ok ($X->interp('XFixesSaveSetMap',1), 'Unmap');
}

#------------------------------------------------------------------------------
# _num_xinputdevice()

{
  ok (X11::Protocol::Ext::XFIXES::_num_xinputdevice('AllDevices'), 0);
  ok (X11::Protocol::Ext::XFIXES::_num_xinputdevice('AllMasterDevices'), 1);
}


#------------------------------------------------------------------------------
# XFixesCursorNotify event

{
  my $aref = $X->{'ext'}->{'XFIXES'};
  my ($request_num, $event_num, $error_num, $obj) = @$aref;

  my $more;
  foreach $more (0, 1) {
    my $time;
    foreach $time ('CurrentTime', 103) {
      my %input = (# can't use "name" on an extension event, at least in 0.56
                   # name      => "XFixesCursorNotify",
                   synthetic => 1,
                   code      => $event_num+1,
                   sequence_number => 100,

                   subtype       => 'DisplayCursor',
                   window        => 102,
                   cursor_serial => 103,
                   time          => $time,
                   cursor_name   => 104);

      my $data = $X->pack_event(%input);
      ok (length($data), 32);

      my %output = $X->unpack_event($data);
      ### %output

      ok ($output{'code'},          $input{'code'});
      ok ($output{'name'},          'XFixesCursorNotify');
      ok ($output{'synthetic'},     $input{'synthetic'});
      ok ($output{'window'},        $input{'window'});
      ok ($output{'cursor_serial'}, $input{'cursor_serial'});
      ok ($output{'time'},          $input{'time'});
      ok ($output{'cursor_name'},   $input{'cursor_name'});
    }
  }
}


#------------------------------------------------------------------------------
# XFixesQueryVersion

{
  my $client_major = 1;
  my $client_minor = 0;
  my @ret = $X->XFixesQueryVersion ($client_major, $client_minor);
  MyTestHelpers::diag ("XFixesQueryVersion ask for $client_major.$client_minor got server version ", join('.',@ret));
  ok (scalar(@ret), 2);
  ok ($ret[0] <= $client_major, 1);
}
$X->QueryPointer($X->root); # sync


#------------------------------------------------------------------------------
# XFixesChangeSaveSet

{
  $X->XFixesChangeSaveSet ($other_window, 'Insert', 'Root', 'Unmap');
  $X->QueryPointer($X->root); # sync
  $X->XFixesChangeSaveSet ($other_window, 'Delete', 'Nearest', 'Map');
  $X->QueryPointer($X->root); # sync
}


#------------------------------------------------------------------------------
# XFixesSelectSelectionInput

{
  my $window = $X->new_rsrc;
  $X->CreateWindow ($window,
                    $X->root,         # parent
                    'InputOutput',
                    0,                # depth, from parent
                    'CopyFromParent', # visual
                    0,0,              # x,y
                    100,100,          # width,height
                    0,                # border
                    background_pixel => $X->white_pixel,
                   );
  $X->QueryPointer($X->root); # sync

  $X->XFixesSelectSelectionInput ($window, $X->atom('PRIMARY'), 0x07);
  $X->QueryPointer($X->root); # sync
  $X->XFixesSelectSelectionInput ($window, $X->atom('PRIMARY'), 0);
  $X->QueryPointer($X->root); # sync

  $X->DestroyWindow ($window);
  $X->QueryPointer($X->root); # sync
}

#------------------------------------------------------------------------------
# XFixesSelectCursorInput

{
  $X->XFixesSelectCursorInput ($X->root, 1);
  $X->QueryPointer($X->root); # sync
  $X->XFixesSelectCursorInput ($X->root, 0);
  $X->QueryPointer($X->root); # sync
}

#------------------------------------------------------------------------------
# XFixesGetCursorImage

{
  # Set a cursor before attempting to read back the image.  With xvfb of
  # x.org 1.11.4 at startup an attempt to XFixesGetCursorImage() or
  # XFixesGetCursorImageAndName() before a cursor has been set results in a
  # BadCursor error.  Sounds like a server bug or misfeature, but force it
  # as a workaround.

  my $cursor_font = $X->new_rsrc;
  $X->OpenFont ($cursor_font, "cursor");
  my $cursor = $X->new_rsrc;
  $X->CreateGlyphCursor ($cursor,
                         $cursor_font,  # cursor font
                         $cursor_font,  # mask font
                         0,  # X_cursor glyph
                         1,  # X_cursor mask
                         0,0,0,
                         0xFFFF, 0xFFFF, 0xFFFF);
  $X->CloseFont ($cursor_font);
  $X->QueryPointer($X->root); # sync

  my $screen_info;
  foreach $screen_info (@{$X->{'screens'}}) {
    $X->ChangeWindowAttributes ($screen_info->{'root'},
                                cursor => $cursor);
  }
  $X->FreeCursor ($cursor);
  $X->QueryPointer($X->root); # sync
}

{
  my ($root_x,$root_y, $width,$height, $xhot,$yhot, $serial, $pixels)
    = $X->XFixesGetCursorImage ();
  $X->QueryPointer($X->root); # sync

  ok (length($pixels), 4*$width*$height);
}


#------------------------------------------------------------------------------
exit 0;