LibraryExplorer.xaml.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. using MediaBrowser.Common;
  2. using MediaBrowser.Controller.Entities;
  3. using MediaBrowser.Controller.Entities.Movies;
  4. using MediaBrowser.Controller.Entities.TV;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Localization;
  7. using MediaBrowser.Controller.Persistence;
  8. using MediaBrowser.Model.Entities;
  9. using MediaBrowser.Model.Logging;
  10. using MediaBrowser.Model.Querying;
  11. using MediaBrowser.Model.Serialization;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Globalization;
  15. using System.Linq;
  16. using System.Text;
  17. using System.Threading;
  18. using System.Threading.Tasks;
  19. using System.Windows;
  20. using System.Windows.Controls;
  21. using System.Windows.Data;
  22. using System.Windows.Input;
  23. using System.Windows.Media.Imaging;
  24. namespace MediaBrowser.ServerApplication
  25. {
  26. /// <summary>
  27. /// Interaction logic for LibraryExplorer.xaml
  28. /// </summary>
  29. public partial class LibraryExplorer : Window
  30. {
  31. private readonly ILogger _logger;
  32. private readonly IJsonSerializer _jsonSerializer;
  33. private readonly ILibraryManager _libraryManager;
  34. private readonly IDisplayPreferencesRepository _displayPreferencesManager;
  35. /// <summary>
  36. /// The current user
  37. /// </summary>
  38. private User CurrentUser;
  39. /// <summary>
  40. /// Initializes a new instance of the <see cref="LibraryExplorer" /> class.
  41. /// </summary>
  42. /// <param name="jsonSerializer">The json serializer.</param>
  43. /// <param name="logger">The logger.</param>
  44. /// <param name="appHost">The app host.</param>
  45. /// <param name="userManager">The user manager.</param>
  46. /// <param name="libraryManager">The library manager.</param>
  47. /// <param name="displayPreferencesManager">The display preferences manager.</param>
  48. public LibraryExplorer(IJsonSerializer jsonSerializer, ILogger logger, IApplicationHost appHost, IUserManager userManager, ILibraryManager libraryManager, IDisplayPreferencesRepository displayPreferencesManager)
  49. {
  50. _logger = logger;
  51. _jsonSerializer = jsonSerializer;
  52. _libraryManager = libraryManager;
  53. _displayPreferencesManager = displayPreferencesManager;
  54. InitializeComponent();
  55. lblVersion.Content = "Version: " + appHost.ApplicationVersion;
  56. foreach (var user in userManager.Users)
  57. ddlProfile.Items.Add(user);
  58. ddlProfile.Items.Insert(0, new User { Name = "Physical" });
  59. ddlProfile.SelectedIndex = 0;
  60. ddlIndexBy.Visibility = ddlSortBy.Visibility = lblIndexBy.Visibility = lblSortBy.Visibility = Visibility.Hidden;
  61. }
  62. /// <summary>
  63. /// Handles the Click event of the btnLoad control.
  64. /// </summary>
  65. /// <param name="sender">The source of the event.</param>
  66. /// <param name="e">The <see cref="RoutedEventArgs" /> instance containing the event data.</param>
  67. private void btnLoad_Click(object sender, RoutedEventArgs e)
  68. {
  69. }
  70. /// <summary>
  71. /// Loads the tree.
  72. /// </summary>
  73. /// <returns>Task.</returns>
  74. private async Task LoadTree()
  75. {
  76. tvwLibrary.Items.Clear();
  77. lblLoading.Visibility = Visibility.Visible;
  78. //grab UI context so we can update within the below task
  79. var ui = TaskScheduler.FromCurrentSynchronizationContext();
  80. //this whole async thing doesn't really work in this instance since all my work pretty much needs to be on the UI thread...
  81. Cursor = Cursors.Wait;
  82. await Task.Run(() =>
  83. {
  84. IEnumerable<BaseItem> children = CurrentUser.Name == "Physical" ? new[] { _libraryManager.RootFolder } : _libraryManager.RootFolder.GetChildren(CurrentUser, true);
  85. children = OrderByName(children, CurrentUser);
  86. foreach (Folder folder in children)
  87. {
  88. var currentFolder = folder;
  89. Task.Factory.StartNew(() =>
  90. {
  91. var prefs = ddlProfile.SelectedItem != null ? _displayPreferencesManager.GetDisplayPreferences(currentFolder.DisplayPreferencesId, (ddlProfile.SelectedItem as User).Id, "LibraryExplorer") ?? new DisplayPreferences { SortBy = ItemSortBy.SortName } : new DisplayPreferences { SortBy = ItemSortBy.SortName };
  92. var node = new TreeViewItem { Tag = currentFolder };
  93. var subChildren = currentFolder.GetChildren(CurrentUser, true);
  94. subChildren = OrderByName(subChildren, CurrentUser);
  95. AddChildren(node, subChildren, CurrentUser);
  96. node.Header = currentFolder.Name + " (" +
  97. node.Items.Count + ")";
  98. tvwLibrary.Items.Add(node);
  99. }, CancellationToken.None, TaskCreationOptions.None, ui);
  100. }
  101. });
  102. lblLoading.Visibility = Visibility.Hidden;
  103. Cursor = Cursors.Arrow;
  104. }
  105. /// <summary>
  106. /// Orders the name of the by.
  107. /// </summary>
  108. /// <param name="items">The items.</param>
  109. /// <param name="user">The user.</param>
  110. /// <returns>IEnumerable{BaseItem}.</returns>
  111. private IEnumerable<BaseItem> OrderByName(IEnumerable<BaseItem> items, User user)
  112. {
  113. return OrderBy(items, user, ItemSortBy.SortName);
  114. }
  115. /// <summary>
  116. /// Orders the name of the by.
  117. /// </summary>
  118. /// <param name="items">The items.</param>
  119. /// <param name="user">The user.</param>
  120. /// <returns>IEnumerable{BaseItem}.</returns>
  121. private IEnumerable<BaseItem> OrderBy(IEnumerable<BaseItem> items, User user, string order)
  122. {
  123. return _libraryManager.Sort(items, user, new[] { order }, SortOrder.Ascending);
  124. }
  125. /// <summary>
  126. /// Adds the children.
  127. /// </summary>
  128. /// <param name="parent">The parent.</param>
  129. /// <param name="children">The children.</param>
  130. /// <param name="user">The user.</param>
  131. private void AddChildren(TreeViewItem parent, IEnumerable<BaseItem> children, User user)
  132. {
  133. foreach (var item in children)
  134. {
  135. var node = new TreeViewItem { Tag = item };
  136. var subFolder = item as Folder;
  137. if (subFolder != null)
  138. {
  139. var prefs = _displayPreferencesManager.GetDisplayPreferences(subFolder.DisplayPreferencesId, user.Id, "LibraryExplorer");
  140. AddChildren(node, OrderBy(subFolder.GetChildren(user, true), user, prefs.SortBy), user);
  141. node.Header = item.Name + " (" + node.Items.Count + ")";
  142. }
  143. else
  144. {
  145. node.Header = item.Name;
  146. }
  147. parent.Items.Add(node);
  148. }
  149. }
  150. /// <summary>
  151. /// TVWs the library_ selected item changed.
  152. /// </summary>
  153. /// <param name="sender">The sender.</param>
  154. /// <param name="e">The e.</param>
  155. private async void tvwLibrary_SelectedItemChanged(object sender, RoutedPropertyChangedEventArgs<object> e)
  156. {
  157. if (tvwLibrary.SelectedItem != null)
  158. {
  159. var item = (BaseItem)(tvwLibrary.SelectedItem as TreeViewItem).Tag;
  160. lblObjType.Content = "Type: " + item.GetType().Name;
  161. var movie = item as Movie;
  162. var folder = item as Folder;
  163. if (folder != null)
  164. {
  165. lblIndexBy.Visibility = ddlIndexBy.Visibility = ddlSortBy.Visibility = lblSortBy.Visibility = Visibility.Visible;
  166. ddlIndexBy.ItemsSource = folder.IndexByOptionStrings;
  167. ddlSortBy.ItemsSource = new[]
  168. {
  169. ItemSortBy.SortName,
  170. ItemSortBy.Album,
  171. ItemSortBy.AlbumArtist,
  172. ItemSortBy.Artist,
  173. ItemSortBy.CommunityRating,
  174. ItemSortBy.DateCreated,
  175. ItemSortBy.DatePlayed,
  176. ItemSortBy.PremiereDate,
  177. ItemSortBy.ProductionYear,
  178. ItemSortBy.Random,
  179. ItemSortBy.Runtime
  180. };
  181. var prefs = _displayPreferencesManager.GetDisplayPreferences(folder.DisplayPreferencesId, (ddlProfile.SelectedItem as User).Id, "LibraryExplorer");
  182. ddlIndexBy.SelectedItem = prefs != null
  183. ? prefs.IndexBy ?? LocalizedStrings.Instance.GetString("NoneDispPref")
  184. : LocalizedStrings.Instance.GetString("NoneDispPref");
  185. ddlSortBy.SelectedItem = prefs != null
  186. ? prefs.SortBy ?? ItemSortBy.SortName
  187. : ItemSortBy.SortName;
  188. }
  189. else
  190. {
  191. lblIndexBy.Visibility = ddlIndexBy.Visibility = ddlSortBy.Visibility = lblSortBy.Visibility = Visibility.Hidden;
  192. }
  193. txtData.Text = FormatJson(_jsonSerializer.SerializeToString(item));
  194. var previews = new List<PreviewItem>();
  195. await Task.Run(() =>
  196. {
  197. if (!string.IsNullOrEmpty(item.PrimaryImagePath))
  198. {
  199. previews.Add(new PreviewItem(item.PrimaryImagePath, "Primary"));
  200. }
  201. if (item.HasImage(ImageType.Banner))
  202. {
  203. previews.Add(new PreviewItem(item.GetImage(ImageType.Banner), "Banner"));
  204. }
  205. if (item.HasImage(ImageType.Logo))
  206. {
  207. previews.Add(new PreviewItem(item.GetImage(ImageType.Logo), "Logo"));
  208. }
  209. if (item.HasImage(ImageType.Art))
  210. {
  211. previews.Add(new PreviewItem(item.GetImage(ImageType.Art), "Art"));
  212. }
  213. if (item.HasImage(ImageType.Thumb))
  214. {
  215. previews.Add(new PreviewItem(item.GetImage(ImageType.Thumb), "Thumb"));
  216. }
  217. previews.AddRange(
  218. item.BackdropImagePaths.Select(
  219. image => new PreviewItem(image, "Backdrop")));
  220. });
  221. lstPreviews.ItemsSource = previews;
  222. lstPreviews.Items.Refresh();
  223. }
  224. }
  225. /// <summary>
  226. /// The INDEN t_ STRING
  227. /// </summary>
  228. private const string INDENT_STRING = " ";
  229. /// <summary>
  230. /// Formats the json.
  231. /// </summary>
  232. /// <param name="str">The STR.</param>
  233. /// <returns>System.String.</returns>
  234. private static string FormatJson(string str)
  235. {
  236. var indent = 0;
  237. var quoted = false;
  238. var sb = new StringBuilder();
  239. for (var i = 0; i < str.Length; i++)
  240. {
  241. var ch = str[i];
  242. switch (ch)
  243. {
  244. case '{':
  245. case '[':
  246. sb.Append(ch);
  247. if (!quoted)
  248. {
  249. sb.AppendLine();
  250. Enumerable.Range(0, ++indent).ForEach(item => sb.Append(INDENT_STRING));
  251. }
  252. break;
  253. case '}':
  254. case ']':
  255. if (!quoted)
  256. {
  257. sb.AppendLine();
  258. Enumerable.Range(0, --indent).ForEach(item => sb.Append(INDENT_STRING));
  259. }
  260. sb.Append(ch);
  261. break;
  262. case '"':
  263. sb.Append(ch);
  264. bool escaped = false;
  265. var index = i;
  266. while (index > 0 && str[--index] == '\\')
  267. escaped = !escaped;
  268. if (!escaped)
  269. quoted = !quoted;
  270. break;
  271. case ',':
  272. sb.Append(ch);
  273. if (!quoted)
  274. {
  275. sb.AppendLine();
  276. Enumerable.Range(0, indent).ForEach(item => sb.Append(INDENT_STRING));
  277. }
  278. break;
  279. case ':':
  280. sb.Append(ch);
  281. if (!quoted)
  282. sb.Append(" ");
  283. break;
  284. default:
  285. sb.Append(ch);
  286. break;
  287. }
  288. }
  289. return sb.ToString();
  290. }
  291. /// <summary>
  292. /// Handles the SelectionChanged event of the ddlProfile control.
  293. /// </summary>
  294. /// <param name="sender">The source of the event.</param>
  295. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  296. private void ddlProfile_SelectionChanged(object sender, SelectionChangedEventArgs e)
  297. {
  298. CurrentUser = ddlProfile.SelectedItem as User;
  299. if (CurrentUser != null)
  300. LoadTree().ConfigureAwait(false);
  301. }
  302. /// <summary>
  303. /// Handles the Click event of the btnRefresh control.
  304. /// </summary>
  305. /// <param name="sender">The source of the event.</param>
  306. /// <param name="e">The <see cref="RoutedEventArgs" /> instance containing the event data.</param>
  307. private void btnRefresh_Click(object sender, RoutedEventArgs e)
  308. {
  309. if (tvwLibrary.SelectedItem != null)
  310. {
  311. var item = ((TreeViewItem)tvwLibrary.SelectedItem).Tag as BaseItem;
  312. if (item != null)
  313. {
  314. item.RefreshMetadata(CancellationToken.None, forceRefresh: cbxForce.IsChecked.Value);
  315. tvwLibrary_SelectedItemChanged(this, null);
  316. }
  317. }
  318. }
  319. /// <summary>
  320. /// Handles the SelectionChanged event of the ddlIndexBy control.
  321. /// </summary>
  322. /// <param name="sender">The source of the event.</param>
  323. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  324. private async void ddlIndexBy_SelectionChanged(object sender, SelectionChangedEventArgs e)
  325. {
  326. if (ddlIndexBy.SelectedItem != null)
  327. {
  328. var treeItem = tvwLibrary.SelectedItem as TreeViewItem;
  329. var folder = treeItem != null
  330. ? treeItem.Tag as Folder
  331. : null;
  332. var prefs = folder != null ? _displayPreferencesManager.GetDisplayPreferences(folder.DisplayPreferencesId, CurrentUser.Id, "LibraryExplorer") : new DisplayPreferences { SortBy = ItemSortBy.SortName };
  333. if (folder != null && prefs.IndexBy != ddlIndexBy.SelectedItem as string)
  334. {
  335. //grab UI context so we can update within the below task
  336. var ui = TaskScheduler.FromCurrentSynchronizationContext();
  337. Cursor = Cursors.Wait;
  338. await Task.Factory.StartNew(() =>
  339. {
  340. using (
  341. new Profiler("Explorer full index expansion for " +
  342. folder.Name, _logger))
  343. {
  344. //re-build the current item's children as an index
  345. prefs.IndexBy = ddlIndexBy.SelectedItem as string;
  346. treeItem.Items.Clear();
  347. AddChildren(treeItem, OrderBy(folder.GetChildren(CurrentUser, true), CurrentUser, prefs.SortBy), CurrentUser);
  348. treeItem.Header = folder.Name + "(" +
  349. treeItem.Items.Count + ")";
  350. Cursor = Cursors.Arrow;
  351. }
  352. }, CancellationToken.None, TaskCreationOptions.None,
  353. ui);
  354. }
  355. }
  356. }
  357. /// <summary>
  358. /// Handles the SelectionChanged event of the ddlSortBy control.
  359. /// </summary>
  360. /// <param name="sender">The source of the event.</param>
  361. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  362. private async void ddlSortBy_SelectionChanged(object sender, SelectionChangedEventArgs e)
  363. {
  364. if (ddlSortBy.SelectedItem != null)
  365. {
  366. var treeItem = tvwLibrary.SelectedItem as TreeViewItem;
  367. var folder = treeItem != null
  368. ? treeItem.Tag as Folder
  369. : null;
  370. var prefs = folder != null ? _displayPreferencesManager.GetDisplayPreferences(folder.DisplayPreferencesId, CurrentUser.Id, "LibraryExplorer") : new DisplayPreferences();
  371. if (folder != null && prefs.SortBy != ddlSortBy.SelectedItem as string)
  372. {
  373. //grab UI context so we can update within the below task
  374. var ui = TaskScheduler.FromCurrentSynchronizationContext();
  375. Cursor = Cursors.Wait;
  376. await Task.Factory.StartNew(() =>
  377. {
  378. using (
  379. new Profiler("Explorer sorting by " + ddlSortBy.SelectedItem + " for " +
  380. folder.Name, _logger))
  381. {
  382. //re-sort
  383. prefs.SortBy = ddlSortBy.SelectedItem as string;
  384. treeItem.Items.Clear();
  385. AddChildren(treeItem, OrderBy(folder.GetChildren(CurrentUser, true), CurrentUser, prefs.SortBy ?? ItemSortBy.SortName), CurrentUser);
  386. treeItem.Header = folder.Name + "(" +
  387. treeItem.Items.Count + ")";
  388. Cursor = Cursors.Arrow;
  389. }
  390. }, CancellationToken.None, TaskCreationOptions.None,
  391. ui);
  392. }
  393. }
  394. }
  395. }
  396. /// <summary>
  397. /// Class PreviewItem
  398. /// </summary>
  399. public class PreviewItem
  400. {
  401. /// <summary>
  402. /// The preview
  403. /// </summary>
  404. private readonly string preview;
  405. /// <summary>
  406. /// The name
  407. /// </summary>
  408. private readonly string name;
  409. /// <summary>
  410. /// Gets the preview.
  411. /// </summary>
  412. /// <value>The preview.</value>
  413. public string Preview
  414. {
  415. get { return preview; }
  416. }
  417. /// <summary>
  418. /// Gets the name.
  419. /// </summary>
  420. /// <value>The name.</value>
  421. public string Name
  422. {
  423. get { return name; }
  424. }
  425. /// <summary>
  426. /// Initializes a new instance of the <see cref="PreviewItem" /> class.
  427. /// </summary>
  428. /// <param name="p">The p.</param>
  429. /// <param name="n">The n.</param>
  430. public PreviewItem(string p, string n)
  431. {
  432. preview = p;
  433. name = n;
  434. }
  435. }
  436. /// <summary>
  437. /// Class Extensions
  438. /// </summary>
  439. static class Extensions
  440. {
  441. /// <summary>
  442. /// Fors the each.
  443. /// </summary>
  444. /// <typeparam name="T"></typeparam>
  445. /// <param name="ie">The ie.</param>
  446. /// <param name="action">The action.</param>
  447. public static void ForEach<T>(this IEnumerable<T> ie, Action<T> action)
  448. {
  449. foreach (var i in ie)
  450. {
  451. action(i);
  452. }
  453. }
  454. }
  455. #region ItemToImageConverter
  456. /// <summary>
  457. /// Class ItemToImageConverter
  458. /// </summary>
  459. [ValueConversion(typeof(string), typeof(bool))]
  460. public class ItemToImageConverter : IValueConverter
  461. {
  462. /// <summary>
  463. /// The instance
  464. /// </summary>
  465. public static ItemToImageConverter Instance =
  466. new ItemToImageConverter();
  467. /// <summary>
  468. /// Converts a value.
  469. /// </summary>
  470. /// <param name="value">The value produced by the binding source.</param>
  471. /// <param name="targetType">The type of the binding target property.</param>
  472. /// <param name="parameter">The converter parameter to use.</param>
  473. /// <param name="culture">The culture to use in the converter.</param>
  474. /// <returns>A converted value. If the method returns null, the valid null value is used.</returns>
  475. public object Convert(object value, Type targetType,
  476. object parameter, CultureInfo culture)
  477. {
  478. var item = value as BaseItem ?? new Folder();
  479. switch (item.DisplayMediaType)
  480. {
  481. case "DVD":
  482. case "HD DVD":
  483. case "Blu-ray":
  484. case "Blu-Ray":
  485. case "Movie":
  486. {
  487. var uri = new Uri
  488. ("pack://application:,,,/Resources/Images/movie.png");
  489. var source = new BitmapImage(uri);
  490. return source;
  491. }
  492. case "Series":
  493. {
  494. var uri = new Uri
  495. ("pack://application:,,,/Resources/Images/series.png");
  496. var source = new BitmapImage(uri);
  497. return source;
  498. }
  499. case "Season":
  500. {
  501. var uri = new Uri
  502. ("pack://application:,,,/Resources/Images/season.png");
  503. var source = new BitmapImage(uri);
  504. return source;
  505. }
  506. case "Episode":
  507. {
  508. var uri = new Uri
  509. ("pack://application:,,,/Resources/Images/episode.png");
  510. var source = new BitmapImage(uri);
  511. return source;
  512. }
  513. case "BoxSet":
  514. {
  515. var uri = new Uri
  516. ("pack://application:,,,/Resources/Images/boxset.png");
  517. var source = new BitmapImage(uri);
  518. return source;
  519. }
  520. case "Audio":
  521. {
  522. var uri = new Uri
  523. ("pack://application:,,,/Resources/Images/audio.png");
  524. var source = new BitmapImage(uri);
  525. return source;
  526. }
  527. case "Person":
  528. {
  529. var uri = new Uri
  530. ("pack://application:,,,/Resources/Images/persons.png");
  531. var source = new BitmapImage(uri);
  532. return source;
  533. }
  534. case "MusicArtist":
  535. {
  536. var uri = new Uri
  537. ("pack://application:,,,/Resources/Images/artist.png");
  538. var source = new BitmapImage(uri);
  539. return source;
  540. }
  541. case "MusicAlbum":
  542. {
  543. var uri = new Uri
  544. ("pack://application:,,,/Resources/Images/album.png");
  545. var source = new BitmapImage(uri);
  546. return source;
  547. }
  548. case "Trailer":
  549. {
  550. var uri = new Uri
  551. ("pack://application:,,,/Resources/Images/trailer.png");
  552. var source = new BitmapImage(uri);
  553. return source;
  554. }
  555. case "None":
  556. {
  557. Uri uri;
  558. if (item is Movie)
  559. uri = new Uri("pack://application:,,,/Resources/Images/movie.png");
  560. else if (item is Series)
  561. uri = new Uri("pack://application:,,,/Resources/Images/series.png");
  562. else if (item is BoxSet)
  563. uri = new Uri("pack://application:,,,/Resources/Images/boxset.png");
  564. else
  565. uri = new Uri("pack://application:,,,/Resources/Images/folder.png");
  566. return new BitmapImage(uri);
  567. }
  568. default:
  569. {
  570. var uri = new Uri("pack://application:,,,/Resources/Images/folder.png");
  571. var source = new BitmapImage(uri);
  572. return source;
  573. }
  574. }
  575. }
  576. /// <summary>
  577. /// Converts a value.
  578. /// </summary>
  579. /// <param name="value">The value that is produced by the binding target.</param>
  580. /// <param name="targetType">The type to convert to.</param>
  581. /// <param name="parameter">The converter parameter to use.</param>
  582. /// <param name="culture">The culture to use in the converter.</param>
  583. /// <returns>A converted value. If the method returns null, the valid null value is used.</returns>
  584. /// <exception cref="System.NotSupportedException">Cannot convert back</exception>
  585. public object ConvertBack(object value, Type targetType,
  586. object parameter, CultureInfo culture)
  587. {
  588. throw new NotSupportedException("Cannot convert back");
  589. }
  590. }
  591. #endregion // ItemToImageConverter
  592. }