|
using System; |
|
using System.Linq; |
|
using System.Web.UI; |
|
using Telerik.Sitefinity.Data.ContentLinks; |
|
using Telerik.Sitefinity.Data.Linq.Dynamic; |
|
using Telerik.Sitefinity.DynamicModules; |
|
using Telerik.Sitefinity.DynamicModules.Model; |
|
using Telerik.Sitefinity.GenericContent.Model; |
|
using Telerik.Sitefinity.Model; |
|
using Telerik.Sitefinity.Model.ContentLinks; |
|
|
|
namespace SitefinityWebApp.Examples |
|
{ |
|
public partial class DynamicLINQJoins_EdgeCases : Page |
|
{ |
|
private Type hotelType = Telerik.Sitefinity.Utilities.TypeConverters.TypeResolutionService.ResolveType("Telerik.Sitefinity.DynamicTypes.Model.Hotels.Hotel"); |
|
private Type hotelRoomType = Telerik.Sitefinity.Utilities.TypeConverters.TypeResolutionService.ResolveType("Telerik.Sitefinity.DynamicTypes.Model.Hotels.Room"); |
|
private DynamicModuleManager modManager = DynamicModuleManager.GetManager(); |
|
private IQueryable<DynamicContent> hotels; |
|
private IQueryable<DynamicContent> hotelRooms; |
|
private IQueryable<ContentLink> relatedData; |
|
|
|
public DynamicLINQJoins_EdgeCases() |
|
{ |
|
this.hotels = this.modManager.GetDataItems(this.hotelType) |
|
.Where("Status == Master"); |
|
this.hotelRooms = this.modManager.GetDataItems(this.hotelRoomType) |
|
.Where(x => x.Status == ContentLifecycleStatus.Master); |
|
|
|
// Static type - ContentLink |
|
this.relatedData = ContentLinksManager.GetManager().GetContentLinks(); |
|
} |
|
|
|
// CASE 1: When ONLY TOuter is DynamicContent |
|
protected void UseCasesPeculiarities_TOuterIsDynamicContent() |
|
{ |
|
// This approach is backward compatible, but we do not recommend to continue using it |
|
// You cannot return dynamic type results as properties of objects. |
|
// You can return dynamic type results directly (TResult is DynamicContent) and you must use extension methods in |
|
// Telerik.Sitefinity.Data.Linq.Dynamic to query them afterwards. |
|
var dynamicResult = Queryable.Join(this.hotels, this.relatedData, h => h.Id, rd => rd.ParentItemId, |
|
(h, i) => h) |
|
.Where("Title == \"Hotel_2\""); |
|
|
|
// Complex type results are supported in this case so long as dynamic type results are not used. |
|
// You can extract properties of dynamic types and assign them to other properties on a new static type. |
|
var nonDynamicResult = Queryable.Join(this.hotels, this.relatedData, h => h.Id, rd => rd.ParentItemId, |
|
(h, i) => new JoinResult { HotelTitle = h.GetValue<string>("Title"), RelatedDataId = i.Id }) |
|
.Where(x => x.HotelTitle == "Hotel_2"); |
|
|
|
// It is not possible to return anonymous object types - attempting to do so throws an exception. |
|
try |
|
{ |
|
var dynamicOuterStaticInnerReturnsAnonymous = Queryable.Join(this.hotels, this.relatedData, h => h.Id, rd => rd.ParentItemId, |
|
(h, i) => new { Hotel = h }) |
|
.ToList(); |
|
} |
|
catch (Exception) |
|
{ |
|
// Expected, because this case is not supported |
|
} |
|
|
|
// It is not possible to return DynamicContent types on properties and attempting to do so throws an exception on the Where clause |
|
try |
|
{ |
|
var dynamicOuterStaticInnerReturnsAnonymous = Queryable.Join(this.hotels, this.relatedData, h => h.Id, rd => rd.ParentItemId, |
|
(h, i) => new JoinResult { Hotel = h }) |
|
.Where(x => x.Hotel.GetValue<string>("Title") == "Hotel_2") |
|
.ToList(); |
|
} |
|
catch (Exception) |
|
{ |
|
// Expected, because this case is not supported |
|
} |
|
} |
|
|
|
// When only TInner is DynamicContent or |
|
// When both TOuter and TInner are DynamicContent |
|
protected void UseCasesPeculiarities_TInnerIsDynamicContentOrBothAreDynamicContent() |
|
{ |
|
// This always throws an exception when calling the join. |
|
// You can return anonymous types, but you cannot return dynamic types directly. |
|
// If you want to return dynamic objects in the result, you must wrap them as properties of anonymous return type. |
|
// Extensions in Telerik.Sitefinity.Data.Linq.Dynamic will not be working after the join, and |
|
// you must use standard Linq extension methods in subsequent expressions. |
|
// Extensions in Telerik.Sitefinity.Data.Linq.Dynamic are working, if used in expressions that were passed as parameters to the Join method. |
|
this.hotels = this.modManager.GetDataItems(this.hotelType) |
|
.Where("Status == Master"); |
|
// This works, because it is before the Jopin method call. |
|
|
|
var outerStaticInnerDynamicAnonymousReturnType = Queryable.Join(this.relatedData, this.hotels, rd => rd.ParentItemId, h => h.Id, (rd, h) => new { Hotel = h, ImageId = rd.ChildItemId }); |
|
var res5 = outerStaticInnerDynamicAnonymousReturnType |
|
//.Where("Hotel.Title == \"Hotel_1\""); // does not work, |
|
// because the calls to Telerik.Sitefinity.Data.Linq.Dynamic extension methods is after the Join method |
|
.Where(x => x.Hotel.GetValue<Lstring>("Title") == "Hotel_2"); |
|
|
|
var res6 = res5.ToList(); |
|
|
|
// The same rules apply when TOuter and TInner are both dynamic types |
|
var outerDynamicInnerStaticAnonymousReturnType = Queryable.Join(this.hotels, this.hotelRooms, x => x.OriginalContentId, x => x.SystemParentId, (x, y) => new |
|
{ |
|
Hotel = x, |
|
Room = y, |
|
RoomSize = y.GetValue<decimal?>("Size").Value, |
|
HotelStars = x.GetValue<decimal?>("Stars").Value |
|
}); |
|
var res7 = outerDynamicInnerStaticAnonymousReturnType.Where(x => x.RoomSize == 1).OrderBy(x => x.Hotel.GetValue<decimal?>("Stars").Value); |
|
var res8 = res7.ToList(); |
|
|
|
try |
|
{ |
|
// It is not possible to return dynamic type directly. This will throw an exception on the Where clause. |
|
// Instead, you must wrap the dynamic types in anonymous objects or custom result objects. |
|
var dynamicReturnType = Queryable.Join(this.hotels, this.hotelRooms, x => x.OriginalContentId, x => x.SystemParentId, (x, y) => y); |
|
var res9 = dynamicReturnType.Where(x => x.GetValue<decimal?>("Size").Value == 1); |
|
var res10 = res9.ToList(); |
|
} |
|
catch (Exception) |
|
{ } |
|
} |
|
} |
|
} |