Getting Started with the Cinema 4D C++ SDK on macOS

Learn how to build, debug, and extend the C++ SDK on macOS.

Prerequisites

To follow this guide, you need the following things:

  • A copy of the extended Cinema 4D C++ SDK. You can download the SDK from the download section of developers.maxon.net. You must use an extended version of the SDK to follow this guide, the sdk.zip file shipped with Cinema 4D is not sufficient.
  • An installation of Xcode matching the version of the SDK you want to build for. You must use the Xcode version recommend on the Development Requirements pag for the targeted SDK. You can download a wide range of Xcode versions from the xcodereleases.com.
  • An installation of Cinema 4D you want to build the SDK for. You can download a trial version from the Maxon website.
Note
The code and projectdefinitio.txt instructions shown in this guide are identical for all platforms. The only difference between building the SDK on Windows, macOS, or Linux is the way the solution files are generated and how to interact with the IDEs such as Visual Studio, Xcode, or Eclipse.
Warning
Xcode 13 is incompatible with more recent versions of macOS such as Ventura and Sonoma. But you must still use Xcode 13 to build the more recent Cinema 4D C++ SDKs for macOS, as Xcode 13 is the last release that supports the legacy build system. We strongly recommend using macOS Monterey to run Xcode 13 as the circulating hack of reaching into the Xcode 13 package to directly invoke the Xcode app is more and more error prone the further you update macOS. Attempting to build the Cinema 4D SDK with Xcode 14 or later will result in build errors. We are working on supporting the standard build system of Xcode.

Building the SDK

To get started with the C++ SDK, you must first generate a solution using the project tool and then build that solution.

Note
The project tool is a command-line tool that is used to generate the solution files for the Cinema 4D C++ SDK. It is a separate tool from the Cinema 4D application and must be run from the command line. The project tool is located in the tools folder of the SDK. The file generate_solution_osx.command is a batch file that runs the project tool with the correct arguments to generate the solution files for macOS.
Warning
The project tool is the only supported way to generate solutions for the C++ SDK. Custom build configurations or manual changes to the solution files are not supported and can lead to issues when building the SDK.

Running the Project Tool

The Cinema 4D SDK is not shipped with project files for Visual Studio, XCode, or Eclipse. Instead, the project files must be generated with the Project Tool. The project tool is a command-line tool that is used to generate the solution files for the Cinema 4D C++ SDK, reading in projectdefinition.txt files placed in a solution. The project tool is located in the tools folder of the SDK and can also be run manually.

To run the project tool in a more convenient fashion, double click the file generate_solution_osx.command. The script can be used both to create a new solution from scratch or to update an existing solution when source files have been added or removed. The script will open shell, displaying the output of the project tool.

Fig. I: Running the project tool on macOS via the generate_solution_osx.command script will open the shell containing the output of the project tool. There should be no errors in the output of the project tool.

Exploring the Solution

Once the generate_solution_osx.command script has run, a file named solution_osx.command will be present in the SDK folder. The file contains the command to forcefully open the generated solution in /plugins/project/plugins.xcodeproj with the app Xcode found in 'Applications'.

Warning
The script assumes that /Applications/Xcode.app is the path to the Xcode 13 app. If Xcode is installed in a different location, the script must be edited to reflect the correct path. The script will allow you to open the solution in Xcode by double-clicking it; to some degree also for macOS versions beyond Monterey. But the further you update your macOS beyond the supported versions of Xcode 13, the more likely it is that the script will not work anymore as Xcode is then throwing more and more errors when starting (usually related to duplicate or missing GPU libraries).

When your macOS version is strongly incompatible with Xcode 13, you must open the solution manually. To do so, navigate to your Xcode 13 package in the Finder, right-click it, and select Show Package Contents. In the package navigate to Contents/MacOS and double click Xcode file. Xcode will now start and do a lot of complaining along the way but will eventually succeed in booting. In Xcode, open the solution by selecting File > Open and then selecting the plugins/project/plugins.xcodeproj file in the SDK folder.

The solution reflects the structure of the SDK and contains all the projects and files that are part of the SDK. The frameworks folder in the solution contains the content of the frameworks folder of the SDK, here you can find the definition of the Cinema 4D API. A framework bundles a section of the Cinema 4D API that is thematically related, the asset.framework contains for example the API for the underlying data types of the Asset Browser. Frameworks are also used to include aspects of the Cinema 4D API with a project, so that the header files of that framework can be included in a source file of that project.

The modules folder in the solution contains the content of the plugins folder of the SDK. Here you find the code examples of the SDK and can also place your own code and projects. When getting started with the SDK, we recommend focusing on the examples.main project in the modules folder as the other projects are about more advanced topics.

Fig. II: The SDK solution is split into the frameworks and modules folders. The frameworks folder contains the API definition, the modules folder contains the code examples and your own projects.
Warning
Never modify the frameworks of the SDK as your plugin will otherwise become incompatible with Cinema 4D. See also: ABI Compatibility .

Building the Solution

Before you can build the solution, you must ewabled the legacy build system in Xcode. To do so, go to File > Project Settings and select Legacy Build System (deprecated) from the build system dropdown menu, also tick the checkbox for suppressing deprecation warnings. The legacy build system is required to build the Cinema 4D C++ SDK.

Fig. III: You must enable the legacy build system in order to be able to build the Cinema 4D C++ SDK on macOS.

Now we must set the build target to AllTargets in the status bar at the top of Xcode. As the build destination you should choose My Mac for now. When you want to ship a plugin you can choose Any Mac to compile universal binaries that can run both on Intel and Apple Silicon hardware. These settings can also be found in the Product > Scheme (to select a target) and Product > Destination (to select a destination) menus.

Fig. IV: Selecting the scheme and destination from the status bar in Xcode.

You can now build the solution by pressing CMD+B or selecting Build from the Product menu. The SDK now starts building and doing it for the first time can take a while. Switch to the Report Navigator by clicking the icon in the top left of the Xcode window (where the Project Navigator is located) to see the build progress. When the build has finished you can see the message 'Build succeeded' in the Xcode output window. There should not be any failed projects. If there are, the error messages can also be found in the output window. Feel free to ask for help on the Maxon Developer Forum if you encounter any issues.

Fig. V: The Xcode Report Navigator showing a successfully built SDK.

Once the build has finished, you can find the compiled plugins binaries in the respective project folders in the plugins folder of the SDK. Each project, i.e., subfolder in the plugins folder will be compiled into its own binary. The compiled binaries are the plugins that can be loaded into Cinema 4D. On Windows, the compiled plugins are .xdl64 files, on macOS .xlib files, and on Linux .xso64 files.

Fig. VI: The compiled binaries of a solution can be found at the root of each project. Shown is here the output for the 'examples.asset' project. To ship such plugin, one must bundle up its binary and (when present) 'res' folder.

Debugging the SDK

Now that you have built the Cinema 4D C++ SDK, you can test its binaries by debugging them in Xcode. Running Cinema 4D with a debugger attached allows you to set breakpoints in the SDK code, inspect variables, and make use of debugging features of the Cinema 4D API.

Configuring the Solution

To debug binaries built with the Cinema 4D C++ SDK, you must set the debug application in your build target. The debug application is the Cinema 4D executable that will be started when you start debugging. You must also set the command arguments of the debug application to load the Cinema 4D C++ SDK binaries.

Note
You can always debug a plugin binary with a Cinema 4D version that exactly matches the SDK version the plugin has been compiled with. To learn more about the compatibility between compiled plugin binaries and different Cinema 4D versions, see the ABI Compatibility .

To set these values, open the scheme for the build target by opening Product > Scheme > Edit Scheme. In the scheme editor, select the Run section and set the Executable to the Cinema 4D executable that matches the used SDK.

Fig. VII: Setting a matching Cinema 4D executable as the debug application in the scheme editor of Xcode.

Then switch to the Arguments tab and set the Arguments Passed On Launch to the string shown below which will tell Cinema 4D to load all plugins in the sdk/plugins/ folder. Alternatively, you could also set the plugin path in the preferences of that Cinema 4D installation.

// You must hardcode the path to the SDK plugins folder in the command arguments.
g_additionalModulePath=/path/to/SDK/plugins/
// This does not work at the moment in Xcode.
g_additionalModulePath=$(SRCROOT)/../
Fig. VIII: Setting the command line argument for Cinema 4D to load the SDK plugins in the scheme editor of Xcode. Unlike on Windows, you must hardcode the path to the SDK plugins folder in the command arguments.

Attaching the Debugger

Now you can press CMD+R or the grey 'Play' button in the toolbar of Xcode to start the debugger. The debugger will boot Cinema 4D which will also load your plugin. Once Cinema 4D has started, you can check its Extensions menu to see if the C++ SDK has been loaded.

Fig. VIV: When you have done everything correctly, the C++ SDK will be loaded into Cinema 4D and you see it in the Extensions menu. Congratulations, you have successfully built the Cinema 4D C++ SDK!

Without closing Cinema 4D and without stopping the debugger in Xcode, you now return to Xcode. There you navigate to the file modules/example.main/source/object/roundedtube.cpp in the Project Navigator and open it by double-clicking it. In the file, you open the search by pressing CMD+F and search for the definition of the method RoundedTubeData::GetVirtualObjects. Here you set a breakpoint by left-clicking the line number of one of the lines of the method. The breakpoint should lie within the scope, the curly braces, of the method. The line should now be marked by a blue arrow.

Fig. X: Setting a breakpoint in the RoundedTubeData::GetVirtualObjects() method while the debugger is running. You can also set a breakpoint when the debugger is not running, we just do it here in one go so that you do not have to reboot Cinema 4D between debugging the SDK for the first time and setting your first breakpoint.

Back in Cinema 4D, you can now invoke the Rounded Tube code example from the Commander by pressing SHIFT+C and typing Rounded Tube and then hitting Enter. Cinema 4D will now create an instance of the RoundedTubeData object which will try to build its geometry cache. The debugger will halt on your breakpoint because you have set it in RoundedTubeData::GetVirtualObjects(), the method which is responsible for building the geometry of that plugin.

You can now step through the code by pressing F6 to step over the current line or F7 to step into the current line. You can also inspect variables by hovering over them with your mouse or by adding them to the watch window by right-clicking them and selecting Add Expression.

Fig. XI: Your breakpoint is being triggered, you can see the values of variables of the current scope in the 'Auto' window at the bottom in the middle. You can also add variables to the 'Watch' window on the right by right-clicking them and selecting 'Add Expression'. On the right to the 'Auto' window you see the debug output window of Xcode. On the far right, where usually the Project Navigator is located, you see the stack trace of the current halt.

Advanced Techniques

You can now stop the debugger by pressing the stop button in the toolbar in the top left of Xcode. We will now learn how to print (debug) messages in the Cinema 4D API to its various consoles and how to invoke debug stops from code.

To do that, first disable your breakpoint by again clicking on the blue arrow in the the code editor of the modules/example.main/source/object/roundedtube.cpp file. Then insert the code shown below into the RoundedTubeData::GetVirtualObjects the file so that the snip marks line up with the existing code.

// Existing code ...
if (!dirty)
return op->GetCache();
// --- Snip -----------------------------------------------------------------------------
// Prints to the console of Cinema 4D, the debug console of Xcode, and
// the maxon console of Cinema 4D (when attached).
ApplicationOutput("Application: @", op->GetName());
// Prints to the debug console of Xcode and the maxon console of Cinema 4D
// (when attached).
DiagnosticOutput("Diagnostic: @ (dirty = @)", op->GetName(), dirty);
// Halts like a breakpoint when debugging a debug config binary.
if (true)
DebugStop("This should not happen!"_s);
// Halts like a breakpoint even when debugging a release config binary.
if (true)
CriticalStop("Upsy-daisy!"_s);
// --- Snip -----------------------------------------------------------------------------
// Also existing code ...
BaseContainer* data = op->GetDataInstance();
#define DiagnosticOutput(formatString,...)
Definition: debugdiagnostics.h:170
#define ApplicationOutput(formatString,...)
Definition: debugdiagnostics.h:204
#define DebugStop(...)
Definition: debugdiagnostics.h:225
#define CriticalStop(...)
Definition: debugdiagnostics.h:231
PyObject * op
Definition: object.h:520

Now rebuild the solution by pressing CMD+B or selecting Build from the Product menu. When the build has finished, start the debugger by pressing CMD+R or the grey 'Play' button in the toolbar of Xcode. In Cinema 4D, create again an instance of the Rounded Tube object. The debugger will now halt on its own on the first manual halt in your code, the DebugStop, and then on the second manual halt, the CriticalStop. Press CTRL+CMD+Y to continue the execution of the program after each halt.

Fig. XII: The debugger ist halting on the second manual halt in your code, the CriticalStop. In the lower right corner you see the Xcode output window which shows the output of the ApplicationOutput and DiagnosticOutput calls, as well as the messages of both DebugStop and CriticalStop. The ApplicationOutput call is also shown in the Cinema 4D console, superimposed in the bottom left corner.
Note
To see the output of the ApplicationOutput calls in the Cinema 4D console, you must have the Cinema 4D console open. You can open the Cinema 4D console by selecting Extensions > Console from the Cinema 4D menu. To see output in the maxon console, you must run Cinema 4D with the command line argument g_console=true or g_consoleDebugger=true. The former will only show the Cinema 4D console when no debugger is attached, the latter will always show the Cinema 4D console.

ApplicationOutput is a relatively expensive call and you should avoid it in production code for everything but critical messages which are meant to be seen by the end-user. When you spam the Cinema 4D console with messages, you not only render it unusable for others but also eat up considerable system resources as rendering the console takes time. DiagnosticOutput is a less expensive call than ApplicationOutput and should be used for messages that are meant for developers and not for end-users.

There are more output functions available in the Cinema 4D API, see the Console Output for more information. The Output functions all support the maxon string output syntax which is described in the Output Syntax page. The general idea is that the \@ character is used to insert variables into the string and you can pass as many variables as you want to these functions as they are all variadic functions.

DebugStop and CriticalStop are used to halt the execution of the program just like a breakpoint would. DebugStop will only halt the program when it is being debugged in a debug configuration, CriticalStop will halt the program in all configurations. These functions can be useful to bake breakpoints into your code to halt a debugger when something truly critical happens that you did not expect to happen. For end-users these functions will not do anything, they are only active when the program is being debugged.

But you might also run into these two functions without using them yourself as they are often used in the Cinema 4D frameworks to halt the program when something unexpected happens. When you hit a DebugStop or worse, a CriticalStop, in the frameworks while debugging your plugin, it usually means that you either do something truly wrong or have found a bug in Cinema 4D. Most of our debug stops unfortunately are not very informative and do not have an error message attached. But you can look at the framework code to get a sense of what is going wrong. When you hit a wall in such cases, do not hesitate to ask for help on the Maxon Developer Forum.

Extending the SDK

Now that you have built and debugged the Cinema 4D C++ SDK, it is time to add your first own plugin.

Note
This section focuses on all the boilerplate code and steps necessary to get a new plugin up and running. It is not about how to write a specific plugin type but will contain code for your first minimal plugin. Please refer to the code examples for instructions on how to implement specific plugin types.

To add a new plugin to the SDK, you must either create a new project in the SDK solution or create a new solution with a new project. You technically could setup shop in one of the existing SDK projects, e.g., the examples.asset project, but this would mean you would have to ship your plugin with the binary of the examples.asset project. You therefore need at least one custom project to ship your plugin(s). Within such project you can have multiple plugins, so that you can ship multiple plugins with one binary.

Adding the Project

First you must create a new project folder in the plugins folder of your solution. The name of the new project folder is irrelevant, but it is sensible to name it after your plugin, e.g., myplugin. In this folder, you need to create a project folder and in that folder a projectdefinition.txt file. You also need a source folder in the project folder and in that folder a main.cpp file.

Fig. XIII: The two folders and two files, the projectdefinition.txt and main.cpp files, all projects require to be functional. The projectdefinition.txt file is used by the project tool to generate the solution files, the main.cpp file contains (parts of) the code for the plugin.

Setting up the Project

Now you must fill the projectdefinition.txt file of your new project with instructions for the project tool. The content of your projectdefinition.txt file in root/plugins/myplugin/project should look like as shown below:

// The platforms this project is targeting - can be [Win64;OSX;Linux].
Platform=Win64;OSX;Linux
// Type of project - can be [Lib;DLL;Solution].
Type=DLL
// The frameworks dependencies of this project as shipped in the frameworks folder of an SDK. You
// can only include header files from frameworks that are listed here. This plugin uses only the
// Cinema API frameworks cinema.framework and cinema_hybrid.framework and the Maxon API framework
// core.framework. This means for example that you could not use the Nodes or Asset API. But for
// getting started these three frameworks are enough.
APIS=\
cinema.framework;\
cinema_hybrid.framework;\
core.framework
// If this is a Cinema API plugin, enabling some Cinema API features and disabling automatic style
// checks.
C4D=true
// Sets the rigidity of the style check. The minimum is 0 (checks disabled), the maximum is 3 (very
// strict).
stylecheck.level=3
// The ID of the module/plugin which is being compiled. You should use here the "net.mycompany.xyz"
// format. You cannot use a module ID of the format "net.maxon.xyz" as this domain is reserved
// for Maxon and plugins ignoring this will not be loaded.
ModuleId=net.maxonexample.myplugin
OSX
OS X.
Definition: ge_prepass.h:1
C4D
Definition: lib_net.h:1

The projectdefinition.txt in a project folder is used to describe the project to the project tool. The project tool will generate the solution files for the project based on the information in the projectdefinition.txt file. The projectdefinition.txt file must contain at least the following information:

  • The platforms the project is targeting, e.g., Platform=Win64;OSX;Linux.
  • The type of the project, usually Type=DLL.
  • The frameworks the project uses, e.g., APIS=cinema.framework;cinema_hybrid.framework;core.framework.
  • The module identifier of the project, e.g., ModuleId=net.maxonexample.myplugin.

To learn more about projectdefinition.txt files, see the Project Tool page.

Warning
You cannot use module identifiers in the domain "net.maxon." as they are reserved for Maxon Computer. Plugins using such identifiers will not be loaded by Cinema 4D.

The APIS argument determines which frameworks are included in your project and therefore which header files of the Cinema 4D API you can use in your project. As a third party developer, you can include all frameworks - or at least a broad selection as shown in the examples.main project - with little to no downsides for your project.

But as a beginner you will usually only need the cinema.framework, cinema_hybrid.framework, and core.framework frameworks. The cinema.framework contains most parts of the Cinema API of Cinema 4D, the cinema_hybrid.framework contains parts of the Cinema API and the Maxon API, and the core.framework contains important foundational aspects of the Maxon API. The cinema.framework is the most important framework for you as a beginner as it contains most of the Cinema API of Cinema 4D and with it all the plugin interfaces that are interesting for beginners. The Maxon API will only play a minor role in your first steps without you missing out on anything. See Plugin Types for an overview of the different plugins types.

Note
You will encounter the terms Cinema API and Maxon API often in the Cinema 4D SDK. The Cinema API is the API that has been around since the inception of Cinema 4D and is the API that is used by most plugins. The Maxon API is a new API that is being introduced with Cinema 4D R20. Both APIs coexist and while there is some overlap between them, they mostly follow different paradigms and are used for different purposes. The Cinema API is written in mainly very accessible old-school C++ - pointers everywhere, strong focus on messages, no smart pointers, no templates, no lambdas, almost no automated memory handling. The Maxon API is written in modern C++ and uses features such as templates, smart pointers, and lambdas. The Maxon API has also a custom error handling system replacing C++ exceptions and an automated memory management system.

Defining the Solution

Now you must now decide whether you want to add your plugin to the SDK solution or create your own and new solution. The former is the recommended way as it allows you to have all the code examples of the SDK next to your plugin and also has not real disadvantages except for a bit cluttered working environment. The latter is the way to go when you do not really need the SDK code examples all the time and prefer a clean working environment.

In both cases you must edit the projectdefinition.txt file in the plugins/project folder of the SDK. This projectdefinition.txt file at the root of the plugins folder contains a list of all projects that are part of the solution. You must either add the name of your plugin folder to the Solution list or you must remove all projects from the projectdefinition.txt file except for your own (and optionally delete all other project folders in the plugins folder).

The content of the projectdefinition.txt file in the plugins/project folder for extending the SDK solution should look like as shown below:

// The platforms this solution is targeting - can be [Win64;OSX;Linux].
Platform=Win64;OSX;Linux
// The type of this definition, a solution.
Type=Solution
// The plugin projects that are part of this solution. Each of them will be compiled into its own
// binary and corresponds to a folder in /plugins/, each with its own projectdefinition.txt file.
Solution=\
plugins/example.main;\
plugins/example.migration_2024;\
plugins/example.assets;\
plugins/example.nodes;\
plugins/example.image;\
plugins/myplugin

The solution definition for a new solution would look similar to the one shown above, but with only your plugin in the Solution list.

// The platforms this solution is targeting - can be [Win64;OSX;Linux].
Platform=Win64;OSX;Linux
// The type of this definition, a solution.
Type=Solution
// The plugin projects that are part of this solution. Each of them will be compiled into its own
// binary and corresponds to a folder in /plugins/, each with its own projectdefinition.txt file.
Solution=\
plugins/myplugin

Now that you have taken these steps, you can run the generate_solution_osx.command script again to update the solution and project files. When you still had the solution open in Xcode, Xcode will reload the solution on its own. Otherwise you can double-click the symbolic link solution_osx.command in the SDK folder to open the updated solution in Xcode.

Note
It is strongly recommended to always add or remove files in a solution in the manner as shown in this guide by manually creating them in the place they should be placed in and then running the generate_solution_osx.command script. When you add files using the tools of Xcode, it will not place them in the source folder. Your project will still compile but your solution will be different from a pure project tool solution.
Fig. XIV: The updated solution in Xcode after adding a new project to the SDK solution. The solution should contain your myproject project and the main.cpp file of your plugin.

Writing the Plugin

Now it is time to finally fill your main.cpp file with the code of your plugin. This section is mostly about the boilerplate code you need to get a plugin up and running and uses the minimal code necessary to implement a plugin. The plugin will be a simple command that opens a message dialog with the text "Hello World!" when invoked.

Note
This section demonstrates how to implement Cinema API plugins which make up the vast majority of plugin types such as commands, tools, messages handlers, scene hooks, objects, tags, (Cinema API legacy) materials, (Cinema API legacy) shaders and more. Plugins in the Maxon API are implemented entirely differently and are not covered in this section. The Maxon API primarily covers plugins that implement custom assets for the Asset API and plugins that implement custom node spaces, node systems, and node templates for the Node API. See Plugin Types for an overview of the different plugins types.

Most plugins also split up their code into multiple files, the main.cpp file and one or multiple files that implement the actual plugin. The main.cpp is usually only used to handle plugin messages sent to the module the main.cpp is the entry point of, while the actual plugin code is implemented in other files. This example condenses all the code into the main.cpp file for simplicity.

The content of the main.cpp file in the plugins/myplugin/project/source folder should look like as shown below:

/* Realizes a simple command plugin that opens a message dialog with the text "Hello World!" once it
has been invoked.
The main focus of this example is to get the plugin message infrastructure up and running that is
required to register all Cinema API plugins.
*/
// -------------------------------------------------------------------------------------------------
// This section contains the code that is usually placed in the main.cpp file of a project. It is
// custom to separate the a project into a main.cpp file - which handles all the plugin messages
// for its plugins - and one or multiple files which realize the actual plugins.
// -------------------------------------------------------------------------------------------------
#include "c4d_plugin.h"
#include "c4d_resource.h"
// Forward declaration of the the #RegisterMyPlugin() function we implement below.
bool RegisterMyPlugin();
// Called by Cinema 4D in its boot sequence when plugins are about to be loaded. This is the entry
// point to register all Cinema API plugins. A plugin which does not register itself or whose
// registration failed will not be accessible to users as it will not be loaded.
{
// Attempt to register our plugin.
if (!RegisterMyPlugin())
return false;
return true;
}
// Called by Cinema 4D in its shutdown sequence when plugins are about to be unloaded. Most plugins
// use the empty implementation shown here as they do not have to do any cleanup.
void PluginEnd()
{
}
// Called by Cinema 4D when a plugin message is sent to our module. There are several messages but
// all plugins must implement at least C4DPL_INIT_SYS as shown below. Otherwise the module and its
// plugins will not work.
Bool PluginMessage(::Int32 id, void* data)
{
switch (id)
{
// g_resource is a global variable that is used to manage resources in Cinema 4D. It is
// defined in the c4d_resource.h file which we include above. We cannot start our module
// if the resource management has not been initialized.
{
if (!g_resource.Init())
return false;
return true;
}
}
return false;
}
// -------------------------------------------------------------------------------------------------
// This section contains the code that is usually placed in a myplugin.cpp and/or myplugin.h file.
// It contains the actual plugin implementation and usually also the registration function for the
// plugin as that function must have access to the plugin class.
// -------------------------------------------------------------------------------------------------
#include "c4d.h"
#include "c4d_gui.h"
// Realizes the most basic plugin possible, a command that opens a message dialog with the text
// "Hello World!" once it has been invoked.
class MyPluginCommand : public CommandData
{
public:
// Called by Cinema 4D when this command is being invoked, either be a user pressing a menu item,
// a button, or by a CallCommand() call made from C++ or Python code.
virtual Bool Execute(BaseDocument* doc, GeDialog* parentManager)
{
MessageDialog("Hello World!"_s);
return true;
}
};
// The ID to register the plugin with. This ID must be unique to this registration call and new IDs
// must be obtained from https://developers.maxon.net/forum/pid .
const Int32 g_plugin_id = 1063013;
// Calls the registration function for a CommandData plugin with an instance of our command class.
Bool RegisterMyPlugin()
{
g_plugin_id, "MyPlugin Menu Label"_s, 0, nullptr, "MyPlugin Tooltip"_s,
NewObjClear(MyPluginCommand));
}
#define C4DPL_INIT_SYS
Initialize system.
Definition: c4d_plugin.h:27
void PluginEnd()
Bool PluginMessage(Int32 id, void *data)
maxon::Bool Bool
Definition: ge_sys_math.h:46
void MessageDialog(const maxon::String &str)
GeResource g_resource
Global resources for Cinema 4D.
maxon::Int32 Int32
Definition: ge_sys_math.h:51
Bool PluginStart()
Bool RegisterCommandPlugin(Int32 id, const maxon::String &str, Int32 info, BaseBitmap *icon, const maxon::String &help, CommandData *dat)
const char * doc
Definition: pyerrors.h:226

The important aspects of getting a module and plugin up an running are:

  • The PluginStart function is the entry point for registering all Cinema API plugins. A plugin that does not register itself or whose registration fails will not be accessible to users as it will not be loaded. This function is usually implemented in the main.cpp file of a project.
  • The PluginEnd function is called by Cinema 4D when plugins are about to be unloaded. Most plugins use the empty implementation shown here as they do not have to do any manual cleanup. This function is usually implemented in the main.cpp file of a project.
  • The PluginMessage function is called by Cinema 4D when a plugin message is sent to the module. There are several messages but all plugins must implement at least C4DPL_INIT_SYS as shown here. Otherwise, the module and its plugins will not work. This function is usually implemented in the main.cpp file of a project. See Plugin Messages for more information on the plugin message system.
  • All plugins must register themselves with Cinema 4D by using the specific registration functions intended for their plugin type. In this case, the RegisterCommandPlugin function is used to register a command plugin. One or multiple of these registration function calls are then usually bundled up in a RegisterMyPlugin() function that is called from PluginStart() in the module. The RegisterMyPlugin() function is often implemented in one of the plugin files, e.g., myplugin.cpp and is then 'included' in the main.cpp file as a forward declaration.

The arguments passed to plugin registration functions such as RegisterCommandPlugin differs from plugin type to plugin type but there are some commonalties:

  • The first argument is the identifier of the plugin. This identifier must be unique and new identifiers must be obtained from developers.maxon.net.
  • Passed are so usually also a plugin label, a tooltip, and some form of icon (there some plugin types which do not have an icon or tooltip).
  • And finally either an instance of the plugin class or a function that returns an instance of the plugin class. The plugin class must inherit from a specific plugin hook of the Cinema 4D API, e.g., CommandData for a command plugin.
Warning
You can 'invent' plugin identifiers for a plugins you use only for testing purposes. But when you want to ship a plugin or plan to uses it for a prolonged time, you must obtain a plugin identifier from developers.maxon.net. Otherwise you risk identifier conflicts with other plugins which will lead to one of the plugins not being loaded.

To run your plugin, you must now rebuild the solution by pressing F7. Make sure that you have no build errors and then start the debugger by pressing F5. In Cinema 4D, you can now find the plugin as Extensions > MyPlugin Menu Label. When you click the item, a message dialog with the text "Hello World!" will open.

Fig. XV: The plugin is being invoked from the Extensions menu.

Further Reading

  • Plugin Types : Describes the different types of plugins that can be created with the Cinema 4D C++ SDK.
  • Frameworks Overview : Describes the frameworks that are part of the Cinema 4D C++ SDK.
  • Project Tool : Describes the project tool that is used to generate the solution files for the Cinema 4D C++ SDK.
  • Source Processor : Describes the source processor that is used to preprocess the Cinema 4D C++ SDK source files.