2009-09-03 23:15:12 +02:00
|
|
|
|
#region License
|
|
|
|
|
//
|
|
|
|
|
// The Open Toolkit Library License
|
|
|
|
|
//
|
|
|
|
|
// Copyright (c) 2006 - 2009 the Open Toolkit library.
|
|
|
|
|
//
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
// in the Software without restriction, including without limitation the rights to
|
|
|
|
|
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
|
// the Software, and to permit persons to whom the Software is furnished to do
|
|
|
|
|
// so, subject to the following conditions:
|
|
|
|
|
//
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
|
//
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
|
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
|
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
|
// OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
//
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
using System;
|
2010-10-20 16:33:23 +02:00
|
|
|
|
using System.Collections.Specialized;
|
2009-09-03 21:01:11 +02:00
|
|
|
|
using System.Text;
|
|
|
|
|
|
|
|
|
|
namespace OpenTK.Input
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
2009-09-03 22:53:48 +02:00
|
|
|
|
/// Encapsulates the state of a Keyboard device.
|
2009-09-03 21:01:11 +02:00
|
|
|
|
/// </summary>
|
|
|
|
|
public struct KeyboardState : IEquatable<KeyboardState>
|
|
|
|
|
{
|
2009-09-03 22:53:48 +02:00
|
|
|
|
#region Fields
|
|
|
|
|
|
2010-10-20 17:14:26 +02:00
|
|
|
|
// Allocate enough ints to store all keyboard keys
|
2010-10-28 10:14:28 +02:00
|
|
|
|
const int IntSize = sizeof(int);
|
|
|
|
|
const int NumInts = ((int)Key.LastKey + IntSize - 1) / IntSize;
|
2010-10-20 16:33:23 +02:00
|
|
|
|
// The following line triggers bogus CS0214 in gmcs 2.0.1, sigh...
|
2010-10-20 17:14:26 +02:00
|
|
|
|
unsafe fixed int Keys[NumInts];
|
2009-09-03 21:01:11 +02:00
|
|
|
|
|
2009-09-03 22:53:48 +02:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region Public Members
|
|
|
|
|
|
2010-10-28 10:14:28 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a <see cref="System.Boolean"/> indicating whether the specified
|
|
|
|
|
/// <see cref="OpenTK.Input.Key"/> is pressed.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="key">The <see cref="OpenTK.Input.Key"/> to check.</param>
|
|
|
|
|
/// <returns>True if key is pressed; false otherwise.</returns>
|
|
|
|
|
public bool this[Key key]
|
|
|
|
|
{
|
|
|
|
|
get { return IsKeyDown(key); }
|
|
|
|
|
internal set
|
|
|
|
|
{
|
|
|
|
|
if (value)
|
|
|
|
|
EnableBit((int)key);
|
|
|
|
|
else
|
|
|
|
|
DisableBit((int)key);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-03 22:53:48 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a <see cref="System.Boolean"/> indicating whether this key is down.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="key">The <see cref="OpenTK.Input.Key"/> to check.</param>
|
|
|
|
|
public bool IsKeyDown(Key key)
|
|
|
|
|
{
|
2010-10-20 16:33:23 +02:00
|
|
|
|
return ReadBit((int)key);
|
2009-09-03 22:53:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a <see cref="System.Boolean"/> indicating whether this key is up.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="key">The <see cref="OpenTK.Input.Key"/> to check.</param>
|
|
|
|
|
public bool IsKeyUp(Key key)
|
2009-09-03 21:01:11 +02:00
|
|
|
|
{
|
2010-10-20 16:33:23 +02:00
|
|
|
|
return !ReadBit((int)key);
|
2009-09-03 21:01:11 +02:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-28 10:41:48 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Checks whether two <see cref="KeyboardState" /> instances are equal.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="left">
|
|
|
|
|
/// A <see cref="KeyboardState"/> instance.
|
|
|
|
|
/// </param>
|
|
|
|
|
/// <param name="right">
|
|
|
|
|
/// A <see cref="KeyboardState"/> instance.
|
|
|
|
|
/// </param>
|
|
|
|
|
/// <returns>
|
|
|
|
|
/// True if both left is equal to right; false otherwise.
|
|
|
|
|
/// </returns>
|
|
|
|
|
public static bool operator ==(KeyboardState left, KeyboardState right)
|
|
|
|
|
{
|
|
|
|
|
return left.Equals(right);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Checks whether two <see cref="KeyboardState" /> instances are not equal.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="left">
|
|
|
|
|
/// A <see cref="KeyboardState"/> instance.
|
|
|
|
|
/// </param>
|
|
|
|
|
/// <param name="right">
|
|
|
|
|
/// A <see cref="KeyboardState"/> instance.
|
|
|
|
|
/// </param>
|
|
|
|
|
/// <returns>
|
|
|
|
|
/// True if both left is not equal to right; false otherwise.
|
|
|
|
|
/// </returns>
|
|
|
|
|
public static bool operator !=(KeyboardState left, KeyboardState right)
|
|
|
|
|
{
|
|
|
|
|
return !left.Equals(right);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Compares to an object instance for equality.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="obj">
|
|
|
|
|
/// The <see cref="System.Object"/> to compare to.
|
|
|
|
|
/// </param>
|
|
|
|
|
/// <returns>
|
|
|
|
|
/// True if this instance is equal to obj; false otherwise.
|
|
|
|
|
/// </returns>
|
|
|
|
|
public override bool Equals(object obj)
|
|
|
|
|
{
|
|
|
|
|
if (obj is KeyboardState)
|
|
|
|
|
{
|
|
|
|
|
return this == (KeyboardState)obj;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Generates a hashcode for the current instance.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>
|
|
|
|
|
/// A <see cref="System.Int32"/> represting the hashcode for this instance.
|
|
|
|
|
/// </returns>
|
|
|
|
|
public override int GetHashCode()
|
|
|
|
|
{
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
|
|
|
|
fixed (int* k = Keys)
|
|
|
|
|
{
|
|
|
|
|
int hashcode = 0;
|
|
|
|
|
for (int i = 0; i < NumInts; i++)
|
|
|
|
|
hashcode ^= (k + i)->GetHashCode();
|
|
|
|
|
return hashcode;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-03 21:01:11 +02:00
|
|
|
|
#endregion
|
|
|
|
|
|
2009-09-03 22:53:48 +02:00
|
|
|
|
#region Internal Members
|
|
|
|
|
|
2010-10-20 16:33:23 +02:00
|
|
|
|
internal bool ReadBit(int offset)
|
2009-09-03 22:53:48 +02:00
|
|
|
|
{
|
2010-10-28 10:14:28 +02:00
|
|
|
|
ValidateOffset(offset);
|
|
|
|
|
|
2010-10-20 16:33:23 +02:00
|
|
|
|
int int_offset = offset / 32;
|
|
|
|
|
int bit_offset = offset % 32;
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
|
|
|
|
fixed (int* k = Keys)
|
|
|
|
|
{
|
|
|
|
|
return (*(k + int_offset) & (1 << bit_offset)) != 0u;
|
|
|
|
|
}
|
|
|
|
|
}
|
2009-09-03 22:53:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-20 16:33:23 +02:00
|
|
|
|
internal void EnableBit(int offset)
|
2009-09-03 22:53:48 +02:00
|
|
|
|
{
|
2010-10-28 10:14:28 +02:00
|
|
|
|
ValidateOffset(offset);
|
|
|
|
|
|
2010-10-20 16:33:23 +02:00
|
|
|
|
int int_offset = offset / 32;
|
|
|
|
|
int bit_offset = offset % 32;
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
|
|
|
|
fixed (int* k = Keys)
|
|
|
|
|
{
|
|
|
|
|
*(k + int_offset) |= 1 << bit_offset;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal void DisableBit(int offset)
|
|
|
|
|
{
|
2010-10-28 10:14:28 +02:00
|
|
|
|
ValidateOffset(offset);
|
|
|
|
|
|
2010-10-20 16:33:23 +02:00
|
|
|
|
int int_offset = offset / 32;
|
|
|
|
|
int bit_offset = offset % 32;
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
|
|
|
|
fixed (int* k = Keys)
|
|
|
|
|
{
|
|
|
|
|
*(k + int_offset) &= ~(1 << bit_offset);
|
|
|
|
|
}
|
|
|
|
|
}
|
2009-09-03 22:53:48 +02:00
|
|
|
|
}
|
2009-09-03 21:01:11 +02:00
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2010-10-28 10:14:28 +02:00
|
|
|
|
#region Private Members
|
|
|
|
|
|
|
|
|
|
static void ValidateOffset(int offset)
|
|
|
|
|
{
|
|
|
|
|
if (offset < 0 || offset >= NumInts * IntSize)
|
|
|
|
|
throw new ArgumentOutOfRangeException("offset");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2009-09-03 21:01:11 +02:00
|
|
|
|
#region IEquatable<KeyboardState> Members
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Compares two KeyboardState instances.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="other">The instance to compare two.</param>
|
|
|
|
|
/// <returns>True, if both instances are equal; false otherwise.</returns>
|
|
|
|
|
public bool Equals(KeyboardState other)
|
|
|
|
|
{
|
2010-10-20 16:33:23 +02:00
|
|
|
|
bool equal = true;
|
|
|
|
|
unsafe
|
|
|
|
|
{
|
2010-10-22 10:16:37 +02:00
|
|
|
|
int* k2 = other.Keys;
|
2010-10-20 16:33:23 +02:00
|
|
|
|
fixed (int* k1 = Keys)
|
|
|
|
|
{
|
2010-10-20 17:14:26 +02:00
|
|
|
|
for (int i = 0; equal && i < NumInts; i++)
|
2010-10-20 16:33:23 +02:00
|
|
|
|
equal &= *(k1 + i) == *(k2 + i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return equal;
|
2009-09-03 21:01:11 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
|
|
|
|
}
|