When you changed the refresh rate for all workspaces, then selected "cancel" the changes didn't revert... Fixed. Some more cleanups. Some code (not compiled right now, it works partially) to change the way the example screen is rendered (bitmaps instead of drawing)

git-svn-id: file:///srv/svn/repos/haiku/trunk/current@3450 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Stefano Ceccherini 2003-06-09 09:34:12 +00:00
parent c59538b901
commit 518c1e3f62
4 changed files with 205 additions and 161 deletions

View File

@ -1,8 +1,10 @@
#include <Bitmap.h>
#include <InterfaceDefs.h> #include <InterfaceDefs.h>
#include <Message.h> #include <Message.h>
#include <Roster.h> #include <Roster.h>
#include <Screen.h> #include <Screen.h>
#include <String.h> #include <String.h>
#include <TranslationUtils.h>
#include <View.h> #include <View.h>
#include <cstdlib> #include <cstdlib>
@ -22,7 +24,21 @@ ScreenDrawView::ScreenDrawView(BRect rect, char *name)
display_mode mode; display_mode mode;
screen.GetMode(&mode); screen.GetMode(&mode);
fResolution = mode.virtual_width; fWidth = mode.virtual_width;
fHeight = mode.virtual_height;
#ifdef USE_BITMAPS
fScreen1 = BTranslationUtils::GetBitmap("screen_1");
fScreen2 = BTranslationUtils::GetBitmap("screen_2");
#endif
}
ScreenDrawView::~ScreenDrawView()
{
#ifdef USE_BITMAPS
delete fScreen1;
delete fScreen2;
#endif
} }
@ -44,12 +60,13 @@ ScreenDrawView::MouseDown(BPoint point)
void void
ScreenDrawView::Draw(BRect updateRect) ScreenDrawView::Draw(BRect updateRect)
{ {
#ifndef USE_BITMAPS
rgb_color red = { 228, 0, 0, 255 }; rgb_color red = { 228, 0, 0, 255 };
rgb_color black = { 0, 0, 0, 255 }; rgb_color black = { 0, 0, 0, 255 };
rgb_color darkColor = {160, 160, 160, 255}; rgb_color darkColor = {160, 160, 160, 255};
//FIXME: Make the draw code resolution independent //FIXME: Make the draw code resolution independent
if (fResolution == 1600) if (fWidth == 1600)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -71,7 +88,7 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(4.0, 75.0), BPoint(5.0, 75.0)); StrokeLine(BPoint(4.0, 75.0), BPoint(5.0, 75.0));
} }
else if(fResolution == 1280) else if(fWidth == 1280)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -93,7 +110,7 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(15.0, 70.0), BPoint(16.0, 70.0)); StrokeLine(BPoint(15.0, 70.0), BPoint(16.0, 70.0));
} }
else if(fResolution == 1152) else if(fWidth == 1152)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -115,7 +132,7 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(19.0, 66.0), BPoint(20.0, 66.0)); StrokeLine(BPoint(19.0, 66.0), BPoint(20.0, 66.0));
} }
else if(fResolution == 1024) else if(fWidth == 1024)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -137,7 +154,7 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(22.0, 62.0), BPoint(23.0, 62.0)); StrokeLine(BPoint(22.0, 62.0), BPoint(23.0, 62.0));
} }
else if(fResolution == 800) else if(fWidth == 800)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -159,7 +176,7 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(29.0, 56.0), BPoint(30.0, 56.0)); StrokeLine(BPoint(29.0, 56.0), BPoint(30.0, 56.0));
} }
else if(fResolution == 640) else if(fWidth == 640)
{ {
SetHighColor(darkColor); SetHighColor(darkColor);
@ -181,6 +198,27 @@ ScreenDrawView::Draw(BRect updateRect)
StrokeLine(BPoint(35.0, 53.0), BPoint(36.0, 53.0)); StrokeLine(BPoint(35.0, 53.0), BPoint(36.0, 53.0));
} }
#else
SetHighColor(desktopColor);
FillRect(Bounds());
BRect bitmapBounds(fScreen1->Bounds());
BRect dest;
dest.left = Bounds().left;
dest.top = Bounds().top;
dest.right = Bounds().left + (bitmapBounds.Width() * 100) / fWidth;
dest.bottom = Bounds().top + (bitmapBounds.Height() * 100) / fHeight;
DrawBitmap(fScreen1, dest);
BRect bitmapBounds2(fScreen2->Bounds());
BRect dest2;
dest2.top = Bounds().top;
dest2.right = Bounds().right;
dest2.bottom = Bounds().top + (bitmapBounds2.Height() * 100) / fHeight;
dest2.left = Bounds().right - (bitmapBounds2.Width() * 100) / fWidth;
DrawBitmap(fScreen2, dest2);
#endif
} }
@ -194,10 +232,12 @@ ScreenDrawView::MessageReceived(BMessage* message)
BString resolution; BString resolution;
message->FindString("resolution", &resolution); message->FindString("resolution", &resolution);
int32 nextfResolution = atoi(resolution.String()); int32 nextWidth = atoi(resolution.String());
resolution.Truncate(resolution.FindFirst('x') + 1);
if (fResolution != nextfResolution) { if (fWidth != nextWidth) {
fResolution = nextfResolution; fWidth = nextWidth;
fHeight = atoi(resolution.String());
Invalidate(); Invalidate();
} }

View File

@ -7,6 +7,7 @@ class ScreenDrawView : public BView
{ {
public: public:
ScreenDrawView(BRect frame, char *name); ScreenDrawView(BRect frame, char *name);
~ScreenDrawView();
virtual void AttachedToWindow(); virtual void AttachedToWindow();
virtual void Draw(BRect updateRect); virtual void Draw(BRect updateRect);
virtual void MessageReceived(BMessage *message); virtual void MessageReceived(BMessage *message);
@ -14,7 +15,10 @@ public:
private: private:
rgb_color desktopColor; rgb_color desktopColor;
int32 fResolution; int32 fWidth;
int32 fHeight;
BBitmap *fScreen1,
*fScreen2;
}; };
#endif #endif

View File

@ -20,6 +20,33 @@
#include "AlertWindow.h" #include "AlertWindow.h"
#include "Constants.h" #include "Constants.h"
#include "Utility.h" #include "Utility.h"
static const char*
colorspace_to_string(uint32 colorspace)
{
switch (colorspace) {
case B_RGB32:
return "32 Bits/Pixel";
case B_RGB16:
return"16 Bits/Pixel";
case B_RGB15:
return "15 Bits/Pixel";
case B_CMAP8:
default:
return "8 Bits/Pixel";
}
}
static const char*
mode_to_string(display_mode mode)
{
char string[128];
sprintf(string, "%d x %d", mode.virtual_width, mode.virtual_height);
return string;
}
ScreenWindow::ScreenWindow(ScreenSettings *Settings) ScreenWindow::ScreenWindow(ScreenSettings *Settings)
: BWindow(Settings->WindowFrame(), "Screen", B_TITLED_WINDOW, B_NOT_RESIZABLE | B_NOT_ZOOMABLE, B_ALL_WORKSPACES) : BWindow(Settings->WindowFrame(), "Screen", B_TITLED_WINDOW, B_NOT_RESIZABLE | B_NOT_ZOOMABLE, B_ALL_WORKSPACES)
@ -45,24 +72,21 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
BRect ControlMenuRect; BRect ControlMenuRect;
BRect ButtonRect; BRect ButtonRect;
fScreenBox = new BBox(ScreenBoxRect); BBox *screenBox = new BBox(ScreenBoxRect);
fScreenBox->SetBorder(B_FANCY_BORDER); screenBox->SetBorder(B_FANCY_BORDER);
fScreenDrawView = new ScreenDrawView(ScreenDrawViewRect, "ScreenDrawView"); fScreenDrawView = new ScreenDrawView(ScreenDrawViewRect, "ScreenDrawView");
BString string; fWorkspaceCountMenu = new BPopUpMenu("", true, true);
string << count_workspaces();
fWorkspaceCountMenu = new BPopUpMenu(string.String(), true, true);
for (int32 count = 1; count <= 32; count++) { for (int32 count = 1; count <= 32; count++) {
string.Truncate(0); BString workspaceCount;
string << count; workspaceCount << count;
fWorkspaceCountMenu->AddItem(new BMenuItem(string.String(), new BMessage(POP_WORKSPACE_CHANGED_MSG))); fWorkspaceCountMenu->AddItem(new BMenuItem(workspaceCount.String(),
new BMessage(POP_WORKSPACE_CHANGED_MSG)));
} }
string.Truncate(0); BString string;
string << count_workspaces(); string << count_workspaces();
BMenuItem *marked = fWorkspaceCountMenu->FindItem(string.String()); BMenuItem *marked = fWorkspaceCountMenu->FindItem(string.String());
@ -74,9 +98,9 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fWorkspaceCountField->SetDivider(91.0); fWorkspaceCountField->SetDivider(91.0);
fScreenBox->AddChild(fScreenDrawView); screenBox->AddChild(fScreenDrawView);
fScreenBox->AddChild(fWorkspaceCountField); screenBox->AddChild(fWorkspaceCountField);
fScreenView->AddChild(fScreenBox); fScreenView->AddChild(screenBox);
fWorkspaceMenu = new BPopUpMenu("Current Workspace", true, true); fWorkspaceMenu = new BPopUpMenu("Current Workspace", true, true);
fAllWorkspacesItem = new BMenuItem("All Workspaces", new BMessage(WORKSPACE_CHECK_MSG)); fAllWorkspacesItem = new BMenuItem("All Workspaces", new BMessage(WORKSPACE_CHECK_MSG));
@ -91,9 +115,9 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
ControlsBoxRect.Set(164.0, 7.0, 345.0, 155.0); ControlsBoxRect.Set(164.0, 7.0, 345.0, 155.0);
fControlsBox = new BBox(ControlsBoxRect); BBox *controlsBox = new BBox(ControlsBoxRect);
fControlsBox->SetBorder(B_FANCY_BORDER); controlsBox->SetBorder(B_FANCY_BORDER);
fControlsBox->SetLabel(fWorkspaceField); controlsBox->SetLabel(fWorkspaceField);
ButtonRect.Set(88.0, 114.0, 200.0, 150.0); ButtonRect.Set(88.0, 114.0, 200.0, 150.0);
@ -104,16 +128,13 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fApplyButton->ResizeToPreferred(); fApplyButton->ResizeToPreferred();
fApplyButton->SetEnabled(false); fApplyButton->SetEnabled(false);
fControlsBox->AddChild(fApplyButton); controlsBox->AddChild(fApplyButton);
fResolutionMenu = new BPopUpMenu("", true, true);
fColorsMenu = new BPopUpMenu("", true, true);
CheckUpdateDisplayModes();
fResolutionMenu = new BPopUpMenu("", true, true);
for (uint32 c = 0; c < fTotalModes; c++) {
BString mode;
mode << (int32)fSupportedModes[c].virtual_width << " x " << (int32)fSupportedModes[c].virtual_height;
if (!fResolutionMenu->FindItem(mode.String()))
fResolutionMenu->AddItem(new BMenuItem(mode.String(),
new BMessage(POP_RESOLUTION_MSG)));
}
ControlMenuRect.Set(33.0, 30.0, 171.0, 48.0); ControlMenuRect.Set(33.0, 30.0, 171.0, 48.0);
fResolutionField = new BMenuField(ControlMenuRect, "ResolutionMenu", "Resolution:", fResolutionMenu, true); fResolutionField = new BMenuField(ControlMenuRect, "ResolutionMenu", "Resolution:", fResolutionMenu, true);
@ -123,32 +144,7 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fResolutionField->SetDivider(55.0); fResolutionField->SetDivider(55.0);
fControlsBox->AddChild(fResolutionField); controlsBox->AddChild(fResolutionField);
fColorsMenu = new BPopUpMenu("", true, true);
// Get the supported colorspaces
for (uint32 c = 0; c < fTotalModes; c++) {
BString colorSpace;
switch (fSupportedModes[c].space) {
case B_CMAP8:
colorSpace = "8 Bits/Pixel";
break;
case B_RGB15:
colorSpace = "15 Bits/Pixel";
break;
case B_RGB16:
colorSpace = "16 Bits/Pixel";
break;
case B_RGB32:
colorSpace = "32 Bits/Pixel";
break;
}
if (!fColorsMenu->FindItem(colorSpace.String()))
fColorsMenu->AddItem(new BMenuItem(colorSpace.String(),
new BMessage(POP_COLORS_MSG)));
}
ControlMenuRect.Set(50.0, 58.0, 171.0, 76.0); ControlMenuRect.Set(50.0, 58.0, 171.0, 76.0);
@ -159,14 +155,19 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fColorsField->SetDivider(38.0); fColorsField->SetDivider(38.0);
fControlsBox->AddChild(fColorsField); controlsBox->AddChild(fColorsField);
// XXX: When we'll start to use OpenBeOS BScreen, we will be able to dynamically
// build this menu using the available refresh rates, without limiting ourself
// to 85 hz.
fRefreshMenu = new BPopUpMenu("", true, true); fRefreshMenu = new BPopUpMenu("", true, true);
fRefreshMenu->AddItem(new BMenuItem("56 Hz", new BMessage(POP_REFRESH_MSG))); fRefreshMenu->AddItem(new BMenuItem("56 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("60 Hz", new BMessage(POP_REFRESH_MSG))); fRefreshMenu->AddItem(new BMenuItem("60 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("70 Hz", new BMessage(POP_REFRESH_MSG))); fRefreshMenu->AddItem(new BMenuItem("70 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("72 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("75 Hz", new BMessage(POP_REFRESH_MSG))); fRefreshMenu->AddItem(new BMenuItem("75 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("85 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("100 Hz", new BMessage(POP_REFRESH_MSG)));
fRefreshMenu->AddItem(new BMenuItem("Other...", new BMessage(POP_OTHER_REFRESH_MSG))); fRefreshMenu->AddItem(new BMenuItem("Other...", new BMessage(POP_OTHER_REFRESH_MSG)));
ControlMenuRect.Set(19.0, 86.0, 171.0, 104.0); ControlMenuRect.Set(19.0, 86.0, 171.0, 104.0);
@ -178,9 +179,9 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fRefreshField->SetDivider(69.0); fRefreshField->SetDivider(69.0);
fControlsBox->AddChild(fRefreshField); controlsBox->AddChild(fRefreshField);
fScreenView->AddChild(fControlsBox); fScreenView->AddChild(controlsBox);
ButtonRect.Set(10.0, 167, 100.0, 200.0); ButtonRect.Set(10.0, 167, 100.0, 200.0);
@ -218,25 +219,8 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
fInitialResolution = marked; fInitialResolution = marked;
switch (mode.space) { marked = fColorsMenu->FindItem(colorspace_to_string(mode.space));
case B_RGB32:
marked = fColorsMenu->FindItem("32 Bits/Pixel");
break;
case B_RGB16:
marked = fColorsMenu->FindItem("16 Bits/Pixel");
break;
case B_RGB15:
marked = fColorsMenu->FindItem("15 Bits/Pixel");
break;
case B_CMAP8:
default:
marked = fColorsMenu->FindItem("8 Bits/Pixel");
break;
}
marked->SetMarked(true); marked->SetMarked(true);
fInitialColors = marked; fInitialColors = marked;
@ -252,13 +236,10 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
marked = fRefreshMenu->FindItem(string.String()); marked = fRefreshMenu->FindItem(string.String());
if (marked != NULL) { if (marked != NULL) {
marked->SetMarked(true); marked->SetMarked(true);
fInitialRefresh = marked; fInitialRefresh = marked;
} else {
} else {
BMenuItem *other = fRefreshMenu->FindItem("Other..."); BMenuItem *other = fRefreshMenu->FindItem("Other...");
string.Truncate(0); string.Truncate(0);
@ -281,6 +262,7 @@ ScreenWindow::ScreenWindow(ScreenSettings *Settings)
} }
} }
ScreenWindow::~ScreenWindow() ScreenWindow::~ScreenWindow()
{ {
delete fSettings; delete fSettings;
@ -320,36 +302,6 @@ ScreenWindow::WorkspaceActivated(int32 ws, bool state)
} }
void
ScreenWindow::CheckApplyEnabled()
{
int equals = 0;
if (fResolutionMenu->FindMarked() == fInitialResolution)
equals++;
else
equals--;
if (fColorsMenu->FindMarked() == fInitialColors)
equals++;
else
equals--;
if (fRefreshMenu->FindMarked() == fInitialRefresh)
equals++;
else
equals--;
if(equals != 3) {
fApplyButton->SetEnabled(true);
fRevertButton->SetEnabled(true);
} else {
fApplyButton->SetEnabled(false);
fRevertButton->SetEnabled(false);
}
}
void void
ScreenWindow::MessageReceived(BMessage* message) ScreenWindow::MessageReceived(BMessage* message)
{ {
@ -375,21 +327,20 @@ ScreenWindow::MessageReceived(BMessage* message)
{ {
CheckApplyEnabled(); CheckApplyEnabled();
BMessage *newMessage = new BMessage(UPDATE_DESKTOP_MSG); BMessage newMessage(UPDATE_DESKTOP_MSG);
const char *resolution = fResolutionMenu->FindMarked()->Label(); const char *resolution = fResolutionMenu->FindMarked()->Label();
newMessage->AddString("resolution", resolution); newMessage.AddString("resolution", resolution);
PostMessage(newMessage, fScreenDrawView); PostMessage(&newMessage, fScreenDrawView);
break; break;
} }
case POP_COLORS_MSG: case POP_COLORS_MSG:
{ {
CheckApplyEnabled(); CheckApplyEnabled();
break; break;
} }
@ -509,8 +460,7 @@ ScreenWindow::MessageReceived(BMessage* message)
mode = &fSupportedModes[c]; mode = &fSupportedModes[c];
} }
mode->timing.pixel_clock = (uint32)((mode->timing.h_total * mode->timing.v_total) * refresh / 1000); mode->timing.pixel_clock = (uint32)((mode->timing.h_total * mode->timing.v_total) * refresh / 1000);
if (fColorsMenu->FindMarked() == fColorsMenu->FindItem("8 Bits/Pixel")) if (fColorsMenu->FindMarked() == fColorsMenu->FindItem("8 Bits/Pixel"))
mode->space = B_CMAP8; mode->space = B_CMAP8;
@ -531,8 +481,11 @@ ScreenWindow::MessageReceived(BMessage* message)
int32 button = WorkspacesAlert->Go(); int32 button = WorkspacesAlert->Go();
if (button == 1) if (button == 1) {
PostMessage(new BMessage(BUTTON_REVERT_MSG));
break; break;
}
int32 old = current_workspace(); int32 old = current_workspace();
int32 totalWorkspaces = count_workspaces(); int32 totalWorkspaces = count_workspaces();
@ -615,3 +568,51 @@ ScreenWindow::MessageReceived(BMessage* message)
break; break;
} }
} }
void
ScreenWindow::CheckApplyEnabled()
{
if ((fResolutionMenu->FindMarked() != fInitialResolution)
|| (fColorsMenu->FindMarked() != fInitialColors)
|| (fRefreshMenu->FindMarked() != fInitialRefresh)) {
fApplyButton->SetEnabled(true);
fRevertButton->SetEnabled(true);
} else {
fApplyButton->SetEnabled(false);
fRevertButton->SetEnabled(false);
}
}
void
ScreenWindow::CheckUpdateDisplayModes()
{
uint32 c;
// Add supported resolutions to the menu
for (c = 0; c < fTotalModes; c++) {
const char *mode = mode_to_string(fSupportedModes[c]);
if (!fResolutionMenu->FindItem(mode))
fResolutionMenu->AddItem(new BMenuItem(mode, new BMessage(POP_RESOLUTION_MSG)));
}
// Add supported colorspaces to the menu
for (c = 0; c < fTotalModes; c++) {
const char *colorSpace = colorspace_to_string(fSupportedModes[c].space);
if (!fColorsMenu->FindItem(colorSpace))
fColorsMenu->AddItem(new BMenuItem(colorSpace, new BMessage(POP_COLORS_MSG)));
}
/*
XXX: BeOS's BScreen limits the refresh to 85 hz. I hope that OpenBeOS BScreen will remove
this limitation.
for (c = 0; c < fTotalModes; c++) {
display_mode *mode = &fSupportedModes[c];
int32 refresh = (mode->timing.pixel_clock * 1000) / (mode->timing.h_total * mode->timing.v_total);
printf("%d\n", refresh);
}
*/
}

View File

@ -25,36 +25,35 @@ public:
virtual void ScreenChanged(BRect frame, color_space mode); virtual void ScreenChanged(BRect frame, color_space mode);
private: private:
void CheckApplyEnabled(); void CheckApplyEnabled();
void CheckUpdateDisplayModes();
ScreenSettings *fSettings; ScreenSettings *fSettings;
ScreenView *fScreenView; ScreenView *fScreenView;
ScreenDrawView *fScreenDrawView; ScreenDrawView *fScreenDrawView;
BPopUpMenu *fWorkspaceMenu; BPopUpMenu *fWorkspaceMenu;
BMenuField *fWorkspaceField; BMenuField *fWorkspaceField;
BPopUpMenu *fWorkspaceCountMenu; BPopUpMenu *fWorkspaceCountMenu;
BMenuField *fWorkspaceCountField; BMenuField *fWorkspaceCountField;
BPopUpMenu *fResolutionMenu; BPopUpMenu *fResolutionMenu;
BMenuField *fResolutionField; BMenuField *fResolutionField;
BPopUpMenu *fColorsMenu; BPopUpMenu *fColorsMenu;
BMenuField *fColorsField; BMenuField *fColorsField;
BPopUpMenu *fRefreshMenu; BPopUpMenu *fRefreshMenu;
BMenuField *fRefreshField; BMenuField *fRefreshField;
BMenuItem *fCurrentWorkspaceItem; BMenuItem *fCurrentWorkspaceItem;
BMenuItem *fAllWorkspacesItem; BMenuItem *fAllWorkspacesItem;
BButton *fDefaultsButton; BButton *fDefaultsButton;
BButton *fApplyButton; BButton *fApplyButton;
BButton *fRevertButton; BButton *fRevertButton;
BBox *fScreenBox; BMenuItem *fInitialResolution;
BBox *fControlsBox; BMenuItem *fInitialColors;
BMenuItem *fInitialResolution; BMenuItem *fInitialRefresh;
BMenuItem *fInitialColors; display_mode fInitialMode;
BMenuItem *fInitialRefresh; display_mode *fSupportedModes;
display_mode fInitialMode; uint32 fTotalModes;
display_mode *fSupportedModes; float fCustomRefresh;
uint32 fTotalModes; float fInitialRefreshN;
float fCustomRefresh;
float fInitialRefreshN;
}; };
#endif #endif