1998-10-20 17:25:25 +04:00
|
|
|
//
|
|
|
|
// Fractal drawing demo for the Fast Light Tool Kit (FLTK).
|
|
|
|
//
|
|
|
|
// This is a GLUT demo program, with modifications to
|
2002-01-03 17:08:08 +03:00
|
|
|
// demonstrate how to add FLTK controls to a GLUT program. The GLUT
|
|
|
|
// code is unchanged except for the end (search for FLTK to find changes).
|
1998-10-20 17:25:25 +04:00
|
|
|
//
|
2016-01-31 07:33:54 +03:00
|
|
|
// Copyright 1998-2016 by Bill Spitzak and others.
|
1998-10-20 17:25:25 +04:00
|
|
|
//
|
2011-07-19 08:49:30 +04:00
|
|
|
// This library is free software. Distribution and use rights are outlined in
|
|
|
|
// the file "COPYING" which should have been included with this file. If this
|
|
|
|
// file is missing or damaged, see the license at:
|
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// https://www.fltk.org/COPYING.php
|
1998-10-20 17:25:25 +04:00
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// Please see the following page on how to report bugs and issues:
|
2005-04-16 04:13:17 +04:00
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// https://www.fltk.org/bugs.php
|
1998-10-20 17:25:25 +04:00
|
|
|
//
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2010-12-15 15:11:16 +03:00
|
|
|
#include <config.h>
|
2001-04-30 21:17:01 +04:00
|
|
|
#if !HAVE_GL || !HAVE_GL_GLU_H
|
1998-10-06 22:21:25 +04:00
|
|
|
#include <FL/Fl.H>
|
|
|
|
#include <FL/fl_message.H>
|
|
|
|
int main(int, char**) {
|
2016-01-31 07:33:54 +03:00
|
|
|
fl_alert("This demo does not work without GL and GLU");
|
1998-10-06 22:21:25 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* To compile: cc -o fractals fractals.c -lGL -lGLU -lX11 -lglut -lXmu -lm
|
|
|
|
*
|
|
|
|
* Usage: fractals
|
|
|
|
*
|
2020-07-01 19:03:10 +03:00
|
|
|
* Homework 6, Part 2: fractal mountains and fractal trees
|
1998-10-06 22:21:25 +04:00
|
|
|
* (Pretty Late)
|
|
|
|
*
|
2020-07-01 19:03:10 +03:00
|
|
|
* Draws fractal mountains and trees -- and an island of mountains in water
|
1998-10-06 22:21:25 +04:00
|
|
|
* (I tried having trees on the island but it didn't work too well.)
|
|
|
|
*
|
|
|
|
* Two viewer modes: polar and flying (both restrained to y>0 for up vector).
|
|
|
|
* Keyboard 0->9 and +/- control speed when flying.
|
|
|
|
*
|
|
|
|
* Only keyboard commands are 0-9 and +/- for speed in flying mode.
|
|
|
|
*
|
|
|
|
* Fog would make the island look much better, but I couldn't get it to work
|
|
|
|
* correctly. Would line up on -z axis not from eye.
|
|
|
|
*
|
|
|
|
* Philip Winston - 3/4/95
|
|
|
|
* pwinston@hmc.edu
|
|
|
|
* http://www.cs.hmc.edu/people/pwinston
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <FL/glut.H>
|
2006-08-25 20:12:13 +04:00
|
|
|
#include <FL/glu.h>
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <limits.h> /* ULONG_MAX is defined here */
|
|
|
|
#include <float.h> /* FLT_MAX is atleast defined here */
|
|
|
|
|
|
|
|
#include <time.h> /* for random seed */
|
|
|
|
|
2002-01-03 17:08:08 +03:00
|
|
|
#include "fracviewer.h"
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2018-02-09 17:39:42 +03:00
|
|
|
#if defined(_WIN32)
|
2002-01-03 17:08:08 +03:00
|
|
|
# define drand48() (((float) rand())/((float) RAND_MAX))
|
|
|
|
# define srand48(x) (srand((x)))
|
2001-12-06 03:17:47 +03:00
|
|
|
#elif defined __APPLE__
|
2002-01-03 17:08:08 +03:00
|
|
|
# define drand48() (((float) rand())/((float) RAND_MAX))
|
|
|
|
# define srand48(x) (srand((x)))
|
1998-10-06 22:21:25 +04:00
|
|
|
#endif
|
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
typedef enum { NOTALLOWED, MOUNTAIN, TREE, ISLAND, BIGMTN, STEM, LEAF,
|
1998-10-06 22:21:25 +04:00
|
|
|
MOUNTAIN_MAT, WATER_MAT, LEAF_MAT, TREE_MAT, STEMANDLEAVES,
|
|
|
|
AXES } DisplayLists;
|
|
|
|
|
|
|
|
#define MAXLEVEL 8
|
|
|
|
|
|
|
|
int Rebuild = 1, /* Rebuild display list in next display? */
|
2006-06-04 14:21:45 +04:00
|
|
|
fractal = TREE, /* What fractal are we building */
|
2020-07-01 19:03:10 +03:00
|
|
|
Level = 4; /* levels of recursion for fractals */
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
int DrawAxes = 0;
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/************************* VECTOR JUNK *************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
/* print vertex to stderr */
|
|
|
|
void printvert(float v[3])
|
|
|
|
{
|
|
|
|
fprintf(stderr, "(%f, %f, %f)\n", v[0], v[1], v[2]);
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
#if 0 // removed for FL, it is in fracviewer.c
|
1998-10-06 22:21:25 +04:00
|
|
|
/* normalizes v */
|
|
|
|
void normalize(GLfloat v[3])
|
|
|
|
{
|
|
|
|
GLfloat d = sqrt(v[0]*v[0] + v[1]*v[1] + v[2]*v[2]);
|
|
|
|
|
|
|
|
if (d == 0)
|
|
|
|
fprintf(stderr, "Zero length vector in normalize\n");
|
|
|
|
else
|
|
|
|
v[0] /= d; v[1] /= d; v[2] /= d;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* calculates a normalized crossproduct to v1, v2 */
|
|
|
|
void ncrossprod(float v1[3], float v2[3], float cp[3])
|
|
|
|
{
|
|
|
|
cp[0] = v1[1]*v2[2] - v1[2]*v2[1];
|
|
|
|
cp[1] = v1[2]*v2[0] - v1[0]*v2[2];
|
|
|
|
cp[2] = v1[0]*v2[1] - v1[1]*v2[0];
|
|
|
|
normalize(cp);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* calculates normal to the triangle designated by v1, v2, v3 */
|
|
|
|
void triagnormal(float v1[3], float v2[3], float v3[3], float norm[3])
|
|
|
|
{
|
|
|
|
float vec1[3], vec2[3];
|
|
|
|
|
|
|
|
vec1[0] = v3[0] - v1[0]; vec2[0] = v2[0] - v1[0];
|
|
|
|
vec1[1] = v3[1] - v1[1]; vec2[1] = v2[1] - v1[1];
|
|
|
|
vec1[2] = v3[2] - v1[2]; vec2[2] = v2[2] - v1[2];
|
|
|
|
|
|
|
|
ncrossprod(vec2, vec1, norm);
|
|
|
|
}
|
|
|
|
|
|
|
|
float xzlength(float v1[3], float v2[3])
|
|
|
|
{
|
|
|
|
return sqrt((v1[0] - v2[0])*(v1[0] - v2[0]) +
|
|
|
|
(v1[2] - v2[2])*(v1[2] - v2[2]));
|
|
|
|
}
|
|
|
|
|
|
|
|
float xzslope(float v1[3], float v2[3])
|
|
|
|
{
|
|
|
|
return ((v1[0] != v2[0]) ? ((v1[2] - v2[2]) / (v1[0] - v2[0]))
|
2020-07-01 19:03:10 +03:00
|
|
|
: FLT_MAX);
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/************************ MOUNTAIN STUFF ***********************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
GLfloat DispFactor[MAXLEVEL]; /* Array of what to multiply random number
|
2020-07-01 19:03:10 +03:00
|
|
|
by for a given level to get midpoint
|
|
|
|
displacement */
|
1998-10-06 22:21:25 +04:00
|
|
|
GLfloat DispBias[MAXLEVEL]; /* Array of what to add to random number
|
2020-07-01 19:03:10 +03:00
|
|
|
before multiplying it by DispFactor */
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
#define NUMRANDS 191
|
|
|
|
float RandTable[NUMRANDS]; /* hash table of random numbers so we can
|
2020-07-01 19:03:10 +03:00
|
|
|
raise the same midpoints by the same amount */
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
/* The following are for permitting an edge of a moutain to be */
|
|
|
|
/* pegged so it won't be displaced up or down. This makes it */
|
|
|
|
/* easier to setup scenes and makes a single moutain look better */
|
|
|
|
|
|
|
|
GLfloat Verts[3][3], /* Vertices of outside edges of mountain */
|
|
|
|
Slopes[3]; /* Slopes between these outside edges */
|
2020-07-01 19:03:10 +03:00
|
|
|
int Pegged[3]; /* Is this edge pegged or not */
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Comes up with a new table of random numbers [0,1)
|
|
|
|
*/
|
|
|
|
void InitRandTable(unsigned int seed)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
srand48((long) seed);
|
|
|
|
for (i = 0; i < NUMRANDS; i++)
|
|
|
|
RandTable[i] = drand48() - 0.5;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* calculate midpoint and displace it if required */
|
|
|
|
void Midpoint(GLfloat mid[3], GLfloat v1[3], GLfloat v2[3],
|
2020-07-01 19:03:10 +03:00
|
|
|
int edge, int level)
|
1998-10-06 22:21:25 +04:00
|
|
|
{
|
|
|
|
unsigned hash;
|
|
|
|
|
|
|
|
mid[0] = (v1[0] + v2[0]) / 2;
|
|
|
|
mid[1] = (v1[1] + v2[1]) / 2;
|
|
|
|
mid[2] = (v1[2] + v2[2]) / 2;
|
2020-07-01 19:03:10 +03:00
|
|
|
if (!Pegged[edge] || (fabs(xzslope(Verts[edge], mid)
|
1998-10-06 22:21:25 +04:00
|
|
|
- Slopes[edge]) > 0.00001)) {
|
|
|
|
srand48((int)((v1[0]+v2[0])*23344));
|
|
|
|
hash = unsigned(drand48() * 7334334);
|
|
|
|
srand48((int)((v2[2]+v1[2])*43433));
|
|
|
|
hash = (unsigned)(drand48() * 634344 + hash) % NUMRANDS;
|
|
|
|
mid[1] += ((RandTable[hash] + DispBias[level]) * DispFactor[level]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2020-07-01 19:03:10 +03:00
|
|
|
* Recursive moutain drawing routine -- from lecture with addition of
|
1998-10-06 22:21:25 +04:00
|
|
|
* allowing an edge to be pegged. This function requires the above
|
2020-07-01 19:03:10 +03:00
|
|
|
* globals to be set, as well as the Level global for fractal level
|
1998-10-06 22:21:25 +04:00
|
|
|
*/
|
|
|
|
static float cutoff = -1;
|
|
|
|
|
|
|
|
void FMR(GLfloat v1[3], GLfloat v2[3], GLfloat v3[3], int level)
|
|
|
|
{
|
|
|
|
if (level == Level) {
|
|
|
|
GLfloat norm[3];
|
|
|
|
if (v1[1] <= cutoff && v2[1]<=cutoff && v3[1]<=cutoff) return;
|
|
|
|
triagnormal(v1, v2, v3, norm);
|
|
|
|
glNormal3fv(norm);
|
|
|
|
glVertex3fv(v1);
|
|
|
|
glVertex3fv(v2);
|
|
|
|
glVertex3fv(v3);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
GLfloat m1[3], m2[3], m3[3];
|
|
|
|
|
|
|
|
Midpoint(m1, v1, v2, 0, level);
|
|
|
|
Midpoint(m2, v2, v3, 1, level);
|
|
|
|
Midpoint(m3, v3, v1, 2, level);
|
|
|
|
|
|
|
|
FMR(v1, m1, m3, level + 1);
|
|
|
|
FMR(m1, v2, m2, level + 1);
|
|
|
|
FMR(m3, m2, v3, level + 1);
|
|
|
|
FMR(m1, m2, m3, level + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* sets up lookup tables and calls recursive mountain function
|
|
|
|
*/
|
|
|
|
void FractalMountain(GLfloat v1[3], GLfloat v2[3], GLfloat v3[3],
|
|
|
|
int pegged[3])
|
|
|
|
{
|
|
|
|
GLfloat lengths[MAXLEVEL];
|
|
|
|
GLfloat fraction[8] = { 0.3, 0.3, 0.4, 0.2, 0.3, 0.2, 0.4, 0.4 };
|
|
|
|
GLfloat bias[8] = { 0.1, 0.1, 0.1, 0.1, 0.1, 0.1, 0.1, 0.1 };
|
|
|
|
int i;
|
2020-07-01 19:03:10 +03:00
|
|
|
float avglen = (xzlength(v1, v2) +
|
1998-10-06 22:21:25 +04:00
|
|
|
xzlength(v2, v3) +
|
2020-07-01 19:03:10 +03:00
|
|
|
xzlength(v3, v1) / 3);
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
|
|
Verts[0][i] = v1[i]; /* set mountain vertex globals */
|
|
|
|
Verts[1][i] = v2[i];
|
|
|
|
Verts[2][i] = v3[i];
|
|
|
|
Pegged[i] = pegged[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
Slopes[0] = xzslope(Verts[0], Verts[1]); /* set edge slope globals */
|
|
|
|
Slopes[1] = xzslope(Verts[1], Verts[2]);
|
|
|
|
Slopes[2] = xzslope(Verts[2], Verts[0]);
|
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
lengths[0] = avglen;
|
|
|
|
for (i = 1; i < Level; i++) {
|
1998-10-06 22:21:25 +04:00
|
|
|
lengths[i] = lengths[i-1]/2; /* compute edge length for each level */
|
|
|
|
}
|
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
for (i = 0; i < Level; i++) { /* DispFactor and DispBias arrays */
|
1998-10-06 22:21:25 +04:00
|
|
|
DispFactor[i] = (lengths[i] * ((i <= 7) ? fraction[i] : fraction[7]));
|
|
|
|
DispBias[i] = ((i <= 7) ? bias[i] : bias[7]);
|
2020-07-01 19:03:10 +03:00
|
|
|
}
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
glBegin(GL_TRIANGLES);
|
|
|
|
FMR(v1, v2, v3, 0); /* issues no GL but vertex calls */
|
|
|
|
glEnd();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* draw a mountain and build the display list
|
|
|
|
*/
|
|
|
|
void CreateMountain(void)
|
|
|
|
{
|
|
|
|
GLfloat v1[3] = { 0, 0, -1 }, v2[3] = { -1, 0, 1 }, v3[3] = { 1, 0, 1 };
|
|
|
|
int pegged[3] = { 1, 1, 1 };
|
|
|
|
|
|
|
|
glNewList(MOUNTAIN, GL_COMPILE);
|
|
|
|
glPushAttrib(GL_LIGHTING_BIT);
|
|
|
|
glCallList(MOUNTAIN_MAT);
|
|
|
|
FractalMountain(v1, v2, v3, pegged);
|
|
|
|
glPopAttrib();
|
|
|
|
glEndList();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* new random numbers to make a different moutain
|
|
|
|
*/
|
|
|
|
void NewMountain(void)
|
|
|
|
{
|
|
|
|
InitRandTable(time(NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/***************************** TREE ****************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
long TreeSeed; /* for srand48 - remember so we can build "same tree"
|
|
|
|
at a different level */
|
|
|
|
|
|
|
|
/*
|
2020-07-01 19:03:10 +03:00
|
|
|
* recursive tree drawing thing, fleshed out from class notes pseudocode
|
1998-10-06 22:21:25 +04:00
|
|
|
*/
|
2010-12-06 18:39:31 +03:00
|
|
|
void FractalTree(int level, long level_seed)
|
1998-10-06 22:21:25 +04:00
|
|
|
{
|
|
|
|
if (level == Level) {
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(drand48()*180, 0, 1, 0);
|
|
|
|
glCallList(STEMANDLEAVES);
|
|
|
|
glPopMatrix();
|
|
|
|
} else {
|
|
|
|
glCallList(STEM);
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(drand48()*180, 0, 1, 0);
|
|
|
|
glTranslatef(0, 1, 0);
|
|
|
|
glScalef(0.7, 0.7, 0.7);
|
|
|
|
|
2010-12-06 18:39:31 +03:00
|
|
|
srand48(level_seed+1);
|
2020-07-01 19:03:10 +03:00
|
|
|
glPushMatrix();
|
1998-10-06 22:21:25 +04:00
|
|
|
glRotatef(110 + drand48()*40, 0, 1, 0);
|
|
|
|
glRotatef(30 + drand48()*20, 0, 0, 1);
|
2010-12-06 18:39:31 +03:00
|
|
|
FractalTree(level + 1, level_seed+4);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPopMatrix();
|
|
|
|
|
2010-12-06 18:39:31 +03:00
|
|
|
srand48(level_seed+2);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(-130 + drand48()*40, 0, 1, 0);
|
|
|
|
glRotatef(30 + drand48()*20, 0, 0, 1);
|
2010-12-06 18:39:31 +03:00
|
|
|
FractalTree(level + 1, level_seed+5);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPopMatrix();
|
|
|
|
|
2010-12-06 18:39:31 +03:00
|
|
|
srand48(level_seed+3);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(-20 + drand48()*40, 0, 1, 0);
|
|
|
|
glRotatef(30 + drand48()*20, 0, 0, 1);
|
2010-12-06 18:39:31 +03:00
|
|
|
FractalTree(level + 1, level_seed+6);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPopMatrix();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create display lists for a leaf, a set of leaves, and a stem
|
|
|
|
*/
|
|
|
|
void CreateTreeLists(void)
|
|
|
|
{
|
|
|
|
GLUquadricObj *cylquad = gluNewQuadric();
|
|
|
|
int i;
|
|
|
|
|
|
|
|
glNewList(STEM, GL_COMPILE);
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(-90, 1, 0, 0);
|
|
|
|
gluCylinder(cylquad, 0.1, 0.08, 1, 10, 2 );
|
|
|
|
glPopMatrix();
|
|
|
|
glEndList();
|
|
|
|
|
|
|
|
glNewList(LEAF, GL_COMPILE); /* I think this was jeff allen's leaf idea */
|
|
|
|
glBegin(GL_TRIANGLES);
|
|
|
|
glNormal3f(-0.1, 0, 0.25); /* not normalized */
|
|
|
|
glVertex3f(0, 0, 0);
|
|
|
|
glVertex3f(0.25, 0.25, 0.1);
|
|
|
|
glVertex3f(0, 0.5, 0);
|
|
|
|
|
|
|
|
glNormal3f(0.1, 0, 0.25);
|
|
|
|
glVertex3f(0, 0, 0);
|
|
|
|
glVertex3f(0, 0.5, 0);
|
|
|
|
glVertex3f(-0.25, 0.25, 0.1);
|
|
|
|
glEnd();
|
|
|
|
glEndList();
|
|
|
|
|
|
|
|
glNewList(STEMANDLEAVES, GL_COMPILE);
|
|
|
|
glPushMatrix();
|
|
|
|
glPushAttrib(GL_LIGHTING_BIT);
|
|
|
|
glCallList(STEM);
|
|
|
|
glCallList(LEAF_MAT);
|
|
|
|
for(i = 0; i < 3; i++) {
|
|
|
|
glTranslatef(0, 0.333, 0);
|
|
|
|
glRotatef(90, 0, 1, 0);
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(0, 0, 1, 0);
|
|
|
|
glRotatef(50, 1, 0, 0);
|
|
|
|
glCallList(LEAF);
|
|
|
|
glPopMatrix();
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(180, 0, 1, 0);
|
|
|
|
glRotatef(60, 1, 0, 0);
|
|
|
|
glCallList(LEAF);
|
|
|
|
glPopMatrix();
|
|
|
|
}
|
|
|
|
glPopAttrib();
|
|
|
|
glPopMatrix();
|
|
|
|
glEndList();
|
1999-03-25 16:37:00 +03:00
|
|
|
|
|
|
|
gluDeleteQuadric(cylquad);
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* draw and build display list for tree
|
|
|
|
*/
|
|
|
|
void CreateTree(void)
|
|
|
|
{
|
|
|
|
srand48(TreeSeed);
|
|
|
|
|
|
|
|
glNewList(TREE, GL_COMPILE);
|
|
|
|
glPushMatrix();
|
|
|
|
glPushAttrib(GL_LIGHTING_BIT);
|
|
|
|
glCallList(TREE_MAT);
|
|
|
|
glTranslatef(0, -1, 0);
|
2010-12-06 18:39:31 +03:00
|
|
|
FractalTree(0, TreeSeed);
|
1998-10-06 22:21:25 +04:00
|
|
|
glPopAttrib();
|
|
|
|
glPopMatrix();
|
2020-07-01 19:03:10 +03:00
|
|
|
glEndList();
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* new seed for a new tree (groan)
|
|
|
|
*/
|
|
|
|
void NewTree(void)
|
|
|
|
{
|
|
|
|
TreeSeed = time(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/*********************** FRACTAL PLANET ************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
void CreateIsland(void)
|
|
|
|
{
|
|
|
|
cutoff = .06;
|
|
|
|
CreateMountain();
|
|
|
|
cutoff = -1;
|
|
|
|
glNewList(ISLAND, GL_COMPILE);
|
|
|
|
glPushAttrib(GL_LIGHTING_BIT);
|
|
|
|
glMatrixMode(GL_MODELVIEW);
|
|
|
|
glPushMatrix();
|
|
|
|
glCallList(WATER_MAT);
|
|
|
|
|
|
|
|
glBegin(GL_QUADS);
|
|
|
|
glNormal3f(0, 1, 0);
|
|
|
|
glVertex3f(10, 0.01, 10);
|
|
|
|
glVertex3f(10, 0.01, -10);
|
|
|
|
glVertex3f(-10, 0.01, -10);
|
|
|
|
glVertex3f(-10, 0.01, 10);
|
|
|
|
glEnd();
|
|
|
|
|
|
|
|
glPushMatrix();
|
|
|
|
glTranslatef(0, -0.1, 0);
|
|
|
|
glCallList(MOUNTAIN);
|
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(135, 0, 1, 0);
|
|
|
|
glTranslatef(0.2, -0.15, -0.4);
|
|
|
|
glCallList(MOUNTAIN);
|
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(-60, 0, 1, 0);
|
|
|
|
glTranslatef(0.7, -0.07, 0.5);
|
|
|
|
glCallList(MOUNTAIN);
|
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(-175, 0, 1, 0);
|
|
|
|
glTranslatef(-0.7, -0.05, -0.5);
|
|
|
|
glCallList(MOUNTAIN);
|
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPushMatrix();
|
|
|
|
glRotatef(165, 0, 1, 0);
|
|
|
|
glTranslatef(-0.9, -0.12, 0.0);
|
|
|
|
glCallList(MOUNTAIN);
|
|
|
|
glPopMatrix();
|
|
|
|
|
|
|
|
glPopMatrix();
|
|
|
|
glPopAttrib();
|
2020-07-01 19:03:10 +03:00
|
|
|
glEndList();
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void NewFractals(void)
|
|
|
|
{
|
|
|
|
NewMountain();
|
|
|
|
NewTree();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Create(int fract)
|
|
|
|
{
|
|
|
|
switch(fract) {
|
|
|
|
case MOUNTAIN:
|
|
|
|
CreateMountain();
|
|
|
|
break;
|
|
|
|
case TREE:
|
|
|
|
CreateTree();
|
|
|
|
break;
|
|
|
|
case ISLAND:
|
|
|
|
CreateIsland();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/**************************** OPENGL ***************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
void SetupMaterials(void)
|
|
|
|
{
|
|
|
|
GLfloat mtn_ambuse[] = { 0.426, 0.256, 0.108, 1.0 };
|
|
|
|
GLfloat mtn_specular[] = { 0.394, 0.272, 0.167, 1.0 };
|
|
|
|
GLfloat mtn_shininess[] = { 10 };
|
|
|
|
|
|
|
|
GLfloat water_ambuse[] = { 0.0, 0.1, 0.5, 1.0 };
|
|
|
|
GLfloat water_specular[] = { 0.0, 0.1, 0.5, 1.0 };
|
|
|
|
GLfloat water_shininess[] = { 10 };
|
|
|
|
|
|
|
|
GLfloat tree_ambuse[] = { 0.4, 0.25, 0.1, 1.0 };
|
|
|
|
GLfloat tree_specular[] = { 0.0, 0.0, 0.0, 1.0 };
|
|
|
|
GLfloat tree_shininess[] = { 0 };
|
|
|
|
|
|
|
|
GLfloat leaf_ambuse[] = { 0.0, 0.8, 0.0, 1.0 };
|
|
|
|
GLfloat leaf_specular[] = { 0.0, 0.8, 0.0, 1.0 };
|
|
|
|
GLfloat leaf_shininess[] = { 10 };
|
|
|
|
|
|
|
|
glNewList(MOUNTAIN_MAT, GL_COMPILE);
|
|
|
|
glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, mtn_ambuse);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SPECULAR, mtn_specular);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SHININESS, mtn_shininess);
|
|
|
|
glEndList();
|
|
|
|
|
|
|
|
glNewList(WATER_MAT, GL_COMPILE);
|
|
|
|
glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, water_ambuse);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SPECULAR, water_specular);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SHININESS, water_shininess);
|
|
|
|
glEndList();
|
|
|
|
|
|
|
|
glNewList(TREE_MAT, GL_COMPILE);
|
|
|
|
glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, tree_ambuse);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SPECULAR, tree_specular);
|
|
|
|
glMaterialfv(GL_FRONT, GL_SHININESS, tree_shininess);
|
|
|
|
glEndList();
|
|
|
|
|
|
|
|
glNewList(LEAF_MAT, GL_COMPILE);
|
|
|
|
glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, leaf_ambuse);
|
|
|
|
glMaterialfv(GL_FRONT_AND_BACK, GL_SPECULAR, leaf_specular);
|
|
|
|
glMaterialfv(GL_FRONT_AND_BACK, GL_SHININESS, leaf_shininess);
|
|
|
|
glEndList();
|
|
|
|
}
|
|
|
|
|
|
|
|
void myGLInit(void)
|
|
|
|
{
|
|
|
|
GLfloat light_ambient[] = { 0.0, 0.0, 0.0, 1.0 };
|
|
|
|
GLfloat light_diffuse[] = { 1.0, 1.0, 1.0, 1.0 };
|
|
|
|
GLfloat light_specular[] = { 1.0, 1.0, 1.0, 1.0 };
|
|
|
|
GLfloat light_position[] = { 0.0, 0.3, 0.3, 0.0 };
|
|
|
|
|
|
|
|
GLfloat lmodel_ambient[] = { 0.4, 0.4, 0.4, 1.0 };
|
|
|
|
|
|
|
|
glLightfv(GL_LIGHT0, GL_AMBIENT, light_ambient);
|
|
|
|
glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse);
|
|
|
|
glLightfv(GL_LIGHT0, GL_SPECULAR, light_specular);
|
|
|
|
glLightfv(GL_LIGHT0, GL_POSITION, light_position);
|
2020-07-01 19:03:10 +03:00
|
|
|
|
1998-10-06 22:21:25 +04:00
|
|
|
glLightModelfv(GL_LIGHT_MODEL_AMBIENT, lmodel_ambient);
|
|
|
|
|
|
|
|
glEnable(GL_LIGHTING);
|
|
|
|
glEnable(GL_LIGHT0);
|
|
|
|
|
|
|
|
glDepthFunc(GL_LEQUAL);
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
|
|
|
|
glEnable(GL_NORMALIZE);
|
|
|
|
#if 0
|
|
|
|
glEnable(GL_CULL_FACE);
|
|
|
|
glCullFace(GL_BACK);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
glShadeModel(GL_SMOOTH);
|
|
|
|
#if 0
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
SetupMaterials();
|
|
|
|
CreateTreeLists();
|
|
|
|
|
|
|
|
glFlush();
|
2020-07-01 19:03:10 +03:00
|
|
|
}
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/************************ GLUT STUFF ***************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
2005-05-12 23:16:27 +04:00
|
|
|
int winwidth = 1;
|
|
|
|
int winheight = 1;
|
|
|
|
|
1998-10-06 22:21:25 +04:00
|
|
|
void reshape(int w, int h)
|
|
|
|
{
|
|
|
|
glViewport(0,0,w,h);
|
2005-05-12 23:16:27 +04:00
|
|
|
|
|
|
|
winwidth = w;
|
|
|
|
winheight = h;
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void display(void)
|
2005-05-12 23:16:27 +04:00
|
|
|
{
|
|
|
|
time_t curtime;
|
|
|
|
char buf[255];
|
|
|
|
static time_t fpstime = 0;
|
|
|
|
static int fpscount = 0;
|
|
|
|
static int fps = 0;
|
|
|
|
|
1998-10-06 22:21:25 +04:00
|
|
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
|
|
|
|
|
|
|
glMatrixMode(GL_PROJECTION);
|
2005-05-12 23:16:27 +04:00
|
|
|
glLoadIdentity();
|
|
|
|
gluPerspective(60.0, (GLdouble)winwidth/winheight, 0.01, 100);
|
1998-10-06 22:21:25 +04:00
|
|
|
agvViewTransform();
|
|
|
|
|
|
|
|
glMatrixMode(GL_MODELVIEW);
|
|
|
|
glLoadIdentity();
|
|
|
|
|
|
|
|
if (Rebuild) {
|
2006-06-04 14:21:45 +04:00
|
|
|
Create(fractal);
|
1998-10-06 22:21:25 +04:00
|
|
|
Rebuild = 0;
|
|
|
|
}
|
|
|
|
|
2006-06-04 14:21:45 +04:00
|
|
|
glCallList(fractal);
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
if (DrawAxes)
|
|
|
|
glCallList(AXES);
|
|
|
|
|
2005-05-12 23:16:27 +04:00
|
|
|
glMatrixMode(GL_PROJECTION);
|
|
|
|
glLoadIdentity();
|
|
|
|
gluOrtho2D(0.0, winwidth, 0.0, winheight);
|
|
|
|
|
|
|
|
sprintf(buf, "FPS=%d", fps);
|
|
|
|
glColor3f(1.0f, 1.0f, 1.0f);
|
|
|
|
gl_font(FL_HELVETICA, 12);
|
|
|
|
gl_draw(buf, 10, 10);
|
|
|
|
|
2002-01-03 17:08:08 +03:00
|
|
|
//
|
|
|
|
// Use glFinish() instead of glFlush() to avoid getting many frames
|
|
|
|
// ahead of the display (problem with some Linux OpenGL implementations...)
|
|
|
|
//
|
|
|
|
|
|
|
|
glFinish();
|
2005-05-12 23:16:27 +04:00
|
|
|
|
|
|
|
// Update frames-per-second
|
|
|
|
fpscount ++;
|
|
|
|
curtime = time(NULL);
|
|
|
|
if ((curtime - fpstime) >= 2)
|
|
|
|
{
|
|
|
|
fps = (fps + fpscount / (curtime - fpstime)) / 2;
|
|
|
|
fpstime = curtime;
|
|
|
|
fpscount = 0;
|
|
|
|
}
|
1998-10-06 22:21:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void visible(int v)
|
|
|
|
{
|
|
|
|
if (v == GLUT_VISIBLE)
|
|
|
|
agvSetAllowIdle(1);
|
|
|
|
else {
|
|
|
|
glutIdleFunc(NULL);
|
|
|
|
agvSetAllowIdle(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void menuuse(int v)
|
|
|
|
{
|
|
|
|
if (v == GLUT_MENU_NOT_IN_USE)
|
|
|
|
agvSetAllowIdle(1);
|
|
|
|
else {
|
|
|
|
glutIdleFunc(NULL);
|
|
|
|
agvSetAllowIdle(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/******************* MENU SETUP & HANDLING *********************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
typedef enum { MENU_QUIT, MENU_RAND, MENU_MOVE, MENU_AXES } MenuChoices;
|
|
|
|
|
|
|
|
void setlevel(int value)
|
|
|
|
{
|
|
|
|
Level = value;
|
|
|
|
Rebuild = 1;
|
|
|
|
glutPostRedisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
void choosefract(int value)
|
|
|
|
{
|
2006-06-04 14:21:45 +04:00
|
|
|
fractal = value;
|
1998-10-06 22:21:25 +04:00
|
|
|
Rebuild = 1;
|
|
|
|
glutPostRedisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
void handlemenu(int value)
|
|
|
|
{
|
|
|
|
switch (value) {
|
|
|
|
case MENU_QUIT:
|
|
|
|
exit(0);
|
|
|
|
break;
|
|
|
|
case MENU_RAND:
|
|
|
|
NewFractals();
|
|
|
|
Rebuild = 1;
|
|
|
|
glutPostRedisplay();
|
|
|
|
break;
|
|
|
|
case MENU_AXES:
|
|
|
|
DrawAxes = !DrawAxes;
|
|
|
|
glutPostRedisplay();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MenuInit(void)
|
|
|
|
{
|
|
|
|
int submenu3, submenu2, submenu1;
|
|
|
|
|
|
|
|
submenu1 = glutCreateMenu(setlevel);
|
2007-05-20 04:01:06 +04:00
|
|
|
glutAddMenuEntry((char *)"0", 0); glutAddMenuEntry((char *)"1", 1);
|
|
|
|
glutAddMenuEntry((char *)"2", 2); glutAddMenuEntry((char *)"3", 3);
|
|
|
|
glutAddMenuEntry((char *)"4", 4); glutAddMenuEntry((char *)"5", 5);
|
|
|
|
glutAddMenuEntry((char *)"6", 6); glutAddMenuEntry((char *)"7", 7);
|
|
|
|
glutAddMenuEntry((char *)"8", 8);
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
submenu2 = glutCreateMenu(choosefract);
|
2007-05-20 04:01:06 +04:00
|
|
|
glutAddMenuEntry((char *)"Moutain", MOUNTAIN);
|
|
|
|
glutAddMenuEntry((char *)"Tree", TREE);
|
|
|
|
glutAddMenuEntry((char *)"Island", ISLAND);
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
submenu3 = glutCreateMenu(agvSwitchMoveMode);
|
2007-05-20 04:01:06 +04:00
|
|
|
glutAddMenuEntry((char *)"Flying", FLYING);
|
|
|
|
glutAddMenuEntry((char *)"Polar", POLAR);
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
glutCreateMenu(handlemenu);
|
2007-05-20 04:01:06 +04:00
|
|
|
glutAddSubMenu((char *)"Level", submenu1);
|
|
|
|
glutAddSubMenu((char *)"Fractal", submenu2);
|
|
|
|
glutAddSubMenu((char *)"Movement", submenu3);
|
|
|
|
glutAddMenuEntry((char *)"New Fractal", MENU_RAND);
|
|
|
|
glutAddMenuEntry((char *)"Toggle Axes", MENU_AXES);
|
|
|
|
glutAddMenuEntry((char *)"Quit", MENU_QUIT);
|
1998-10-06 22:21:25 +04:00
|
|
|
glutAttachMenu(GLUT_RIGHT_BUTTON);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/**************************** MAIN *****************************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
2002-01-03 17:08:08 +03:00
|
|
|
// FLTK-style callbacks to Glut menu callback translators:
|
2010-12-08 17:00:35 +03:00
|
|
|
void setlevel(Fl_Widget*, void *value) {setlevel(fl_intptr_t(value));}
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2010-12-08 17:00:35 +03:00
|
|
|
void choosefract(Fl_Widget*, void *value) {choosefract(fl_intptr_t(value));}
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2010-12-08 17:00:35 +03:00
|
|
|
void handlemenu(Fl_Widget*, void *value) {handlemenu(fl_intptr_t(value));}
|
1998-10-06 22:21:25 +04:00
|
|
|
|
|
|
|
#include <FL/Fl_Button.H>
|
|
|
|
#include <FL/Fl_Group.H>
|
|
|
|
#include <FL/Fl_Window.H>
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
Changed OpenGL support for the Mac OS X platform: use cocoa instead of deprecated AGL.
All changes are mac-specific, except a very minor change in file src/gl_draw.cxx
where string drawing wrongly claimed to support @symbol, not possible
because symbols are drawn using non-GL primitives.
Unchanged application code can use the new FLTK code.
In addition, the new code allows mac applications to draw OpenGL scenes
at high resolution on so-called 'retina' displays, but this requires some
support from app code. They must call, before opening GL windows,
Fl::use_high_resolution(1);
and change their glViewport() calls as follows
glViewport(0, 0, pxel_w(), pixel_h());
This uses 2 new member functions of the Fl_Gl_Window class,
pixel_w() and pixel_h() returning the window dimensions in pixel
units, that is, twice the w() and h() when the window is mapped
on a retina display.
git-svn-id: file:///fltk/svn/fltk/branches/branch-1.3@10498 ea41ed52-d2ee-0310-a9c1-e6b18d33e121
2014-12-20 10:19:23 +03:00
|
|
|
Fl::use_high_res_GL(1);
|
2002-01-03 17:08:08 +03:00
|
|
|
// glutInit(&argc, argv); // this line removed for FLTK
|
1998-10-06 22:21:25 +04:00
|
|
|
|
2002-01-03 17:08:08 +03:00
|
|
|
// create FLTK window:
|
1998-10-06 22:21:25 +04:00
|
|
|
Fl_Window window(512+20, 512+100);
|
|
|
|
window.resizable(window);
|
|
|
|
|
|
|
|
// create a bunch of buttons:
|
|
|
|
Fl_Group *g = new Fl_Group(110,50,400-110,30,"Level:");
|
|
|
|
g->align(FL_ALIGN_LEFT);
|
|
|
|
g->begin();
|
|
|
|
Fl_Button *b;
|
|
|
|
b = new Fl_Button(110,50,30,30,"0"); b->callback(setlevel,(void*)0);
|
|
|
|
b = new Fl_Button(140,50,30,30,"1"); b->callback(setlevel,(void*)1);
|
|
|
|
b = new Fl_Button(170,50,30,30,"2"); b->callback(setlevel,(void*)2);
|
|
|
|
b = new Fl_Button(200,50,30,30,"3"); b->callback(setlevel,(void*)3);
|
|
|
|
b = new Fl_Button(230,50,30,30,"4"); b->callback(setlevel,(void*)4);
|
|
|
|
b = new Fl_Button(260,50,30,30,"5"); b->callback(setlevel,(void*)5);
|
|
|
|
b = new Fl_Button(290,50,30,30,"6"); b->callback(setlevel,(void*)6);
|
|
|
|
b = new Fl_Button(320,50,30,30,"7"); b->callback(setlevel,(void*)7);
|
|
|
|
b = new Fl_Button(350,50,30,30,"8"); b->callback(setlevel,(void*)8);
|
|
|
|
g->end();
|
|
|
|
|
|
|
|
b = new Fl_Button(400,50,100,30,"New Fractal"); b->callback(handlemenu,(void*)MENU_RAND);
|
2020-07-01 19:03:10 +03:00
|
|
|
|
1998-10-06 22:21:25 +04:00
|
|
|
b = new Fl_Button( 10,10,100,30,"Mountain"); b->callback(choosefract,(void*)MOUNTAIN);
|
|
|
|
b = new Fl_Button(110,10,100,30,"Tree"); b->callback(choosefract,(void*)TREE);
|
|
|
|
b = new Fl_Button(210,10,100,30,"Island"); b->callback(choosefract,(void*)ISLAND);
|
|
|
|
b = new Fl_Button(400,10,100,30,"Quit"); b->callback(handlemenu,(void*)MENU_QUIT);
|
|
|
|
|
|
|
|
|
|
|
|
window.show(argc,argv); // glut will die unless parent window visible
|
|
|
|
window.begin(); // this will cause Glut window to be a child
|
|
|
|
glutInitWindowSize(512, 512);
|
|
|
|
glutInitWindowPosition(10,90); // place it inside parent window
|
|
|
|
glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA | GLUT_DEPTH | GLUT_MULTISAMPLE);
|
|
|
|
glutCreateWindow("Fractal Planet?");
|
|
|
|
window.end();
|
|
|
|
window.resizable(glut_window);
|
|
|
|
|
|
|
|
agvInit(1); /* 1 cause we don't have our own idle */
|
|
|
|
|
|
|
|
glutReshapeFunc(reshape);
|
|
|
|
glutDisplayFunc(display);
|
|
|
|
glutVisibilityFunc(visible);
|
|
|
|
glutMenuStateFunc(menuuse);
|
|
|
|
|
|
|
|
NewFractals();
|
|
|
|
agvMakeAxesList(AXES);
|
2020-07-01 19:03:10 +03:00
|
|
|
myGLInit();
|
1998-10-06 22:21:25 +04:00
|
|
|
MenuInit();
|
|
|
|
|
1999-01-19 22:10:39 +03:00
|
|
|
glutMainLoop(); // you could use Fl::run() instead
|
|
|
|
|
1998-10-06 22:21:25 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|