Add a scientific mode to Deskcalc.

Deskcalc already contains support for all the functions in scientific mode
but up until now you had to know what they were called and type them in to
figure them out. Scientific mode gives you access to most of the available
functions via buttons.

Pushing one of the the scientific mode buttons inserts the function name
along with an innertube () at the current cursor location. If you have some
text highlighted when you push a scientific mode button it will put that
text inside the innertube. So you can type 0.5, then highlight the text with
the mouse, and then push the sin button and you will get sin(0.5).

The contextual menu has been altered to support the new mode.
Instead of having a single show keypad option in the contextual menu there
are 3 new options instead. Compact mode, Basic mode, and Scientific mode.
Basic mode is the default mode showing the basic keypad. Compact mode is the
same as show keypad turned off, showing just a bare text field. Scientific
mode is the new mode which adds buttons for the different transcendental
functions and constants that Deskcalc supports. You can also use Alt+0, Alt+1,
and Alt+2 keyboard modifiers to switch between the modes.

In addition to accepting the word 'pi' for the circumference of the unit
circle, Deskcalc now also recognizes the UTF-8 character π which has a
dedicated button in scientific mode. I also changed the parser so that
lowercase 'e' always means Euler's number and uppercase 'E' always means
'times 10 to the' so 1E5 means 1 times 10 to the 5th.

Another small tweak I did was to adjust the minimum basic mode width so that
the window is flush with the tab.

I also renamed fColums to fColumns, took out some spaces and other style
changes and bumped the version to 2.2.0.

git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@43199 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
John Scipione 2011-11-06 09:38:39 +00:00
parent c80c50772e
commit 60ba75c5ec
10 changed files with 354 additions and 124 deletions

View File

@ -1,5 +1,5 @@
/*
* Copyright 2006 Haiku, Inc. All Rights Reserved.
* Copyright 2006-2011 Haiku, Inc. All Rights Reserved.
* Copyright 1997, 1998 R3 Software Ltd. All Rights Reserved.
* Distributed under the terms of the MIT License.
*

View File

@ -20,7 +20,7 @@ CalcOptions::CalcOptions()
:
auto_num_lock(false),
audio_feedback(false),
show_keypad(true)
keypad_mode(KEYPAD_MODE_BASIC)
{
}
@ -29,6 +29,7 @@ void
CalcOptions::LoadSettings(const BMessage* archive)
{
bool option;
uint8 keypad_mode_option;
if (archive->FindBool("auto num lock", &option) == B_OK)
auto_num_lock = option;
@ -36,8 +37,8 @@ CalcOptions::LoadSettings(const BMessage* archive)
if (archive->FindBool("audio feedback", &option) == B_OK)
audio_feedback = option;
if (archive->FindBool("show keypad", &option) == B_OK)
show_keypad = option;
if (archive->FindUInt8("keypad mode", &keypad_mode_option) == B_OK)
keypad_mode = keypad_mode_option;
}
@ -50,7 +51,7 @@ CalcOptions::SaveSettings(BMessage* archive) const
ret = archive->AddBool("audio feedback", audio_feedback);
if (ret == B_OK)
ret = archive->AddBool("show keypad", show_keypad);
ret = archive->AddUInt8("keypad mode", keypad_mode);
return ret;
}

View File

@ -13,12 +13,18 @@
#include <SupportDefs.h>
enum {
KEYPAD_MODE_COMPACT,
KEYPAD_MODE_BASIC,
KEYPAD_MODE_SCIENTIFIC
};
class BMessage;
struct CalcOptions {
bool auto_num_lock; // automatically activate numlock
bool audio_feedback; // provide audio feedback
bool show_keypad; // show or hide the buttons
uint8 keypad_mode; // keypad mode options
CalcOptions();

View File

@ -1,5 +1,5 @@
/*
* Copyright 2006-2009, Haiku, Inc. All rights reserved.
* Copyright 2006-2011, Haiku, Inc. All rights reserved.
* Copyright 1997, 1998 R3 Software Ltd. All Rights Reserved.
* Distributed under the terms of the MIT License.
*
@ -54,21 +54,36 @@ const float kDisplayScaleY = 0.2f;
static const bigtime_t kFlashOnOffInterval = 100000;
enum {
MSG_OPTIONS_AUTO_NUM_LOCK = 'opan',
MSG_OPTIONS_AUDIO_FEEDBACK = 'opaf',
MSG_OPTIONS_SHOW_KEYPAD = 'opsk',
static const float kMinimumWidthCompact = 130.0f;
static const float kMaximumWidthCompact = 400.0f;
static const float kMinimumHeightCompact = 20.0f;
static const float kMaximumHeightCompact = 60.0f;
MSG_UNFLASH_KEY = 'uflk'
};
// Basic mode size limits are defined in CalcView.h so
// that they can be used by the CalcWindow constructor.
// default calculator key pad layout
const char *kDefaultKeypadDescription =
static const float kMinimumWidthScientific = 240.0f;
static const float kMaximumWidthScientific = 400.0f;
static const float kMinimumHeightScientific = 200.0f;
static const float kMaximumHeightScientific = 400.0f;
// basic mode keypad layout (default)
const char *kKeypadDescriptionBasic =
"7 8 9 ( ) \n"
"4 5 6 * / \n"
"1 2 3 + - \n"
"0 . BS = C \n";
// scientific mode keypad layout
const char *kKeypadDescriptionScientific =
"ln sin cos tan π \n"
"log asin acos atan sqrt \n"
"exp sinh cosh tanh cbrt \n"
"! ceil floor E ^ \n"
"7 8 9 ( ) \n"
"4 5 6 * / \n"
"1 2 3 + - \n"
"0 . BS = C \n";
enum {
FLAGS_FLASH_KEY = 1 << 0,
@ -98,7 +113,7 @@ CalcView::CalcView(BRect frame, rgb_color rgbBaseColor, BMessage* settings)
:
BView(frame, "DeskCalc", B_FOLLOW_ALL_SIDES,
B_WILL_DRAW | B_FULL_UPDATE_ON_RESIZE | B_FRAME_EVENTS),
fColums(5),
fColumns(5),
fRows(4),
fBaseColor(rgbBaseColor),
@ -107,7 +122,7 @@ CalcView::CalcView(BRect frame, rgb_color rgbBaseColor, BMessage* settings)
fWidth(1),
fHeight(1),
fKeypadDescription(strdup(kDefaultKeypadDescription)),
fKeypadDescription(strdup(kKeypadDescriptionBasic)),
fKeypad(NULL),
#ifdef __HAIKU__
@ -119,14 +134,13 @@ CalcView::CalcView(BRect frame, rgb_color rgbBaseColor, BMessage* settings)
fPopUpMenu(NULL),
fAutoNumlockItem(NULL),
fAudioFeedbackItem(NULL),
fShowKeypadItem(NULL),
fOptions(new CalcOptions()),
fShowKeypad(true)
fOptions(new CalcOptions())
{
// create expression text view
fExpressionTextView = new ExpressionTextView(_ExpressionRect(), this);
AddChild(fExpressionTextView);
// read data from archive
_LoadSettings(settings);
// tell the app server not to erase our b/g
@ -141,6 +155,7 @@ CalcView::CalcView(BRect frame, rgb_color rgbBaseColor, BMessage* settings)
// create pop-up menu system
_CreatePopUpMenu();
// Fetch the calc icon for compact view
_FetchAppIcon(fCalcIcon);
}
@ -148,7 +163,7 @@ CalcView::CalcView(BRect frame, rgb_color rgbBaseColor, BMessage* settings)
CalcView::CalcView(BMessage* archive)
:
BView(archive),
fColums(5),
fColumns(5),
fRows(4),
fBaseColor((rgb_color){ 128, 128, 128, 255 }),
@ -157,7 +172,7 @@ CalcView::CalcView(BMessage* archive)
fWidth(1),
fHeight(1),
fKeypadDescription(strdup(kDefaultKeypadDescription)),
fKeypadDescription(strdup(kKeypadDescriptionBasic)),
fKeypad(NULL),
#ifdef __HAIKU__
@ -169,9 +184,7 @@ CalcView::CalcView(BMessage* archive)
fPopUpMenu(NULL),
fAutoNumlockItem(NULL),
fAudioFeedbackItem(NULL),
fShowKeypadItem(NULL),
fOptions(new CalcOptions()),
fShowKeypad(true)
fOptions(new CalcOptions())
{
// Do not restore the follow mode, in shelfs, we never follow.
SetResizingMode(B_FOLLOW_NONE);
@ -182,12 +195,11 @@ CalcView::CalcView(BMessage* archive)
// read data from archive
_LoadSettings(archive);
// replicant means size and window limit dont need changes
fShowKeypad = fOptions->show_keypad;
// create pop-up menu system
_CreatePopUpMenu();
// Fetch the calc icon for compact view
_FetchAppIcon(fCalcIcon);
}
@ -209,8 +221,7 @@ CalcView::AttachedToWindow()
BRect frame(Frame());
FrameResized(frame.Width(), frame.Height());
fPopUpMenu->SetTargetForItems(this);
_ShowKeypad(fOptions->show_keypad);
SetKeypadMode(fOptions->keypad_mode);
}
@ -254,22 +265,6 @@ CalcView::MessageReceived(BMessage* message)
AboutRequested();
break;
case MSG_OPTIONS_AUTO_NUM_LOCK:
fOptions->auto_num_lock = !fOptions->auto_num_lock;
fAutoNumlockItem->SetMarked(fOptions->auto_num_lock);
break;
case MSG_OPTIONS_AUDIO_FEEDBACK:
fOptions->audio_feedback = !fOptions->audio_feedback;
fAudioFeedbackItem->SetMarked(fOptions->audio_feedback);
break;
case MSG_OPTIONS_SHOW_KEYPAD:
fOptions->show_keypad = !fOptions->show_keypad;
fShowKeypadItem->SetMarked(fOptions->show_keypad);
_ShowKeypad(fOptions->show_keypad);
break;
case MSG_UNFLASH_KEY:
{
int32 key;
@ -299,7 +294,7 @@ CalcView::Draw(BRect updateRect)
SetHighColor(fBaseColor);
BRect expressionRect(_ExpressionRect());
if (updateRect.Intersects(expressionRect)) {
if (!fShowKeypad
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT
&& expressionRect.Height() >= fCalcIcon->Bounds().Height()) {
// render calc icon
expressionRect.left = fExpressionTextView->Frame().right + 2;
@ -328,7 +323,7 @@ CalcView::Draw(BRect updateRect)
// render border around expression text view
expressionRect = fExpressionTextView->Frame();
expressionRect.InsetBy(-2, -2);
if (fShowKeypad && drawBackground) {
if (fOptions->keypad_mode != KEYPAD_MODE_COMPACT && drawBackground) {
expressionRect.InsetBy(-2, -2);
StrokeRect(expressionRect);
expressionRect.InsetBy(1, 1);
@ -379,7 +374,7 @@ CalcView::Draw(BRect updateRect)
}
}
if (!fShowKeypad)
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT)
return;
// calculate grid sizes
@ -392,7 +387,7 @@ CalcView::Draw(BRect updateRect)
}
float sizeDisp = keypadRect.top;
float sizeCol = (keypadRect.Width() + 1) / (float)fColums;
float sizeCol = (keypadRect.Width() + 1) / (float)fColumns;
float sizeRow = (keypadRect.Height() + 1) / (float)fRows;
if (!updateRect.Intersects(keypadRect))
@ -403,7 +398,7 @@ CalcView::Draw(BRect updateRect)
if (be_control_look != NULL) {
CalcKey* key = fKeypad;
for (int row = 0; row < fRows; row++) {
for (int col = 0; col < fColums; col++) {
for (int col = 0; col < fColumns; col++) {
BRect frame;
frame.left = keypadRect.left + col * sizeCol;
frame.right = keypadRect.left + (col + 1) * sizeCol - 1;
@ -445,13 +440,13 @@ CalcView::Draw(BRect updateRect)
FillRect(updateRect & keypadRect);
// render key main grid
BeginLineArray(((fColums + fRows) << 1) + 1);
BeginLineArray(((fColumns + fRows) << 1) + 1);
// render cols
AddLine(BPoint(0.0, sizeDisp),
BPoint(0.0, fHeight),
fLightColor);
for (int col = 1; col < fColums; col++) {
for (int col = 1; col < fColumns; col++) {
AddLine(BPoint(col * sizeCol - 1.0, sizeDisp),
BPoint(col * sizeCol - 1.0, fHeight),
fDarkColor);
@ -459,8 +454,8 @@ CalcView::Draw(BRect updateRect)
BPoint(col * sizeCol, fHeight),
fLightColor);
}
AddLine(BPoint(fColums * sizeCol, sizeDisp),
BPoint(fColums * sizeCol, fHeight),
AddLine(BPoint(fColumns * sizeCol, sizeDisp),
BPoint(fColumns * sizeCol, fHeight),
fDarkColor);
// render rows
@ -489,7 +484,7 @@ CalcView::Draw(BRect updateRect)
* (1.0 - kFontScaleY) * 0.5;
CalcKey* key = fKeypad;
for (int row = 0; row < fRows; row++) {
for (int col = 0; col < fColums; col++) {
for (int col = 0; col < fColumns; col++) {
float halfSymbolWidth = StringWidth(key->label) * 0.5f;
DrawString(key->label,
BPoint(col * sizeCol + halfSizeCol - halfSymbolWidth,
@ -524,7 +519,7 @@ CalcView::MouseDown(BPoint point)
return;
}
if (!fShowKeypad) {
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT) {
if (fCalcIcon != NULL) {
BRect bounds(Bounds());
bounds.left = bounds.right - fCalcIcon->Bounds().Width();
@ -538,7 +533,7 @@ CalcView::MouseDown(BPoint point)
// calculate grid sizes
float sizeDisp = fHeight * kDisplayScaleY;
float sizeCol = fWidth / (float)fColums;
float sizeCol = fWidth / (float)fColumns;
float sizeRow = (fHeight - sizeDisp) / (float)fRows;
// calculate location within grid
@ -546,11 +541,11 @@ CalcView::MouseDown(BPoint point)
int gridRow = (int)floorf((point.y - sizeDisp) / sizeRow);
// check limits
if ((gridCol >= 0) && (gridCol < fColums)
if ((gridCol >= 0) && (gridCol < fColumns)
&& (gridRow >= 0) && (gridRow < fRows)) {
// process key press
int key = gridRow * fColums + gridCol;
int key = gridRow * fColumns + gridCol;
_FlashKey(key, FLAGS_MOUSE_DOWN);
_PressKey(key);
@ -563,10 +558,10 @@ CalcView::MouseDown(BPoint point)
void
CalcView::MouseUp(BPoint point)
{
if (!fShowKeypad)
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT)
return;
int keys = fRows * fColums;
int keys = fRows * fColumns;
for (int i = 0; i < keys; i++) {
if (fKeypad[i].flags & FLAGS_MOUSE_DOWN) {
_FlashKey(i, 0);
@ -617,7 +612,7 @@ CalcView::KeyDown(const char* bytes, int32 numBytes)
default: {
// scan the keymap array for match
int keys = fRows * fColums;
int keys = fRows * fColumns;
for (int i = 0; i < keys; i++) {
if (fKeypad[i].keymap[0] == bytes[0]) {
_PressKey(i);
@ -647,6 +642,14 @@ CalcView::MakeFocus(bool focused)
}
void
CalcView::ResizeTo(float width, float height)
{
BView::ResizeTo(width, height);
FrameResized(width, height);
}
void
CalcView::FrameResized(float width, float height)
{
@ -655,19 +658,18 @@ CalcView::FrameResized(float width, float height)
// layout expression text view
BRect frame = _ExpressionRect();
if (fShowKeypad)
frame.InsetBy(4, 4);
else
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT) {
frame.InsetBy(2, 2);
if (!fShowKeypad)
frame.right -= ceilf(fCalcIcon->Bounds().Width() * 1.5);
} else
frame.InsetBy(4, 4);
fExpressionTextView->MoveTo(frame.LeftTop());
fExpressionTextView->ResizeTo(frame.Width(), frame.Height());
// configure expression text view font size and color
float sizeDisp = fShowKeypad ? fHeight * kDisplayScaleY : fHeight;
float sizeDisp = fOptions->keypad_mode == KEYPAD_MODE_COMPACT
? fHeight : fHeight * kDisplayScaleY;
BFont font(be_bold_font);
font.SetSize(sizeDisp * kExpressionFontScaleY);
fExpressionTextView->SetFontAndColor(&font, B_FONT_ALL);
@ -797,15 +799,15 @@ CalcView::_LoadSettings(BMessage* archive)
// record calculator description
const char* calcDesc;
if (archive->FindString("calcDesc", &calcDesc) < B_OK)
calcDesc = kDefaultKeypadDescription;
calcDesc = kKeypadDescriptionBasic;
// save calculator description for reference
free(fKeypadDescription);
fKeypadDescription = strdup(calcDesc);
// read grid dimensions
if (archive->FindInt16("cols", &fColums) < B_OK)
fColums = 5;
if (archive->FindInt16("cols", &fColumns) < B_OK)
fColumns = 5;
if (archive->FindInt16("rows", &fRows) < B_OK)
fRows = 4;
@ -862,7 +864,7 @@ CalcView::SaveSettings(BMessage* archive) const
// record grid dimensions
if (ret == B_OK)
ret = archive->AddInt16("cols", fColums);
ret = archive->AddInt16("cols", fColumns);
if (ret == B_OK)
ret = archive->AddInt16("rows", fRows);
@ -935,6 +937,97 @@ CalcView::FlashKey(const char* bytes, int32 numBytes)
}
void
CalcView::ToggleAutoNumlock(void)
{
fOptions->auto_num_lock = !fOptions->auto_num_lock;
fAutoNumlockItem->SetMarked(fOptions->auto_num_lock);
}
void
CalcView::ToggleAudioFeedback(void)
{
fOptions->audio_feedback = !fOptions->audio_feedback;
fAudioFeedbackItem->SetMarked(fOptions->audio_feedback);
}
void
CalcView::SetKeypadMode(uint8 mode)
{
if (fOptions->keypad_mode == mode)
return;
BWindow* window = Window();
if (window == NULL)
return;
fOptions->keypad_mode = mode;
_MarkKeypadItems(fOptions->keypad_mode);
float width = fWidth;
float height = fHeight;
switch (fOptions->keypad_mode) {
case KEYPAD_MODE_COMPACT:
{
if (window->Bounds() == Frame()) {
window->SetSizeLimits(kMinimumWidthCompact,
kMaximumWidthCompact,
kMinimumHeightCompact,
kMaximumHeightCompact);
window->ResizeTo(width, height * kDisplayScaleY);
} else
ResizeTo(width, height * kDisplayScaleY);
break;
}
case KEYPAD_MODE_SCIENTIFIC:
{
free(fKeypadDescription);
fKeypadDescription = strdup(kKeypadDescriptionScientific);
fRows = 8;
_ParseCalcDesc(fKeypadDescription);
window->SetSizeLimits(kMinimumWidthScientific,
kMaximumWidthScientific,
kMinimumHeightScientific,
kMaximumHeightScientific);
if (width < kMinimumWidthScientific)
width = kMinimumWidthScientific;
if (width > kMaximumWidthScientific)
width = kMaximumWidthScientific;
if (height < kMinimumHeightScientific)
height = kMinimumHeightScientific;
if (height > kMaximumHeightScientific)
height = kMaximumHeightScientific;
ResizeTo(width, height);
break;
}
default: // KEYPAD_MODE_BASIC is the default
{
free(fKeypadDescription);
fKeypadDescription = strdup(kKeypadDescriptionBasic);
fRows = 4;
_ParseCalcDesc(fKeypadDescription);
window->SetSizeLimits(kMinimumWidthBasic, kMaximumWidthBasic,
kMinimumHeightBasic, kMaximumHeightBasic);
if (width < kMinimumWidthBasic)
width = kMinimumWidthBasic;
if (width > kMaximumWidthBasic)
width = kMaximumWidthBasic;
if (height < kMinimumHeightBasic)
height = kMinimumHeightBasic;
if (height > kMaximumHeightBasic)
height = kMaximumHeightBasic;
ResizeTo(width, height);
}
}
}
// #pragma mark -
@ -942,7 +1035,7 @@ void
CalcView::_ParseCalcDesc(const char* keypadDescription)
{
// TODO: should calculate dimensions from desc here!
fKeypad = new CalcKey[fRows * fColums];
fKeypad = new CalcKey[fRows * fColumns];
// scan through calculator description and assemble keypad
CalcKey* key = fKeypad;
@ -984,15 +1077,56 @@ CalcView::_ParseCalcDesc(const char* keypadDescription)
void
CalcView::_PressKey(int key)
{
assert(key < (fRows * fColums));
assert(key < (fRows * fColumns));
assert(key >= 0);
// check for backspace
if (strcmp(fKeypad[key].label, "BS") == 0) {
// BS means backspace
fExpressionTextView->BackSpace();
} else if (strcmp(fKeypad[key].label, "C") == 0) {
// C means clear
fExpressionTextView->Clear();
} else if (strcmp(fKeypad[key].label, "acos") == 0
|| strcmp(fKeypad[key].label, "asin") == 0
|| strcmp(fKeypad[key].label, "atan") == 0
|| strcmp(fKeypad[key].label, "cbrt") == 0
|| strcmp(fKeypad[key].label, "ceil") == 0
|| strcmp(fKeypad[key].label, "cos") == 0
|| strcmp(fKeypad[key].label, "cosh") == 0
|| strcmp(fKeypad[key].label, "exp") == 0
|| strcmp(fKeypad[key].label, "floor") == 0
|| strcmp(fKeypad[key].label, "log") == 0
|| strcmp(fKeypad[key].label, "ln") == 0
|| strcmp(fKeypad[key].label, "sin") == 0
|| strcmp(fKeypad[key].label, "sinh") == 0
|| strcmp(fKeypad[key].label, "sqrt") == 0
|| strcmp(fKeypad[key].label, "tan") == 0
|| strcmp(fKeypad[key].label, "tanh") == 0) {
int32 labelLen = strlen(fKeypad[key].label);
int32 startSelection = 0;
int32 endSelection = 0;
fExpressionTextView->GetSelection(&startSelection, &endSelection);
if (endSelection > startSelection) {
// There is selected text, put it inbetween the parens
fExpressionTextView->Insert(startSelection, fKeypad[key].label,
labelLen);
fExpressionTextView->Insert(startSelection + labelLen, "(", 1);
fExpressionTextView->Insert(endSelection + labelLen + 1, ")", 1);
// Put the cursor after the ending paren
// Need to cast to BTextView because Select() is protected
// in the InputTextView class
static_cast<BTextView*>(fExpressionTextView)->Select(
endSelection + labelLen + 2, endSelection + labelLen + 2);
} else {
// There is no selected text, insert at the cursor location
fExpressionTextView->Insert(fKeypad[key].label);
fExpressionTextView->Insert("()");
// Put the cursor inside the parens so you can enter an argument
// Need to cast to BTextView because Select() is protected
// in the InputTextView class
static_cast<BTextView*>(fExpressionTextView)->Select(
endSelection + labelLen + 1, endSelection + labelLen + 1);
}
} else {
// check for evaluation order
if (fKeypad[key].code[0] == '\n') {
@ -1019,7 +1153,7 @@ CalcView::_PressKey(const char* label)
int32
CalcView::_KeyForLabel(const char* label) const
{
int keys = fRows * fColums;
int keys = fRows * fColumns;
for (int i = 0; i < keys; i++) {
if (strcmp(fKeypad[i].label, label) == 0) {
return i;
@ -1032,7 +1166,7 @@ CalcView::_KeyForLabel(const char* label) const
void
CalcView::_FlashKey(int32 key, uint32 flashFlags)
{
if (!fShowKeypad)
if (fOptions->keypad_mode == KEYPAD_MODE_COMPACT)
return;
if (flashFlags != 0)
@ -1105,13 +1239,17 @@ CalcView::_CreatePopUpMenu()
new BMessage(MSG_OPTIONS_AUTO_NUM_LOCK));
fAudioFeedbackItem = new BMenuItem(B_TRANSLATE("Audio Feedback"),
new BMessage(MSG_OPTIONS_AUDIO_FEEDBACK));
fShowKeypadItem = new BMenuItem(B_TRANSLATE("Show keypad"),
new BMessage(MSG_OPTIONS_SHOW_KEYPAD));
fKeypadModeCompactItem = new BMenuItem(B_TRANSLATE("Compact"),
new BMessage(MSG_OPTIONS_KEYPAD_MODE_COMPACT), '0');
fKeypadModeBasicItem = new BMenuItem(B_TRANSLATE("Basic"),
new BMessage(MSG_OPTIONS_KEYPAD_MODE_BASIC), '1');
fKeypadModeScientificItem = new BMenuItem(B_TRANSLATE("Scientific"),
new BMessage(MSG_OPTIONS_KEYPAD_MODE_SCIENTIFIC), '2');
// apply current settings
fAutoNumlockItem->SetMarked(fOptions->auto_num_lock);
fAudioFeedbackItem->SetMarked(fOptions->audio_feedback);
fShowKeypadItem->SetMarked(fOptions->show_keypad);
_MarkKeypadItems(fOptions->keypad_mode);
// construct menu
fPopUpMenu = new BPopUpMenu("pop-up", false, false);
@ -1120,7 +1258,10 @@ CalcView::_CreatePopUpMenu()
// TODO: Enabled when we use beep events which can be configured in the Sounds
// preflet.
// fPopUpMenu->AddItem(fAudioFeedbackItem);
fPopUpMenu->AddItem(fShowKeypadItem);
fPopUpMenu->AddSeparatorItem();
fPopUpMenu->AddItem(fKeypadModeCompactItem);
fPopUpMenu->AddItem(fKeypadModeBasicItem);
fPopUpMenu->AddItem(fKeypadModeScientificItem);
}
@ -1128,7 +1269,7 @@ BRect
CalcView::_ExpressionRect() const
{
BRect r(0.0, 0.0, fWidth, fHeight);
if (fShowKeypad) {
if (fOptions->keypad_mode != KEYPAD_MODE_COMPACT) {
r.bottom = floorf(fHeight * kDisplayScaleY) + 1;
}
return r;
@ -1139,7 +1280,7 @@ BRect
CalcView::_KeypadRect() const
{
BRect r(0.0, 0.0, -1.0, -1.0);
if (fShowKeypad) {
if (fOptions->keypad_mode != KEYPAD_MODE_COMPACT) {
r.right = fWidth;
r.bottom = fHeight;
r.top = floorf(fHeight * kDisplayScaleY);
@ -1149,26 +1290,25 @@ CalcView::_KeypadRect() const
void
CalcView::_ShowKeypad(bool show)
CalcView::_MarkKeypadItems(uint8 keypad_mode)
{
if (fShowKeypad == show)
return;
switch (keypad_mode) {
case KEYPAD_MODE_COMPACT:
fKeypadModeCompactItem->SetMarked(true);
fKeypadModeBasicItem->SetMarked(false);
fKeypadModeScientificItem->SetMarked(false);
break;
fShowKeypad = show;
if (fShowKeypadItem && fShowKeypadItem->IsMarked() ^ fShowKeypad)
fShowKeypadItem->SetMarked(fShowKeypad);
case KEYPAD_MODE_SCIENTIFIC:
fKeypadModeCompactItem->SetMarked(false);
fKeypadModeBasicItem->SetMarked(false);
fKeypadModeScientificItem->SetMarked(true);
break;
float height
= fShowKeypad ? fHeight / kDisplayScaleY : fHeight * kDisplayScaleY;
BWindow* window = Window();
if (window) {
if (window->Bounds() == Frame()) {
window->SetSizeLimits(100.0, 400.0,
fShowKeypad ? 100.0 : 20.0, fShowKeypad ? 400.0 : 60.0);
window->ResizeTo(fWidth, height);
} else
ResizeTo(fWidth, height);
default: // KEYPAD_MODE_BASIC is the default
fKeypadModeCompactItem->SetMarked(false);
fKeypadModeBasicItem->SetMarked(true);
fKeypadModeScientificItem->SetMarked(false);
}
}

View File

@ -13,6 +13,20 @@
#include <View.h>
enum {
MSG_OPTIONS_AUTO_NUM_LOCK = 'oanl',
MSG_OPTIONS_AUDIO_FEEDBACK = 'oafb',
MSG_OPTIONS_KEYPAD_MODE_COMPACT = 'okmc',
MSG_OPTIONS_KEYPAD_MODE_BASIC = 'okmb',
MSG_OPTIONS_KEYPAD_MODE_SCIENTIFIC = 'okms',
MSG_UNFLASH_KEY = 'uflk'
};
static const float kMinimumWidthBasic = 130.0f;
static const float kMaximumWidthBasic = 400.0f;
static const float kMinimumHeightBasic = 130.0f;
static const float kMaximumHeightBasic = 400.0f;
class BString;
class BMenuItem;
class BPopUpMenu;
@ -42,6 +56,7 @@ class CalcView : public BView {
virtual void MouseUp(BPoint point);
virtual void KeyDown(const char* bytes, int32 numBytes);
virtual void MakeFocus(bool focused = true);
virtual void ResizeTo(float width, float height);
virtual void FrameResized(float width, float height);
// Present about box for view (replicant).
@ -62,10 +77,22 @@ class CalcView : public BView {
// Save current settings
status_t SaveSettings(BMessage* archive) const;
// Evaluate the expression
void Evaluate();
// Flash the key on the keypad
void FlashKey(const char* bytes, int32 numBytes);
// Toggle whether or not the Num Lock key starts on
void ToggleAutoNumlock(void);
// Toggle whether or not to provide audio feedback
// (option currently disabled)
void ToggleAudioFeedback(void);
// Set the keypad mode
void SetKeypadMode(uint8 mode);
private:
void _ParseCalcDesc(const char* keypadDescription);
@ -82,13 +109,14 @@ class CalcView : public BView {
BRect _ExpressionRect() const;
BRect _KeypadRect() const;
void _ShowKeypad(bool show);
void _MarkKeypadItems(uint8 mode);
void _FetchAppIcon(BBitmap* into);
status_t _LoadSettings(BMessage* archive);
// grid dimensions
int16 fColums;
int16 fColumns;
int16 fRows;
// color scheme
@ -119,11 +147,12 @@ class CalcView : public BView {
BPopUpMenu* fPopUpMenu;
BMenuItem* fAutoNumlockItem;
BMenuItem* fAudioFeedbackItem;
BMenuItem* fShowKeypadItem;
BMenuItem* fKeypadModeCompactItem;
BMenuItem* fKeypadModeBasicItem;
BMenuItem* fKeypadModeScientificItem;
// calculator options.
CalcOptions* fOptions;
bool fShowKeypad;
};
#endif // _CALC_VIEW_H

View File

@ -19,6 +19,7 @@
#include <Dragger.h>
#include <Screen.h>
#include "CalcOptions.h"
#include "CalcView.h"
@ -36,7 +37,9 @@ CalcWindow::CalcWindow(BRect frame, BMessage* settings)
BScreen screen(this);
rgb_color baseColor = screen.DesktopColor();
SetSizeLimits(100.0, 400.0, 100.0, 400.0);
// Size Limits are defined in CalcView.h
SetSizeLimits(kMinimumWidthBasic, kMaximumWidthBasic,
kMinimumHeightBasic, kMaximumHeightBasic);
frame.OffsetTo(B_ORIGIN);
fCalcView = new CalcView(frame, baseColor, settings);
@ -57,6 +60,14 @@ CalcWindow::CalcWindow(BRect frame, BMessage* settings)
SetFrame(rect);
else
SetFrame(frame, true);
// Add shortcut keys to menu options
AddShortcut('0', B_COMMAND_KEY,
new BMessage(MSG_OPTIONS_KEYPAD_MODE_COMPACT));
AddShortcut('1', B_COMMAND_KEY,
new BMessage(MSG_OPTIONS_KEYPAD_MODE_BASIC));
AddShortcut('2', B_COMMAND_KEY,
new BMessage(MSG_OPTIONS_KEYPAD_MODE_SCIENTIFIC));
}
@ -65,6 +76,37 @@ CalcWindow::~CalcWindow()
}
void
CalcWindow::MessageReceived(BMessage* msg)
{
switch (msg->what) {
case MSG_OPTIONS_AUTO_NUM_LOCK:
fCalcView->ToggleAutoNumlock();
break;
case MSG_OPTIONS_AUDIO_FEEDBACK:
fCalcView->ToggleAudioFeedback();
break;
case MSG_OPTIONS_KEYPAD_MODE_COMPACT:
fCalcView->SetKeypadMode(KEYPAD_MODE_COMPACT);
break;
case MSG_OPTIONS_KEYPAD_MODE_BASIC:
fCalcView->SetKeypadMode(KEYPAD_MODE_BASIC);
break;
case MSG_OPTIONS_KEYPAD_MODE_SCIENTIFIC:
fCalcView->SetKeypadMode(KEYPAD_MODE_SCIENTIFIC);
break;
default:
BWindow::MessageReceived(msg);
break;
}
}
void
CalcWindow::Show()
{

View File

@ -21,6 +21,7 @@ class CalcWindow : public BWindow {
CalcWindow(BRect frame, BMessage* settings);
virtual ~CalcWindow();
virtual void MessageReceived(BMessage* message);
virtual void Show();
virtual bool QuitRequested();

View File

@ -5,14 +5,14 @@ resource app_name_catalog_entry "x-vnd.Haiku-DeskCalc:System name:DeskCalc";
resource app_version {
major = 2,
middle = 1,
middle = 2,
minor = 0,
variety = B_APPV_ALPHA,
internal = 1,
short_info = "DeskCalc",
long_info = "DeskCalc ©2006-2009 Haiku, Inc."
long_info = "DeskCalc ©2006-2011 Haiku, Inc."
};
resource app_flags B_SINGLE_LAUNCH;

View File

@ -1,5 +1,5 @@
/*
* Copyright 2006 Haiku, Inc. All Rights Reserved.
* Copyright 2006-2011 Haiku, Inc. All Rights Reserved.
* Distributed under the terms of the MIT License.
*
* Authors:
@ -188,7 +188,7 @@ ExpressionTextView::SetValue(BString value)
uint32 mode = B_FONT_ALL;
GetFontAndColor(&font, &mode);
float stringWidth = font.StringWidth(value);
// make the string shorter if it does not fit in the view
float viewWidth = Frame().Width();
if (value.CountChars() > 3 && stringWidth > viewWidth) {
@ -207,20 +207,20 @@ ExpressionTextView::SetValue(BString value)
if (offset == firstDigit + 1) {
// if the value is 0.01 or larger then scientific notation
// won't shorten the string
if (value[firstDigit] != '0' || value[firstDigit+2] != '0'
if (value[firstDigit] != '0' || value[firstDigit + 2] != '0'
|| value[firstDigit + 3] != '0') {
exponent = 0;
} else {
// remove the period
value.Remove(offset, 1);
// check for negative exponent value
exponent = 0;
while (value[firstDigit] == '0') {
value.Remove(firstDigit, 1);
exponent--;
}
// add the period
value.Insert('.', 1, firstDigit + 1);
}
@ -241,12 +241,12 @@ ExpressionTextView::SetValue(BString value)
}
}
}
// add the exponent
offset = value.CountChars() - 1;
if (exponent != 0)
value << "E" << exponent;
// reduce the number of digits until the string fits or can not be
// made any shorter
stringWidth = font.StringWidth(value);
@ -257,13 +257,13 @@ ExpressionTextView::SetValue(BString value)
value.Remove(offset--, 1);
stringWidth = font.StringWidth(value);
}
// there is no need to keep the period if no digits follow
if (value[offset] == '.') {
value.Remove(offset, 1);
offset--;
}
// take care of proper rounding of the result
int digit = (int)lastRemovedDigit - '0'; // ascii to int
if (digit >= 5) {
@ -282,14 +282,14 @@ ExpressionTextView::SetValue(BString value)
value[firstDigit] = '.';
}
value.Insert('1', 1, firstDigit);
// remove the exponent value and the last digit
offset = value.FindFirst('E');
if (offset == B_ERROR)
offset = value.CountChars();
value.Truncate(--offset);
offset--; // offset now points to the last digit
// increase the exponent and add it back to the string
exponent++;
value << 'E' << exponent;
@ -298,11 +298,11 @@ ExpressionTextView::SetValue(BString value)
value[offset] = char(digit + 48);
}
}
// remove trailing zeros
while (value[offset] == '0')
value.Remove(offset--, 1);
// there is no need to keep the period if no digits follow
if (value[offset] == '.')
value.Remove(offset, 1);

View File

@ -1,5 +1,5 @@
/*
* Copyright 2006-2009 Haiku, Inc. All Rights Reserved.
* Copyright 2006-2011 Haiku, Inc. All Rights Reserved.
* Distributed under the terms of the MIT License.
*
* Authors:
@ -159,7 +159,7 @@ class Tokenizer {
}
// optional exponent part
if (*fCurrentChar == 'e' || *fCurrentChar == 'E') {
if (*fCurrentChar == 'E') {
temp << *fCurrentChar;
fCurrentChar++;
@ -206,6 +206,13 @@ class Tokenizer {
fCurrentToken = Token(begin, length, _CurrentPos() - length,
TOKEN_IDENTIFIER);
} else if ((unsigned char)fCurrentChar[0] == 0xCF
&& (unsigned char)fCurrentChar[1] == 0x80) {
// UTF-8 small greek letter PI
fCurrentToken = Token(fCurrentChar, 2, _CurrentPos() - 1,
TOKEN_IDENTIFIER);
fCurrentChar += 2;
} else {
int32 type = TOKEN_NONE;
@ -568,10 +575,14 @@ ExpressionParser::_InitArguments(MAPM values[], int32 argumentCount)
MAPM
ExpressionParser::_ParseFunction(const Token& token)
{
if (strcasecmp("e", token.string.String()) == 0)
if (strcmp("e", token.string.String()) == 0)
return _ParseFactorial(MAPM(MM_E));
else if (strcasecmp("pi", token.string.String()) == 0)
else if (strcasecmp("pi", token.string.String()) == 0
|| ((unsigned char)token.string.String()[0] == 0xCF
&& (unsigned char)token.string.String()[1] == 0x80)) {
// UTF-8 small greek letter PI
return _ParseFactorial(MAPM(MM_PI));
}
// hard coded cases for different count of arguments
// supports functions with 3 arguments at most