Ticket Q450710
Visible to All Users

How to implement a many-to-many relationship with an intermediate table

created 12 years ago

Hello,
I have an m-n relationship between two tables: References----------Subcontractors . There is an intermediate table T_RelReferenceSubcontractor with a compound key.

I'd like to show in the Reference DetailView a list of the sub contractors for a specific Reference. I've been able to acomplish this but, also I need to link new subcontractors to the Reference. How can I do this?

Thanks in advance & regards,
Juan

Answers approved by DevExpress Support

created 12 years ago (modified 5 years ago)

Hello Juan,

This task has multiple solutions, depending on your tables structure. The guide below will help you choose one that is appropriate for your scenario. Please feel free to contact us if you have any questions.

Default Mapping

XPO automatically creates an intermediate table when a many-to-many relationship is declared. Check whether this table's structure matches your intermediate table. For example, if you implement the Person.Tasks and Task.Persons collections, XPO will generate the following table:

PersonTasks_TaskPersons

Column Type Constraint
OID guid Primary Key
Tasks Task key type Foreign Key to Task.Oid
Persons Person key type Foreign Key to Person.Oid
OptimisticLockField int

Data Model Settings

To change an intermediate table name, use the Association attribute's name parameter and set the AssociationAttribute.UseAssociationNameAsIntermediateTableName parameter to true. To change its column names, change the names of the corresponding collection properties.

Customize Metadata

It is possible to access the intermediate class metadata and change its mapping settings dynamically in code. Please see an example in the following article: How to change the name of the underlying table and columns for a many-to-many relationship. To apply this solution in XAF, use the ModuleBase.CustomizeTypesInfo and XpoTypesInfoHelper.GetXpoTypeInfoSource methods:

C#
using DevExpress.Xpo; using DevExpress.Xpo.Metadata; using DevExpress.ExpressApp.Xpo; namespace MainDemo.Module { public sealed partial class MainDemoModule : ModuleBase { public override void CustomizeTypesInfo(ITypesInfo typesInfo) { base.CustomizeTypesInfo(typesInfo); ... XPDictionary xpoDictionary = XpoTypesInfoHelper.GetXpoTypeInfoSource().XPDictionary; XPClassInfo intermediateClassInfo = xpoDictionary.GetClassInfo(typeof(Task)).FindMember(nameof(Task.Persons)).IntermediateClass; //rename PersonTasks_TaskPersons.Persons to PersonTasks_TaskPersons.PersonId intermediateClassInfo.FindMember(nameof(Task.Persons)).AddAttribute(new PersistentAttribute("PersonId")); }

Please refer to the following help topic for additional information: Use Metadata to Customize Business Classes Dynamically.

Use an Intermediate Class

If your intermediate table cannot be converted according to this structure, implement a persistent class mapped to the intermediate table and associate this class with classes that should be associated as many-to-many using one-to-many relationships. Optionally, use the ManyToManyAlias attribute to provide access to related objects directly, without using the intermediate class.
The ManyToManyAlias attribute accepts two parameters:
oneToManyCollectionName - specifies the one-to-many relationship's collection property.
referenceInTheIntermediateTableName - specifies the reference property whose values should compose the ManyToManyAlias collection. The reference property type should match the collection element type.

Here is an example:

C#
public class User : BaseObject { public User(Session session) : base(session) { } private string _FullName; public string FullName { get { return _FullName; } set { SetPropertyValue("FullName", ref _FullName, value); } } [Association("User-GroupLinks"), Browsable(false)] public IList<UserToGroupLink> UserToGroupLinks { get { return GetList<UserToGroupLink>("UserToGroupLinks"); } } [ManyToManyAlias("UserToGroupLinks", "LinkGroup")] public IList<Group> Groups { get { return GetList<Group>("Groups"); } } } public class Group : BaseObject { public Group(Session session) : base(session) { } private string _Name; public string Name { get { return _Name; } set { SetPropertyValue("Name", ref _Name, value); } } [Association("Group-UserLinks"), Browsable(false)] public IList<UserToGroupLink> UserToGroupLinks { get { return GetList<UserToGroupLink>("UserToGroupLinks"); } } [ManyToManyAlias("UserToGroupLinks", "LinkUser")] public IList<User> Members { get { return GetList<User>("Members"); } } } public class UserToGroupLink : BaseObject { public UserToGroupLink(Session session) : base(session) { } private User _LinkUser; [Association("User-GroupLinks")] public User LinkUser { get { return _LinkUser; } set { SetPropertyValue("LinkUser", ref _LinkUser, value); } } private Group _LinkGroup; [Association("Group-UserLinks")] public Group LinkGroup { get { return _LinkGroup; } set { SetPropertyValue("LinkGroup", ref _LinkGroup, value); } } }

As a result, the associated Members will be shown directly in the Group DetailView, and the Link dialog will show a User lookup instead of a lookup of the intermediate class (UserToGroupLink).

If you have a legacy intermediate table with a composite key, use the following approach to map a persistent class to it without creating an additional key column: What is the best solution for mapping a many-to-many relation to an existing database?

    Show previous comments (7)

      Sort of. Basically, it sounds like implementing the user friendly UI won't allow any custom properties in the intermediate table to be exposed in the listview.  I understand that the main functionality of the intermediate table is still maintained.  I know I can still implement the "non-friendly" UI which will allow the intermediate table to work like a normal one to many  (in other words, link/unlink behavior not there) but then I have inconsistency between the user-friendly many-to-many and the ones where I need additional properties.  I may try using an action along with conditional appearance to keep the user friendly UI but allow the user to still be able to mark one shipping address as the primary via the list view.  Does this sound like a good approach if I can't get the boolean from my intermediate table on the listview?

      Anatol (DevExpress) 10 years ago

        I am not sure how you are going to use the Conditional Appearance module in this situation. Do you want to highlight the primary address in the ListView? The Conditional Appearance module can be used for this task. However, it will be necessary to customize the item appearance in code, since it should depend on the current master object, and thus you cannot specify what object should be highlighted in the rule's Criteria. Please refer to the following topics for additional information:

        How to: Customize the Conditional Appearance Module Behavior
        How to: Access the Master Object from a Nested List View

        Alternatively, you can add an unbound column with a check box to the grid control and calculate its value using the CustomUnboundColumnData event.

        In addition, note that you can display a collection of intermediate objects in the UI and implement custom Link and Unlink buttons that will allow selecting associated objects bypassing the intermediate table. You can customize any built-in action displayed for the intermediate objects' nested ListView in this manner - see Customize Controllers and Actions.

        Let me know if you need further assistance.

        JK JK
        Joseph Kellerer 9 years ago

          Anatol, thank you for your code!

          If you use relations this way, you also have to give permissions for the Link objects.

          Other Answers

          created 12 years ago (modified 12 years ago)

          Hello Juan,
          You cannot directly add a new object in the nested ListView for a Many-To-Many collection (refer to the NewObjectViewController.NewObjectAction Property article), but you can do that via an intermediate Link ListVew: execute the Link action and then the New action in the Link ListView. I hope you will find this information helpful.
          Thanks,
          Ilya P

            Comments (2)
            J J
            Juan Carlos Olleros 12 years ago

              Hi Ilya,
              I've looked through the documentation but I don't really understand what I have to do. The thing is that my model has been automatically generated from existing data tables, and the intermediate tables have a compound key. I had to manually create the m-n relationships since the tool doesn't create them. Now I don’t know where I have to add a Link action and a New action, since the Subcontractor objects that I want to add are not directly related to the Reference object.
              I would appreciate you kind help again.
              Regards

              Anatol (DevExpress) 12 years ago

                I apologize for the misunderstanding. I have published a new Solution.

                Disclaimer: The information provided on DevExpress.com and affiliated web properties (including the DevExpress Support Center) is provided "as is" without warranty of any kind. Developer Express Inc disclaims all warranties, either express or implied, including the warranties of merchantability and fitness for a particular purpose. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.

                Confidential Information: Developer Express Inc does not wish to receive, will not act to procure, nor will it solicit, confidential or proprietary materials and information from you through the DevExpress Support Center or its web properties. Any and all materials or information divulged during chats, email communications, online discussions, Support Center tickets, or made available to Developer Express Inc in any manner will be deemed NOT to be confidential by Developer Express Inc. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.