LibraryExplorer.xaml.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  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" ? _libraryManager.RootFolder.Children : _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, prefs.IndexBy);
  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. if (item.BackdropImagePaths != null)
  218. previews.AddRange(
  219. item.BackdropImagePaths.Select(
  220. image => new PreviewItem(image, "Backdrop")));
  221. });
  222. lstPreviews.ItemsSource = previews;
  223. lstPreviews.Items.Refresh();
  224. }
  225. }
  226. /// <summary>
  227. /// The INDEN t_ STRING
  228. /// </summary>
  229. private const string INDENT_STRING = " ";
  230. /// <summary>
  231. /// Formats the json.
  232. /// </summary>
  233. /// <param name="str">The STR.</param>
  234. /// <returns>System.String.</returns>
  235. private static string FormatJson(string str)
  236. {
  237. var indent = 0;
  238. var quoted = false;
  239. var sb = new StringBuilder();
  240. for (var i = 0; i < str.Length; i++)
  241. {
  242. var ch = str[i];
  243. switch (ch)
  244. {
  245. case '{':
  246. case '[':
  247. sb.Append(ch);
  248. if (!quoted)
  249. {
  250. sb.AppendLine();
  251. Enumerable.Range(0, ++indent).ForEach(item => sb.Append(INDENT_STRING));
  252. }
  253. break;
  254. case '}':
  255. case ']':
  256. if (!quoted)
  257. {
  258. sb.AppendLine();
  259. Enumerable.Range(0, --indent).ForEach(item => sb.Append(INDENT_STRING));
  260. }
  261. sb.Append(ch);
  262. break;
  263. case '"':
  264. sb.Append(ch);
  265. bool escaped = false;
  266. var index = i;
  267. while (index > 0 && str[--index] == '\\')
  268. escaped = !escaped;
  269. if (!escaped)
  270. quoted = !quoted;
  271. break;
  272. case ',':
  273. sb.Append(ch);
  274. if (!quoted)
  275. {
  276. sb.AppendLine();
  277. Enumerable.Range(0, indent).ForEach(item => sb.Append(INDENT_STRING));
  278. }
  279. break;
  280. case ':':
  281. sb.Append(ch);
  282. if (!quoted)
  283. sb.Append(" ");
  284. break;
  285. default:
  286. sb.Append(ch);
  287. break;
  288. }
  289. }
  290. return sb.ToString();
  291. }
  292. /// <summary>
  293. /// Handles the SelectionChanged event of the ddlProfile control.
  294. /// </summary>
  295. /// <param name="sender">The source of the event.</param>
  296. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  297. private void ddlProfile_SelectionChanged(object sender, SelectionChangedEventArgs e)
  298. {
  299. CurrentUser = ddlProfile.SelectedItem as User;
  300. if (CurrentUser != null)
  301. LoadTree().ConfigureAwait(false);
  302. }
  303. /// <summary>
  304. /// Handles the Click event of the btnRefresh control.
  305. /// </summary>
  306. /// <param name="sender">The source of the event.</param>
  307. /// <param name="e">The <see cref="RoutedEventArgs" /> instance containing the event data.</param>
  308. private void btnRefresh_Click(object sender, RoutedEventArgs e)
  309. {
  310. if (tvwLibrary.SelectedItem != null)
  311. {
  312. var item = ((TreeViewItem)tvwLibrary.SelectedItem).Tag as BaseItem;
  313. if (item != null)
  314. {
  315. item.RefreshMetadata(CancellationToken.None, forceRefresh: cbxForce.IsChecked.Value);
  316. tvwLibrary_SelectedItemChanged(this, null);
  317. }
  318. }
  319. }
  320. /// <summary>
  321. /// Handles the SelectionChanged event of the ddlIndexBy control.
  322. /// </summary>
  323. /// <param name="sender">The source of the event.</param>
  324. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  325. private async void ddlIndexBy_SelectionChanged(object sender, SelectionChangedEventArgs e)
  326. {
  327. if (ddlIndexBy.SelectedItem != null)
  328. {
  329. var treeItem = tvwLibrary.SelectedItem as TreeViewItem;
  330. var folder = treeItem != null
  331. ? treeItem.Tag as Folder
  332. : null;
  333. var prefs = folder != null ? _displayPreferencesManager.GetDisplayPreferences(folder.DisplayPreferencesId, CurrentUser.Id, "LibraryExplorer") : new DisplayPreferences { SortBy = ItemSortBy.SortName };
  334. if (folder != null && prefs.IndexBy != ddlIndexBy.SelectedItem as string)
  335. {
  336. //grab UI context so we can update within the below task
  337. var ui = TaskScheduler.FromCurrentSynchronizationContext();
  338. Cursor = Cursors.Wait;
  339. await Task.Factory.StartNew(() =>
  340. {
  341. using (
  342. new Profiler("Explorer full index expansion for " +
  343. folder.Name, _logger))
  344. {
  345. //re-build the current item's children as an index
  346. prefs.IndexBy = ddlIndexBy.SelectedItem as string;
  347. treeItem.Items.Clear();
  348. AddChildren(treeItem, OrderBy(folder.GetChildren(CurrentUser, true, prefs.IndexBy), CurrentUser, prefs.SortBy), CurrentUser);
  349. treeItem.Header = folder.Name + "(" +
  350. treeItem.Items.Count + ")";
  351. Cursor = Cursors.Arrow;
  352. }
  353. }, CancellationToken.None, TaskCreationOptions.None,
  354. ui);
  355. }
  356. }
  357. }
  358. /// <summary>
  359. /// Handles the SelectionChanged event of the ddlSortBy control.
  360. /// </summary>
  361. /// <param name="sender">The source of the event.</param>
  362. /// <param name="e">The <see cref="SelectionChangedEventArgs" /> instance containing the event data.</param>
  363. private async void ddlSortBy_SelectionChanged(object sender, SelectionChangedEventArgs e)
  364. {
  365. if (ddlSortBy.SelectedItem != null)
  366. {
  367. var treeItem = tvwLibrary.SelectedItem as TreeViewItem;
  368. var folder = treeItem != null
  369. ? treeItem.Tag as Folder
  370. : null;
  371. var prefs = folder != null ? _displayPreferencesManager.GetDisplayPreferences(folder.DisplayPreferencesId, CurrentUser.Id, "LibraryExplorer") : new DisplayPreferences();
  372. if (folder != null && prefs.SortBy != ddlSortBy.SelectedItem as string)
  373. {
  374. //grab UI context so we can update within the below task
  375. var ui = TaskScheduler.FromCurrentSynchronizationContext();
  376. Cursor = Cursors.Wait;
  377. await Task.Factory.StartNew(() =>
  378. {
  379. using (
  380. new Profiler("Explorer sorting by " + ddlSortBy.SelectedItem + " for " +
  381. folder.Name, _logger))
  382. {
  383. //re-sort
  384. prefs.SortBy = ddlSortBy.SelectedItem as string;
  385. treeItem.Items.Clear();
  386. AddChildren(treeItem, OrderBy(folder.GetChildren(CurrentUser, true, prefs.IndexBy), CurrentUser, prefs.SortBy ?? ItemSortBy.SortName), CurrentUser);
  387. treeItem.Header = folder.Name + "(" +
  388. treeItem.Items.Count + ")";
  389. Cursor = Cursors.Arrow;
  390. }
  391. }, CancellationToken.None, TaskCreationOptions.None,
  392. ui);
  393. }
  394. }
  395. }
  396. }
  397. /// <summary>
  398. /// Class PreviewItem
  399. /// </summary>
  400. public class PreviewItem
  401. {
  402. /// <summary>
  403. /// The preview
  404. /// </summary>
  405. private readonly string preview;
  406. /// <summary>
  407. /// The name
  408. /// </summary>
  409. private readonly string name;
  410. /// <summary>
  411. /// Gets the preview.
  412. /// </summary>
  413. /// <value>The preview.</value>
  414. public string Preview
  415. {
  416. get { return preview; }
  417. }
  418. /// <summary>
  419. /// Gets the name.
  420. /// </summary>
  421. /// <value>The name.</value>
  422. public string Name
  423. {
  424. get { return name; }
  425. }
  426. /// <summary>
  427. /// Initializes a new instance of the <see cref="PreviewItem" /> class.
  428. /// </summary>
  429. /// <param name="p">The p.</param>
  430. /// <param name="n">The n.</param>
  431. public PreviewItem(string p, string n)
  432. {
  433. preview = p;
  434. name = n;
  435. }
  436. }
  437. /// <summary>
  438. /// Class Extensions
  439. /// </summary>
  440. static class Extensions
  441. {
  442. /// <summary>
  443. /// Fors the each.
  444. /// </summary>
  445. /// <typeparam name="T"></typeparam>
  446. /// <param name="ie">The ie.</param>
  447. /// <param name="action">The action.</param>
  448. public static void ForEach<T>(this IEnumerable<T> ie, Action<T> action)
  449. {
  450. foreach (var i in ie)
  451. {
  452. action(i);
  453. }
  454. }
  455. }
  456. #region ItemToImageConverter
  457. /// <summary>
  458. /// Class ItemToImageConverter
  459. /// </summary>
  460. [ValueConversion(typeof(string), typeof(bool))]
  461. public class ItemToImageConverter : IValueConverter
  462. {
  463. /// <summary>
  464. /// The instance
  465. /// </summary>
  466. public static ItemToImageConverter Instance =
  467. new ItemToImageConverter();
  468. /// <summary>
  469. /// Converts a value.
  470. /// </summary>
  471. /// <param name="value">The value produced by the binding source.</param>
  472. /// <param name="targetType">The type of the binding target property.</param>
  473. /// <param name="parameter">The converter parameter to use.</param>
  474. /// <param name="culture">The culture to use in the converter.</param>
  475. /// <returns>A converted value. If the method returns null, the valid null value is used.</returns>
  476. public object Convert(object value, Type targetType,
  477. object parameter, CultureInfo culture)
  478. {
  479. var item = value as BaseItem ?? new Folder();
  480. switch (item.DisplayMediaType)
  481. {
  482. case "DVD":
  483. case "HD DVD":
  484. case "Blu-ray":
  485. case "Blu-Ray":
  486. case "Movie":
  487. {
  488. var uri = new Uri
  489. ("pack://application:,,,/Resources/Images/movie.png");
  490. var source = new BitmapImage(uri);
  491. return source;
  492. }
  493. case "Series":
  494. {
  495. var uri = new Uri
  496. ("pack://application:,,,/Resources/Images/series.png");
  497. var source = new BitmapImage(uri);
  498. return source;
  499. }
  500. case "Season":
  501. {
  502. var uri = new Uri
  503. ("pack://application:,,,/Resources/Images/season.png");
  504. var source = new BitmapImage(uri);
  505. return source;
  506. }
  507. case "Episode":
  508. {
  509. var uri = new Uri
  510. ("pack://application:,,,/Resources/Images/episode.png");
  511. var source = new BitmapImage(uri);
  512. return source;
  513. }
  514. case "BoxSet":
  515. {
  516. var uri = new Uri
  517. ("pack://application:,,,/Resources/Images/boxset.png");
  518. var source = new BitmapImage(uri);
  519. return source;
  520. }
  521. case "Audio":
  522. {
  523. var uri = new Uri
  524. ("pack://application:,,,/Resources/Images/audio.png");
  525. var source = new BitmapImage(uri);
  526. return source;
  527. }
  528. case "Person":
  529. {
  530. var uri = new Uri
  531. ("pack://application:,,,/Resources/Images/persons.png");
  532. var source = new BitmapImage(uri);
  533. return source;
  534. }
  535. case "MusicArtist":
  536. {
  537. var uri = new Uri
  538. ("pack://application:,,,/Resources/Images/artist.png");
  539. var source = new BitmapImage(uri);
  540. return source;
  541. }
  542. case "MusicAlbum":
  543. {
  544. var uri = new Uri
  545. ("pack://application:,,,/Resources/Images/album.png");
  546. var source = new BitmapImage(uri);
  547. return source;
  548. }
  549. case "Trailer":
  550. {
  551. var uri = new Uri
  552. ("pack://application:,,,/Resources/Images/trailer.png");
  553. var source = new BitmapImage(uri);
  554. return source;
  555. }
  556. case "None":
  557. {
  558. Uri uri;
  559. if (item is Movie)
  560. uri = new Uri("pack://application:,,,/Resources/Images/movie.png");
  561. else if (item is Series)
  562. uri = new Uri("pack://application:,,,/Resources/Images/series.png");
  563. else if (item is BoxSet)
  564. uri = new Uri("pack://application:,,,/Resources/Images/boxset.png");
  565. else
  566. uri = new Uri("pack://application:,,,/Resources/Images/folder.png");
  567. return new BitmapImage(uri);
  568. }
  569. default:
  570. {
  571. var uri = new Uri("pack://application:,,,/Resources/Images/folder.png");
  572. var source = new BitmapImage(uri);
  573. return source;
  574. }
  575. }
  576. }
  577. /// <summary>
  578. /// Converts a value.
  579. /// </summary>
  580. /// <param name="value">The value that is produced by the binding target.</param>
  581. /// <param name="targetType">The type to convert to.</param>
  582. /// <param name="parameter">The converter parameter to use.</param>
  583. /// <param name="culture">The culture to use in the converter.</param>
  584. /// <returns>A converted value. If the method returns null, the valid null value is used.</returns>
  585. /// <exception cref="System.NotSupportedException">Cannot convert back</exception>
  586. public object ConvertBack(object value, Type targetType,
  587. object parameter, CultureInfo culture)
  588. {
  589. throw new NotSupportedException("Cannot convert back");
  590. }
  591. }
  592. #endregion // ItemToImageConverter
  593. }