views:

159

answers:

2

I've been asked to apply conditional sorting to a data set and I'm trying to figure out how to achieve this via LINQ. In this particular domain, purchase orders can be marked as primary or secondary. The exact mechanism used to determine primary/secondary status is rather complex and not germane to the problem at hand.

Consider the data set below.

Purchase Order    Ship Date       Shipping Address    Total
6                  1/16/2006    Tallahassee FL    500.45 
19.1             2/25/2006      Milwaukee WI      255.69 
5.1           4/11/2006      Chicago IL          199.99 
8               5/16/2006    Fresno CA        458.22 
19             7/3/2006       Seattle WA       151.55
5                   5/1/2006        Avery UT            788.52    
5.2                 8/22/2006     Rice Lake MO        655.00

Secondary POs are those with a decimal number and primary PO's are those with an integer number. The requirement I'm dealing with stipulates that when a user chooses to sort on a given column, the sort should only be applied to primary POs. Secondary POs are ignored for the purposes of sorting, but should still be listed below their primary PO in ship date descending order.

For example, let's say a user sorts on Shipping Address ascending. The data would be sorted as follows. Notice that if you ignore the secondary POs, the data is sorted by Address ascending (Avery, Fresno, Seattle, Tallahassee)

Purchase Order      Ship Date       Shipping Address     Total
5                   5/1/2006        Avery UT            788.52  
--5.2               8/22/2006     Rice Lake MO        655.00  
--5.1             4/11/2006      Chicago IL          199.99 
8                   5/16/2006       Fresno CA           458.22 
19                  7/3/2006        Seattle WA          151.55
--19.1               2/25/2006      Milwaukee WI      255.69 
6                   1/16/2006       Tallahassee FL      500.45

Is there a way to achieve the desired effect using the OrderBy extension method? Or am I stuck (better off) applying the sort to the two data sets independently and then merging into a single result set?

  public IList<PurchaseOrder> ApplySort(bool sortAsc)
    {
        var primary = purchaseOrders.Where(po => po.IsPrimary).OrderBy(po => po.ShippingAddress).ToList();

        var secondary = purchaseOrders.Where(po => !po.IsPrimary).OrderByDescending(po => po.ShipDate).ToList();

        //merge 2 lists somehow so that secondary POs are inserted after their primary
    }
+3  A: 

Have you seen ThenBy and ThenByDescending methods?

purchaseOrders.Where(po => po.IsPrimary).OrderBy(po => po.ShippingAddress).ThenByDescending(x=>x.ShipDate).ToList();

I'm not sure if this is going to fit your needs because I don't quiet understand well how final list should look like (po.IsPrimary and !po.IsPrimary is confusing me).

Misha N.
I don't think this is the answer. `ThenBy` will work only if more than one value of the `Order` will be equal.
HuBeZa
I agree, but like I said, does not look clear enough to me how final list should be ordered. So,I wanted to give him an idea about ThenBy method that may help.
Misha N.
+2  A: 

The solution for your problem is GroupBy.

First order your object according to selected column:

var ordered = purchaseOrders.OrderBy(po => po.ShippingAddress);

Than you need to group your orders according to the primary order. I assumed the order is a string, so i created a string IEqualityComparer like so:

class OrderComparer : IEqualityComparer<string>
{
    public bool Equals(string x, string y)
    {
        x = x.Contains('.') ? x.Substring(0, x.IndexOf('.')) : x;
        y = y.Contains('.') ? y.Substring(0, y.IndexOf('.')) : y;

        return x.Equals(y);
    }

    public int GetHashCode(string obj)
    {
        return obj.Contains('.') ? obj.Substring(0, obj.IndexOf('.')).GetHashCode() : obj.GetHashCode();
    }
}

and use it to group the orders:

var grouped = ordered.GroupBy(po => po.Order, new OrderComparer());

The result is a tree like structure ordered by the ShippingAddress column and grouped by the primary order id.

HuBeZa
Superb. This is very close to what I need. I still need to sort the secondary events on ShipDate within each grouping. Any ideas?
Dirk