Click here to Skip to main content
Click here to Skip to main content

3DHelper

, 17 Feb 2009 CPOL
Rate this:
Please Sign up or sign in to vote.
Helper class to display 3D data
3DHelperDemo

Introduction

Ever had the problem to display 3D-related data? E.g. to display a 3D-curve in a Windows dialog window? Don't like to implement a whole DirectX/OpenGL class hierarchy?
There is a solution: The 3DHelper class!

More tools can be found on my homepage.

Background  

If you are familiar with 3D programming, you already might have used free/not so free libraries to perform the task of rendering 3D objects onto your screen.
The big (in my opinion!) drawback of that approach is: your code is getting more and more complicated to maintain and even understand. Not to mention portability.

Having just some files to be included into your project, performing all the necessary math-stuff to "get  3D" sounds too nice to be true? Now its real!

The Solution

The aim of the API described in this article is simply to help you to project 3D data into your 2D window.
Given a 3D point (e.g. a point being part of a nurbs-surface, a 3D function value, etc.) the API transforms this point and returns a coordinate pair representing a dot in your window. 

And the best - there is no sophisticated OpenGL/DirectX stuff involved. This is a pure software solution.

Using the Code 

To allow the API to help you, it needs some basic steps to be performed:

  1. Add a variable of type CMF3DHelper to your project.
  2. At start up (and after each resize event) call the CMF3DHelper::Initialize() function of your variable. (see 1.)
    This Initialize() function needs parameters:
    • A pointer to the device context (DC) of your window (used to get the window size)
    • The "cube" representing your data, represented by two vectors: vmin and vmax.
      All of your data points should lay inside that cube. Imagine something like a bounding cube, covering all your data points.
      Example in 2D: you need to display a curve of form y=sin(x) where x is between [-PI] and [+PI]. The result of that function is in range of [-1] up to [+1].
      Therefore you have to define the bounding box (2D!) using two corners:
      (-1, -PI) which is top left and 
      (+1, +PI) similar to bottom right.
    • A flag (bSupportTrackBall) to tell the API whether to perform rotation controlled by mouse or not. 
  3. For each data point to be calculated, call the RenderPoint() function. It returns the corresponding screen(=window) coordinates of your 3D data point.

The best to do is to check 3DHelperDemoSimple project. I reduced the code to the minimum (no checking code) to clarify the steps.
Open the 3DHelperDemoSimple project, and select the OnInitDialog method.

Only the lines below are added to the code generated by the wizard:    

    // Initialize the 3D system
    initialize(); 

The initialize function itself is implemented some lines below.
Steps performed:

  1. Initialize the API.
  2. Start the update timer, used to refresh the mouse (=rotation) information. In our case, the whole scene will be repainted each 25 milliseconds.
// Set up initial 3D system
void CMy3DHelperDemoSimpleDlg::initialize()
{
	CWnd *pTarget = GetDlgItem(IDC_STATIC_PLACEHOLDER);
	m_c3DHelper.Initialize(pTarget->GetDC(), NeHe::Vector(-2.0f, -2.0f, -2.0f), 
		NeHe::Vector(2.0f,2.0f,2.0f), TRUE);

	// Timer used to refresh "trackball" information
	SetTimer(1, 25, NULL);
}

Triggered by the timer, every 25 ms the OnTimer function will be entered.
Inside, the trackball position will be updated (UpdateTrackBall, which again needs the current DC as parameter), and the scene will be redrawn (redraw()).

void CMy3DHelperDemoSimpleDlg::OnTimer(UINT nIDEvent) 
{
	// Timer is used to refresh trackball information
	if(nIDEvent == 1) {
		KillTimer(nIDEvent); // Prevent timer "overlapping"
		m_c3DHelper.UpdateTrackBall
			(GetDlgItem(IDC_STATIC_PLACEHOLDER)->GetDC());
		redraw();
		SetTimer(1, 25, NULL); // Start timer for next redrawing round
	}
	CDialog::OnTimer(nIDEvent);
}

Redraw again is trivial. Just take your data point per point, call RenderPoint() for each of them and display the result using e.g. SetPixel() function.

The 3DHelperDemo project implements a more complex example.

Points of Interest 

I decided not to implement the whole math stuff by myself (think that is not a unique task...). So I have taken some matrix/vector code from NeHe (a GREAT site related to OpenGL, got my 5!).

To get a better idea on how OpenGL is implemented, I took a closer look at the free mesa implementation. This is a must for everyone interested in 3D implementation! 

Notes

Any feedback would be appreciated!

See more tools and updates on SoftwareHive.

History

  • 2009/02/14 - Initial release

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

Share

About the Author

MikeTheDwarf
Business Analyst
Austria Austria
No Biography provided

Comments and Discussions

 
GeneralReading Pinmembermathewmefiu8-Jun-10 23:24 
GeneralRe: Reading PinmemberMikeTheDwarf9-Jun-10 2:14 
GeneralRe: Reading Pinmembermathewmefiu22-Jun-10 22:34 
GeneralRe: Reading PinmemberMikeTheDwarf23-Jun-10 1:59 
Questionuse in VB.net PinmemberGeneralBiSoN28-Feb-09 0:21 
AnswerRe: use in VB.net PinmemberMikeTheDwarf1-Mar-09 22:12 
GeneralRe: use in VB.net [modified] PinmemberGeneralBiSoN4-Mar-09 9:14 
GeneralRe: use in VB.net PinmemberMikeTheDwarf4-Mar-09 9:34 
GeneralRe: use in VB.net PinmemberGeneralBiSoN5-Mar-09 6:16 
GeneralRe: use in VB.net PinmemberMikeTheDwarf5-Mar-09 9:27 
GeneralSounds nice Pinmemberjohn wallis17-Feb-09 12:20 
GeneralRe: Sounds nice PinmemberMikeTheDwarf18-Feb-09 0:47 
GeneralRe: Sounds nice Pinmemberjohn wallis18-Feb-09 13:22 
GeneralRe: Sounds nice Pinmemberjohn wallis18-Feb-09 13:36 
AnswerRe: Sounds nice PinmemberMikeTheDwarf19-Feb-09 7:54 
JokeRe: Sounds nice Pinmemberjohn wallis20-Feb-09 1:34 
GeneralRe: Sounds nice PinmemberMikeTheDwarf20-Feb-09 6:46 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

| Advertise | Privacy | Terms of Use | Mobile
Web02 | 2.8.141216.1 | Last Updated 17 Feb 2009
Article Copyright 2009 by MikeTheDwarf
Everything else Copyright © CodeProject, 1999-2014
Layout: fixed | fluid