Mannually set hair root [SOLVED]

  • On 03/01/2017 at 11:25, xxxxxxxx wrote:

    Is there a way for setting manually guide to and polygon id? I know I can do it automaticly with root => set root but that not allow me do to a very presize work.

    Moreover is there a way for generate very precise hair (I guess saying each vertex of each hair)? For the moment the only solution I have found is to draw guide and generate hair as guide which is really not optimized for animation purpose.

    Just a preview here is in 3ds max rendered with vray, generated with ornatrix

    And here the file exported as oxh(ornatrix export format) in c4d render with arnold / default alshader and skydome using each hair as guide and then doing a root => set root. But as said it's very not suitable for animation I would really love to be able to do create a hair in polygonID xx and then set his parented root to rootID xx.

    Again I'm not sticky to python. But I would prefer.
    And happy new year to every one have a nice year with a lot of plugins !

    Thanks in advance :)

  • On 06/01/2017 at 14:12, xxxxxxxx wrote:

    Hi gr4ph0s, thanks for writing us.
    I've investigated your question and looking between the HairGuides() and the HairObject() classes I might have found something useful for your needs. I actually suggest you to have a look at the Cinema 4D SDK Hair Generator example which might give you some hints where to look for.
    In addition find a short snippet below showing how to create hair from scratch on a generic PolygonObject at its vertices.
    At the moment the code is in C++ since on Python it seems there are some issue I still need to address.

    Best, Riccardo

        if (!doc)
            return false;
        Int32 hairSegs = 12;
        Int32 hairCount = 1000;
        // get the selected object
        BaseObject* activeObj = doc->GetActiveObject();
        if (!activeObj || !activeObj->IsInstanceOf(Opolygon))
            return true;
        // cast to HairObject
        HairObject* hairObj = HairObject::Alloc();
        if (!hairObj)
            return true;
        BaseContainer* hairBC = hairObj->GetDataInstance();
        if (hairBC)
            hairBC->SetLink(HAIRSTYLE_LINK,(C4DAtomGoal* ) activeObj);
            hairBC->SetInt32(HAIRSTYLE_HAIR_COUNT, hairCount);
        // create material
        BaseMaterial* hairMat = BaseMaterial::Alloc(Mhair);
        // create the texture tag and assign to hair object
        TextureTag* ttag = TextureTag::Alloc();
        // get some information to build the new hair guides
        PolygonObject* activePolyObj = ToPoly(activeObj);
        const Vector* activePolyPoints = activePolyObj->GetPointR();
        const CPolygon* activePolyPolygons = activePolyObj->GetPolygonR();
        Int32 activePolyPolygonsCnt = activePolyObj->GetPolygonCount();
        Int32 activePolyPointsCnt = activePolyObj->GetPointCount();
        // define the hair segs and allocate the guide
        HairGuides* customHairGuide = HairGuides::Alloc(activePolyPointsCnt, hairSegs);
        if (!customHairGuide)
            return false;
        // retrieve the pointer to the new guides points
        Vector* customHairGuidePoints = customHairGuide->GetPoints();
        Int32 customHairGuideGuidePointsCnt = customHairGuide->GetGuidePointCount();
        Random rng; rng.Init(SAFEINT32(GeGetMilliSeconds()/1000));
        // create a guide at each vertex of the root object
        for (Int32 i = 0; i < activePolyPolygonsCnt; i++)
            CPolygon poly = activePolyPolygons[i];
            Int32 vtxCnt = poly.IsTriangle() ? 3 : 4;
            for (Int32 k = 0; k < vtxCnt; k++)
                Vector pos = activePolyPoints[poly.GetPoint(k)];
                HairRootData rootData;
                rootData.m_ID = poly.GetPoint(k);
                rootData.m_Type = HAIR_ROOT_TYPE_VERTEX;
                rootData.m_N = CalcFaceNormal(activePolyPoints, poly);
                rootData.m_P = pos;
                // increase y and add some noise to x and z at each point of the hair guide
                for (Int32 j = 0; j < customHairGuideGuidePointsCnt; j++)
                    pos.x += j*2*rng.Get11();
                    pos.y += j*3;
                    pos.z += j*2*rng.Get11();
                    customHairGuidePoints[poly.GetPoint(k) * customHairGuideGuidePointsCnt + j] = pos;
                    // set the hair root
                    if (customHairGuide->GetRoot(poly.GetPoint(k)).m_Type == -1)
                        customHairGuide->SetRoot(poly.GetPoint(k), rootData, false);
        // set the new hair guide
        hairObj->SetGuides(customHairGuide, false);
        // update the hair object
        // add the hair object to the scene
        doc->InsertObject(hairObj, nullptr, nullptr);
        return true;


  • On 06/01/2017 at 15:33, xxxxxxxx wrote:

    Firstly thanks you for your exemple.

    I got few questions about it.

    1. For root placement ornatrix use thoses structures
    struct OrnaExpRootTM
    	float tm[4][3];				///< 4x3 transform matrix relative to object space
    struct OrnaExpSurfDep
    	unsigned long face;			///< index into mesh face array: face this root is placed on
    	float barycentricCoordinate[3];				///< barycentric coordinates of this root on face

    > So for simulating this I need to understand a bit more clearly HairRootData parameters.
    > m_ID, is the root id?
    > m_Type (in my case it will be HAIR_ROOT_TYPE_POLY)
    > m_S / m_T I acutually don't know thoses utilities.
    > m_P is the position. But global position? relative to the object? Relative to polygon?
    > m_N is the normal.
    > 2. Maybe it was not clear in my previous post but is it possible to make something like my screenshot.
    > 1 guide (which is placed where I want regarding my first ask and xx hairs linked to the guide (so to the polygon?) which are placed / lined as I want (doing like you have done in your script but instead for doing it for the guide, doing it for each hairs).
    > Regarding the SDK it seem to not be possible since I was unable to find class related to hair but about the hairObject.
    > 3. I don't really understand the function of SetGuide what this function really do
    And another ask a litlle bit out of context.
    It might be stupid but for executing your code I have register a command plugin is there another way to run C++ directly on the fly? (think not since it's a compilated language but that cost nothing to ask :p)

    Anyway thanks you alot for all the informations I understand a bit more how hair are managed by c4d for now (before I always missunderstand root / guide / hair).

  • On 10/01/2017 at 06:40, xxxxxxxx wrote:

    Hi gr4ph0s, thanks for providing your further comments.

    With reference to your question, i prefer to answer point-by-point:

    • m_ID is the id of the geometrical element the root is bind to: in case of a guide rooted on a polygon it's the polygon index of the face used to let the guide grow, in case of a guide rooted on a vertex is the vertex index
    • m_Type is the type of root you'r going to use: check the available values here
    • m_S / m_T are the s and t parametric coordinates used to locate the guide on a polygon: if you have a face you're allowed by specifying those two values where the guide is effectively located on that polygonal face; these two values are useless when guides are rooted to a vertex
    • m_P is the root position, it's normally expressed in local coordinates with respect to the object; consider that this data is not required when m_S /m_T are specified on guide rooted to a polygon (m_type == HAIR_ROOT_TYPE_POLY)
    • m_N is the root normal;
    1. yes it's indeed possible, simply define one single guide rooted on the polygon, set the number of hair to the desired number and you're done;

    2. the SetGuide() method is used to set the guides you want your hair object to use: as you can see in my example i allocate an HairObject, then i allocate an HairGuides object, then i specify the points of the guides and in the end, by using the SetGuide() method, I set my HairObject object to create hair based on my HairGuides object just created;

    3. no it's not possible. C++ is a compiled language which can't be executed as an interpreted language in a on-the-fly fashion.

    Best, Riccardo

  • On 11/01/2017 at 02:16, xxxxxxxx wrote:

    Thanks you Riccardo for all the informations.
    If I still got some questions I will open new thread in C++ section.

Log in to reply