Whoops! Forgot to add these files to the repo in the r19681. Thanks for pointing this out, Marcus!

git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@19685 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
DarkWyrm 2007-01-03 02:38:05 +00:00
parent cb8f258f72
commit dc887e5ed2
2 changed files with 329 additions and 0 deletions

View File

@ -0,0 +1,254 @@
/*
* Copyright (c) 2006, Haiku, Inc.
* Distributed under the terms of the MIT license.
*
* Author:
* DarkWyrm <darkwyrm@earthlink.net>
*/
#include "AutoTextControl.h"
#include <Window.h>
#include <String.h>
#include <stdio.h>
#include <ctype.h>
AutoTextControlFilter::AutoTextControlFilter(AutoTextControl *box)
: BMessageFilter(B_PROGRAMMED_DELIVERY, B_ANY_SOURCE,B_KEY_DOWN),
fBox(box),
fMessenger(NULL),
fCurrentMessage(NULL)
{
}
AutoTextControlFilter::~AutoTextControlFilter(void)
{
}
filter_result AutoTextControlFilter::Filter(BMessage *msg, BHandler **target)
{
// This is really slick -- all that is needed to allow Escape key cancelling is
// just calling SetEscapeCancel(true) for just *one* AutoTextControl in a window. *heh*
int32 rawchar,mod;
msg->FindInt32("raw_char",&rawchar);
msg->FindInt32("modifiers",&mod);
if(rawchar == B_ESCAPE)
{
if(mod & B_COMMAND_KEY)
return B_DISPATCH_MESSAGE;
if(IsEscapeCancel())
{
BLooper *loop = (*target)->Looper();
if(loop)
{
BMessenger msgr(loop);
msgr.SendMessage(B_QUIT_REQUESTED);
return B_SKIP_MESSAGE;
}
}
}
BView *v=dynamic_cast<BView*>(*target);
if(!v || strcmp("_input_",v->Name())!=0)
return B_DISPATCH_MESSAGE;
AutoTextControl *text = dynamic_cast<AutoTextControl*>(v->Parent());
if(!text || text!=fBox)
return B_DISPATCH_MESSAGE;
// handle instances where numlock is off and the user tries to punch in numbers.
// Instead of simply blocking the resulting keypresses, transform them into legit ones
// and turn NumLock on for the user. This doesn't work on R5, so only do it on other
// versions of BeOS
#ifndef HAIKU_TARGET_PLATFORM_R5
int32 scancode;
if(msg->FindInt32("key",&scancode)!=B_OK)
scancode = -1;
HandleNoNumLock(scancode,rawchar,msg);
#endif
fCurrentMessage = msg;
filter_result result = KeyFilter(rawchar,mod);
fCurrentMessage = NULL;
if(fBox->fCharLimit && result == B_DISPATCH_MESSAGE)
{
// See to it that we still allow shortcut keys
if(mod & B_COMMAND_KEY)
return B_DISPATCH_MESSAGE;
// We don't use strlen() because it is not UTF-8 aware, which can affect
// how many characters can be typed.
if(isprint(rawchar) && (uint32)BString(text->Text()).CountChars()==text->fCharLimit)
return B_SKIP_MESSAGE;
}
return result;
}
filter_result AutoTextControlFilter::KeyFilter(const int32 &rawchar, const int32 &mod)
{
if(fMessenger)
fMessenger->SendMessage(fBox->ModificationMessage());
else
if(fBox)
fBox->Invoke();
return B_DISPATCH_MESSAGE;
}
void AutoTextControlFilter::SetMessenger(BMessenger *msgr)
{
if(fMessenger)
delete fMessenger;
fMessenger = msgr;
}
void AutoTextControlFilter::SendMessage(BMessage *msg)
{
if(!msg)
return;
if(fMessenger)
fMessenger->SendMessage(msg);
else
delete msg;
}
void AutoTextControlFilter::HandleNoNumLock(const int32 &code, int32 &rawchar,
BMessage *msg)
{
switch(code)
{
case 101: // Numlock .
{
rawchar = '.';
break;
}
case 100: // Numlock 0
{
rawchar = '0';
break;
}
case 88:
{
rawchar = '1';
break;
}
case 89:
{
rawchar = '2';
break;
}
case 90:
{
rawchar = '3';
break;
}
case 72:
{
rawchar = '4';
break;
}
case 74:
{
rawchar = '6';
break;
}
case 55:
{
rawchar = '7';
break;
}
case 56:
{
rawchar = '8';
break;
}
case 57:
{
rawchar = '9';
break;
}
default:
return;
}
msg->ReplaceInt32("raw_char",rawchar);
BString string;
string << (char)rawchar;
msg->ReplaceString("bytes",string);
msg->ReplaceInt8("byte",(int8)rawchar);
int32 mod;
if(msg->FindInt32("modifiers",&mod)==B_OK)
{
uint32 mask = B_NUM_LOCK;
if(mod & B_CAPS_LOCK)
mask |= B_CAPS_LOCK;
if(mod & B_SCROLL_LOCK)
mask |= B_SCROLL_LOCK;
set_keyboard_locks(mask);
}
}
AutoTextControl::AutoTextControl(const BRect &frame, const char *name, const char *label,
const char *text, BMessage *msg, uint32 resize, uint32 flags)
: BTextControl(frame,name,label,text,msg,resize,flags),
fFilter(NULL),
fEscapeCancel(false),
fCharLimit(0)
{
SetFilter(new AutoTextControlFilter(this));
}
AutoTextControl::~AutoTextControl(void)
{
if(Window())
Window()->RemoveCommonFilter(fFilter);
delete fFilter;
}
void AutoTextControl::AttachedToWindow(void)
{
BTextControl::AttachedToWindow();
if(fFilter)
Window()->AddCommonFilter(fFilter);
}
void AutoTextControl::DetachedFromWindow(void)
{
if(fFilter)
Window()->RemoveCommonFilter(fFilter);
}
void AutoTextControl::SetCharacterLimit(const uint32 &limit)
{
fCharLimit = limit;
}
uint32 AutoTextControl::GetCharacterLimit(const uint32 &limit)
{
return fCharLimit;
}
void AutoTextControl::SetFilter(AutoTextControlFilter *filter)
{
if(fFilter)
{
if(Window())
Window()->RemoveCommonFilter(fFilter);
delete fFilter;
}
fFilter = filter;
if(Window())
Window()->AddCommonFilter(fFilter);
}

View File

@ -0,0 +1,75 @@
/*
* Copyright (c) 2006, Haiku, Inc.
* Distributed under the terms of the MIT license.
*
* Author:
* DarkWyrm <darkwyrm@earthlink.net>
*/
#ifndef AUTOTEXT_H
#define AUTOTEXT_H
#include <TextControl.h>
#include <MessageFilter.h>
#include <Messenger.h>
class AutoTextControlFilter;
enum
{
M_PREVIOUS_FIELD='mprf',
M_NEXT_FIELD='mnxf',
M_ENTER_NAVIGATION='ennv'
};
class AutoTextControl : public BTextControl
{
public:
AutoTextControl(const BRect &frame, const char *name, const char *label,
const char *text, BMessage *msg,
uint32 resize = B_FOLLOW_LEFT | B_FOLLOW_TOP,
uint32 flags = B_WILL_DRAW | B_NAVIGABLE);
virtual ~AutoTextControl(void);
virtual void AttachedToWindow(void);
virtual void DetachedFromWindow(void);
void SetFilter(AutoTextControlFilter *filter);
AutoTextControlFilter *GetFilter(void) { return fFilter; }
void SetCharacterLimit(const uint32 &limit);
uint32 GetCharacterLimit(const uint32 &limit);
void SetEscapeCancel(const bool &value) { fEscapeCancel = value; }
bool IsEscapeCancel(void) const { return fEscapeCancel; }
private:
friend AutoTextControlFilter;
AutoTextControlFilter *fFilter;
bool fEscapeCancel;
uint32 fCharLimit;
};
class AutoTextControlFilter : public BMessageFilter
{
public:
AutoTextControlFilter(AutoTextControl *checkview);
~AutoTextControlFilter(void);
virtual filter_result Filter(BMessage *msg, BHandler **target);
virtual filter_result KeyFilter(const int32 &key, const int32 &mod);
AutoTextControl *TextControl(void) const { return fBox; }
void SetMessenger(BMessenger *msgr);
BMessenger *GetMessenger(void) const { return fMessenger; }
void SendMessage(BMessage *msg);
BMessage *GetCurrentMessage(void) { return fCurrentMessage; }
protected:
bool IsEscapeCancel(void) const { return fBox->IsEscapeCancel(); }
private:
void HandleNoNumLock(const int32 &code, int32 &rawchar, BMessage *msg);
AutoTextControl *fBox;
BMessenger *fMessenger;
BMessage *fCurrentMessage;
};
#endif