sketchucation logo sketchucation
    • Login
    โ„น๏ธ Licensed Extensions | FredoBatch, ElevationProfile, FredoSketch, LayOps, MatSim and Pic2Shape will require license from Sept 1st More Info

    Find the global position of a Vertex in a Group/Component

    Scheduled Pinned Locked Moved Developers' Forum
    25 Posts 6 Posters 4.7k Views 6 Watching
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • J Offline
      jessejames
      last edited by

      @gbabcock said:

      Being new to this, I've been searching around to find a way to get the global position of a vertex when it is in an unopened group, or even worse when it is in a nested group.

      Lets make this simpler and use the term "collection" to refer to both groups and components. There is no difference in finding the "world coordinate" of a vertex contained in a collection whether or not the "collection" is "open" or "closed". If it's nested yes, but only a minor difference in the form of recursion.

      @gbabcock said:

      I found a lot of good info in this forum, and learned that you need the Point3d of the vertex AND the Transformations of the parent Group(s) and/or Component(s). The problem is how to get the parents starting from an edge without walking the entire Group/Component/Model from the top-down.

      You specifically need the "collections" origin which is a point and the point you wish to find. You see whenever a "collection" is created the origin of that "Collection" is equal to the minX, mixY and minZ "parent coordinates" of the bounding box of that collection. You can easily query this origin at any time by using...

      Tim Toady's first method

      collection.transformation.origin
      

      Tim Toady's second method

      collection.transformation.to_a[-1]
      

      ...since a transformation is just a four element array containing the [[Xaxis], [Yaxis], [Zaxis], [origin]] sub arrays.

      @gbabcock said:

      The SU API seems to be very top-down oriented, and it is not intuitive how you walk up the hierarchy. Assuming that this is correct?:

      HIERARCHY
      === Model
      ====== DefinitionList
      ========= ComponentDefinitions
      ============ ComponentInstances #collection of Group or Components Instances
      ============ Entities
      ============ ComponentDefinitions
      =============== ComponentInstances #collection of Group or Components Instances
      =============== Entities

      No thats completely wrong. The definition list is an attribute of MODEL just like ENTITIES is an attribute. If you where to map out a simplistic version of the API it might look like this...

      [warning: loose code ahead!]

      
      class Model(collection)
        self.definitionList = {def1, def2, ..., defN}
        self.entities = {ent1, ent2, ..., entN} # infinite nesting here!
        # lots more stuff here...
      end
      
      class Group
        self.entities = {ent1, ent2, ..., entN} # infinite nesting here!
        # more stuff
      end
      
      class ComponentInstance
        self.entities = {ent1, ent2, ..., entN} # infinite nesting here!
        # more stuff
      end
      
      model = Model()
      model.entites.push(Group())
      model.entites.push(....)
      ....
      
      

      (of course at the C level it's much different!) Model is just a top level "collection" like groups or components. You add all edges, faces, groups, components to the Model entities. and then you can nest infinitely (or until you run out of memory!) the collections in model.entities

      (I'll probably be lambasted for making such a simplistic model ๐Ÿ˜•, oh well!

      Anyhow just remember a vertex inside a collection has a position relative to the collections origin. So as an example... if vertexA is an entity contained in groupA which is a collection contained in the model.entities array then...

      @unknownuser said:

      if vertexA is at [0,0,1] and groupA's origin at [0,0,1] then the world space coordinates of vertexA will be at [0,0,2]

      PS There may be those of you who are wondering who Tim Toady is. Tim Toady is a pronunciation of the acronym TMTOWTDI (or TIMTOWTDI). Basically it describes a paradigm you new Ruby scripters will need to get acquainted with ...More info can be found here.

      PPS hopefully i have managed to shed some light on your dilemma and not simply confused you even more. I would like to go into more details at this time but "crikey" i need some sleep!

      Always sleep with a loaded gun under your pillow!

      1 Reply Last reply Reply Quote 0
      • thomthomT Offline
        thomthom
        last edited by

        @jessejames said:

        There is no difference in finding the "world coordinate" of a vertex contained in a collection whether or not the "collection" is "open" or "closed". If it's nested yes, but only a minor difference in the form of recursion.

        SU changes the co-ordiates when you open a group/component. When a group or component is open SU returns global co-ordinates for the entities in that context.

        Thomas Thomassen โ€” SketchUp Monkey & Coding addict
        List of my plugins and link to the CookieWare fund

        1 Reply Last reply Reply Quote 0
        • thomthomT Offline
          thomthom
          last edited by

          @gbabcock said:

          The problem is how to get the parents starting from an edge without walking the entire Group/Component/Model from the top-down.

          The only time you can avoid that is if you're making a Tool using the PickHelper to pick out the entity. PickHelper also return the path to that entity.

          But if you're not making a point and click tool - then the only way is to traverse the entire model. It's one of the major complaints about the SketchUp API.

          Thomas Thomassen โ€” SketchUp Monkey & Coding addict
          List of my plugins and link to the CookieWare fund

          1 Reply Last reply Reply Quote 0
          • G Offline
            gbabcock
            last edited by

            jessejames, thats not quite what I was after, but thanks for the post!

            thomthom, thanks for the feedback. I'll check out PickHelper in any case, sounds interesting.

            BTW, the code snippet I posted does work, I was testing it last night, and it doesn't traverse the model. The only caveat was that I had to use group.make_unique on copies of groups due to the bug discussed on this forum.

            Glenn

            Glenn

            1 Reply Last reply Reply Quote 0
            • thomthomT Offline
              thomthom
              last edited by

              @gbabcock said:

              BTW, the code snippet I posted does work, I was testing it last night, and it doesn't traverse the model.

              What was your testcase? With group or components? Did you try with scaled instances as well?

              @gbabcock said:

              The only caveat was that I had to use group.make_unique on copies of groups due to the bug discussed on this forum.

              Which one was that? Getting the correct parent for a group?
              http://forums.sketchucation.com/viewtopic.php?f=180&t=19765

              Thomas Thomassen โ€” SketchUp Monkey & Coding addict
              List of my plugins and link to the CookieWare fund

              1 Reply Last reply Reply Quote 0
              • thomthomT Offline
                thomthom
                last edited by

                I'd avoid making groups unique. I some times get models made by others where groups have been used as components. Provided the groups have not been made unique I can still recover them, by using Selection Toys.

                Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                List of my plugins and link to the CookieWare fund

                1 Reply Last reply Reply Quote 0
                • G Offline
                  gbabcock
                  last edited by

                  @thomthom said:

                  What was your testcase? With group or components? Did you try with scaled instances as well?

                  I tested Groups, nested Groups, scaled Groups, rotated Groups, translated Groups, Group copies, Components (single only). Works for all of them.

                  I've updated my Position Explorer tool, see attached. I'll post the update in the Plugins forum.

                  @thomthom said:

                  Which one was that?

                  http://forums.sketchucation.com/viewtopic.php?f=180&t=14062&p=105782

                  Glenn


                  Position Explorer 13 Feb 2010

                  Glenn

                  1 Reply Last reply Reply Quote 0
                  • G Offline
                    gbabcock
                    last edited by

                    @thomthom said:

                    I'd avoid making groups unique. I some times get models made by others where groups have been used as components. Provided the groups have not been made unique I can still recover them, by using Selection Toys.

                    So how do I work around the bug? If I make a copy of a group and immediately try to get it's position I get the original's transformation.

                    Glenn

                    Glenn

                    1 Reply Last reply Reply Quote 0
                    • thomthomT Offline
                      thomthom
                      last edited by

                      @unknownuser said:

                      #returns an array of Point3d objects that equal the global position of the vertices in an edge
                      #currently only gets the first parent instance, components may have many instances

                      You're not dealing with components with many instances? You only pick the first instance?

                      Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                      List of my plugins and link to the CookieWare fund

                      1 Reply Last reply Reply Quote 0
                      • G Offline
                        gbabcock
                        last edited by

                        @thomthom said:

                        You're not dealing with components with many instances? You only pick the first instance?

                        I am, I just haven't gotten that far with the tool yet!

                        Glenn

                        1 Reply Last reply Reply Quote 0
                        • thomthomT Offline
                          thomthom
                          last edited by

                          @gbabcock said:

                          @thomthom said:

                          You're not dealing with components with many instances? You only pick the first instance?

                          I am, I just haven't gotten that far with the tool yet!

                          Well, that is a key major problem. Getting the correct instance. It's why traversing the parent won't work, as you only get the definition, but no way of getting the right instance.
                          If you make components work, then you make groups with group copies work. They are the same thing under the hood.

                          Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                          List of my plugins and link to the CookieWare fund

                          1 Reply Last reply Reply Quote 0
                          • G Offline
                            gbabcock
                            last edited by

                            Thanks, that's helpful. I'll play around with that.

                            Glenn

                            1 Reply Last reply Reply Quote 0
                            • G Offline
                              gbabcock
                              last edited by

                              @thomthom said:

                              Well, that is a key major problem. Getting the correct instance...

                              OK, I see now. It's not a true hierarchy, since an element (vertex, face, edge, etc.) can be in more than one instance. Parent only gets you the Definition, with no way to get the correct Instance, as you said.

                              Thanks for helping me understand!

                              Glenn

                              1 Reply Last reply Reply Quote 0
                              • thomthomT Offline
                                thomthom
                                last edited by

                                Only way is to map the whole model, going from Model and up. Which is just a brute force way to do it. ๐Ÿ˜•

                                Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                                List of my plugins and link to the CookieWare fund

                                1 Reply Last reply Reply Quote 0
                                • J Offline
                                  jessejames
                                  last edited by

                                  @thomthom said:

                                  SU changes the co-ordiates when you open a group/component. When a group or component is open SU returns global co-ordinates for the entities in that context.

                                  It seems Tim Toady is more prevalent that i had originally feared! ๐Ÿ˜ฎ

                                  PS: Before some hot shot notices i made a mistake in accessing the origin using

                                  collection..transformation[-1]
                                  

                                  Since the transformation is a 16 element array an NOT a 4 element array of 4 element sub arrays that code will only return the last float.

                                  Always sleep with a loaded gun under your pillow!

                                  1 Reply Last reply Reply Quote 0
                                  • AdamBA Offline
                                    AdamB
                                    last edited by

                                    @thomthom said:

                                    Only way is to map the whole model, going from Model and up. Which is just a brute force way to do it. ๐Ÿ˜•

                                    Not really true. Its certainly non-trivial but LightUp does it ๐Ÿ˜„. ie LightUp needs to find the transform of every instance of a Component called PointLightSource. Clearly starting at the top and walking down is "one way" buts its incredibly inefficient. Better is to get each instance (trivial) then work your way back up concatenating transforms as you go - and dealing with the fact that there can be many instances of Components as thomthom points out. Took a fair amount of head scratching but its all in lightcache.rb if you want to take a butchers.

                                    Its invoked by list = LightCache.flat(LightCache.walk(ent, ent.transformation)) which gives you back a list of transforms.

                                    Adam

                                    Developer of LightUp Click for website

                                    1 Reply Last reply Reply Quote 0
                                    • thomthomT Offline
                                      thomthom
                                      last edited by

                                      Very interesting Adam. I'm glad to be proven wrong.

                                      I will for sure poke about that code of yours. Is it ok with you if I adapt it to a generic method and add it to my generic library shared between my plugins?

                                      Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                                      List of my plugins and link to the CookieWare fund

                                      1 Reply Last reply Reply Quote 0
                                      • G Offline
                                        gbabcock
                                        last edited by

                                        @adamb said:

                                        Clearly starting at the top and walking down is "one way" buts its incredibly inefficient. Better is to get each instance (trivial) then work your way back up concatenating transforms as you go - and dealing with the fact that there can be many instances of Components as thomthom points out. Adam

                                        Exactly what I'm trying to do! Thanks, Adam, I'll check this out.

                                        Glenn

                                        1 Reply Last reply Reply Quote 0
                                        • G Offline
                                          gbabcock
                                          last edited by

                                          OK, I think I'm getting a clearer picture now. โ— Let me sum it up to make sure.

                                          Since we can have multiple instances of a Group/Component, there is a One-to-Many relationship between a Vertex and Transformations:

                                          Vertex->ComponentDefinition->ComponentInstances->ComponentInstance->Transformation

                                          model.jpg
                                          Therefore, to get the global position of a Vertex you need BOTH the Vertex (which is in ComponentDefinition) AND the specific ComponentInstance you are evaluating. From there you can walk UP the hierarchy of ComponentInstances and get their Transformations to apply to the Vertex.

                                          So in Ruby we have:
                                          Sketchup.active_model #top of model
                                          Sketchup.active_model.definitions #returns DefinitionList
                                          Sketchup.active_model.definitions[n] #returns ComponentDefinition[n] from array
                                          Sketchup.active_model.definitions[n].instances[n] #returns ComponentInstance[n] from array

                                          You can get a quick picture of this in SU with:

                                          Sketchup.active_model.definitions.each {|definition| puts "#{definition} contains #{definition.instances}\n" }
                                          

                                          Glenn

                                          1 Reply Last reply Reply Quote 0
                                          • thomthomT Offline
                                            thomthom
                                            last edited by

                                            Adam, that code returns the transformation of all the copies of that instance? But no way to track back a single path to the model, like you get with PickHelper?

                                            Thomas Thomassen โ€” SketchUp Monkey & Coding addict
                                            List of my plugins and link to the CookieWare fund

                                            1 Reply Last reply Reply Quote 0
                                            • 1
                                            • 2
                                            • 1 / 2
                                            • First post
                                              Last post
                                            Buy SketchPlus
                                            Buy SUbD
                                            Buy WrapR
                                            Buy eBook
                                            Buy Modelur
                                            Buy Vertex Tools
                                            Buy SketchCuisine
                                            Buy FormFonts

                                            Advertisement