MainWindow.xaml.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  1. using Compdfkit_Tools.PDFControl;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. using System.Windows;
  8. using System.Windows.Controls;
  9. using System.Windows.Data;
  10. using System.Windows.Documents;
  11. using System.Windows.Input;
  12. using System.Windows.Media;
  13. using System.Windows.Media.Imaging;
  14. using System.Windows.Navigation;
  15. using System.IO;
  16. using System.ComponentModel;
  17. using System.Resources;
  18. using System.Runtime.CompilerServices;
  19. using Dragablz;
  20. using Compdfkit_Tools.Helper;
  21. using ComPDFKitViewer;
  22. using ComPDFKit.PDFDocument;
  23. using Compdfkit_Tools.Data;
  24. using Compdfkit_Tools.Common;
  25. using System.Reflection;
  26. using ComPDFKit.NativeMethod;
  27. using System.Threading;
  28. namespace PDFViewer
  29. {
  30. /// <summary>
  31. /// Interaction logic for MainWindow.xaml
  32. /// </summary>
  33. public partial class MainWindow : Window, INotifyPropertyChanged
  34. {
  35. #region Property
  36. private PDFViewControl passwordViewer;
  37. private string[] oldAndNewFilePath;
  38. public string AppInfo
  39. {
  40. get { return Assembly.GetExecutingAssembly().GetName().Name + " " + string.Join(".", Assembly.GetExecutingAssembly().GetName().Version.ToString().Split('.').Take(3)); }
  41. }
  42. #endregion
  43. public MainWindow()
  44. {
  45. InitializeComponent();
  46. Loaded += MainWindow_Loaded;
  47. DataContext = this;
  48. }
  49. private void MainWindow_Loaded(object sender, RoutedEventArgs e)
  50. {
  51. PasswordUI.Closed -= PasswordUI_Closed;
  52. PasswordUI.Canceled -= PasswordUI_Canceled;
  53. PasswordUI.Confirmed -= PasswordUI_Confirmed;
  54. PasswordUI.Confirmed += PasswordUI_Confirmed;
  55. PasswordUI.Canceled += PasswordUI_Canceled;
  56. PasswordUI.Closed += PasswordUI_Closed;
  57. HomePageControl.OpenFileEvent -= OpenFileEventHandler;
  58. HomePageControl.OpenFileEvent += OpenFileEventHandler;
  59. TabControl.SelectionChanged -= TabControl_SelectionChanged;
  60. TabControl.SelectionChanged += TabControl_SelectionChanged;
  61. FirstLoadFile();
  62. CPDFAnnotationData.Author = Properties.Settings.Default.DocumentAuthor;
  63. Thread thread = new Thread(DoWork);
  64. thread.Start();
  65. }
  66. static void DoWork()
  67. {
  68. CPDFFont.InitFont();
  69. }
  70. private void FirstLoadFile()
  71. {
  72. if (Properties.Settings.Default.IsLoadLastFileNeeded)
  73. {
  74. if (LoadLastOpenedDocuments())
  75. {
  76. TabControl.SelectedIndex = Properties.Settings.Default.LastSelectedFileIndex;
  77. if (TabControl.SelectedIndex == -1)
  78. {
  79. HomePageButton.IsToggled = true;
  80. }
  81. Properties.Settings.Default.IsLoadLastFileNeeded = false;
  82. Properties.Settings.Default.Save();
  83. }
  84. else
  85. {
  86. HomePageButton.IsToggled = true;
  87. }
  88. }
  89. HomePageButton.IsToggled = true;
  90. }
  91. private bool LoadLastOpenedDocuments()
  92. {
  93. if (Properties.Settings.Default.LastOpenedFiles != null && Properties.Settings.Default.LastOpenedFiles.Count > 0)
  94. {
  95. foreach (var filePath in Properties.Settings.Default.LastOpenedFiles)
  96. {
  97. TabControlLoadDocument(filePath);
  98. }
  99. return true;
  100. }
  101. return false;
  102. }
  103. private void TabControl_SelectionChanged(object sender, SelectionChangedEventArgs e)
  104. {
  105. var tabablz = sender as Dragablz.TabablzControl;
  106. if (tabablz.SelectedIndex != -1)
  107. {
  108. HomePageButton.IsToggled = false;
  109. }
  110. }
  111. private void OpenFileEventHandler(object sender, OpenFileEventArgs args)
  112. {
  113. if (args.OperationType == FileOperationType.OpenFileDirectly)
  114. {
  115. TabControlLoadDocument(args.FilePath, args.FeatureName);
  116. }
  117. else if (args.OperationType == FileOperationType.CreateNewFile)
  118. {
  119. bool confirmCreate = false;
  120. BlankPageSetting blankPageSetting = new BlankPageSetting();
  121. CreateBlankPageSettingDialog createBlankPageSettingDialog = new CreateBlankPageSettingDialog()
  122. {
  123. Owner = this
  124. };
  125. createBlankPageSettingDialog.CreateBlankPage += (o, setting) =>
  126. {
  127. blankPageSetting = setting;
  128. confirmCreate = true;
  129. };
  130. createBlankPageSettingDialog.ShowDialog();
  131. if (!confirmCreate)
  132. {
  133. return;
  134. }
  135. TabItemExt tabItem = new TabItemExt();
  136. MainPage viewPage = new MainPage();
  137. CPDFDocument document = CPDFDocument.CreateDocument();
  138. document.SetInfo(new CPDFInfo
  139. {
  140. Author = Properties.Settings.Default.DocumentAuthor,
  141. Creator = Assembly.GetExecutingAssembly().GetName().Name,
  142. CreationDate = DateTime.Now.ToString(),
  143. Subject = "Document",
  144. Producer = Assembly.GetExecutingAssembly().GetName().Name,
  145. Keywords = "Document",
  146. Version = string.Join(".", Assembly.GetExecutingAssembly().GetName().Version.ToString().Split('.').Take(3))
  147. });
  148. document.InsertPage(0, (float)(blankPageSetting.Size.Width * 7.2 / 2.54), (float)(blankPageSetting.Size.Height * 7.2 / 2.54), "");
  149. if (blankPageSetting.Orientation == Orientation.Horizontal)
  150. {
  151. document.RotatePage(0, 1);
  152. }
  153. viewPage.CheckExistBeforeOpenFileEvent -= ViewPage_CheckExistBeforeOpenFileEvent;
  154. viewPage.FileChangeEvent -= ViewPage_FileChangeEvent;
  155. viewPage.AfterSaveAsFileEvent -= ViewPage_AfterSaveAsFileEvent;
  156. viewPage.CheckExistBeforeOpenFileEvent += ViewPage_CheckExistBeforeOpenFileEvent;
  157. viewPage.FileChangeEvent += ViewPage_FileChangeEvent;
  158. viewPage.AfterSaveAsFileEvent += ViewPage_AfterSaveAsFileEvent;
  159. viewPage.InitWithDocument(document);
  160. tabItem.Content = viewPage;
  161. tabItem.IsSelected = true;
  162. tabItem.FileName = "Blank Page.pdf";
  163. tabItem.Tag = "Blank Page.pdf";
  164. TabControl.Items.Add(tabItem);
  165. viewPage.CanSave = true;
  166. }
  167. else
  168. {
  169. string filePath = CommonHelper.GetExistedPathOrEmpty();
  170. if (filePath != string.Empty)
  171. {
  172. TabControlLoadDocument(filePath);
  173. }
  174. }
  175. }
  176. private void ViewPage_AfterSaveAsFileEvent(object sender, string e)
  177. {
  178. if (sender is MainPage mainPage)
  179. {
  180. var tabItem = (from object t in TabControl.Items select t as TabItemExt).FirstOrDefault(item => Equals(item.Content, mainPage));
  181. if (tabItem != null)
  182. {
  183. tabItem.FileName = Path.GetFileName(e);
  184. tabItem.Tag = e;
  185. }
  186. mainPage.GetPDFViewControl().GetCPDFViewer().CloseDocument();
  187. mainPage.GetPDFViewControl().InitDocument(e);
  188. App.OpenedFilePathList.Add(e);
  189. }
  190. }
  191. private void TabControlLoadDocument(string filePath, string featureName = "")
  192. {
  193. if (App.OpenedFilePathList.Contains(filePath))
  194. {
  195. for (int i = 0; i < App.Current.Windows.Count; i++)
  196. {
  197. MainWindow win = App.Current.Windows[i] as MainWindow;
  198. if (win != null)
  199. {
  200. var items = win.TabControl.Items;
  201. foreach (TabItemExt item in items)
  202. {
  203. if (item.Tag.ToString().ToLower() == filePath.ToLower())
  204. {
  205. win.Activate();
  206. item.IsSelected = true;
  207. (item.Content as MainPage).SetFeatureMode(featureName);
  208. return;
  209. }
  210. }
  211. }
  212. }
  213. }
  214. TabItemExt tabItem = new TabItemExt();
  215. MainPage viewPage = new MainPage();
  216. viewPage.CheckExistBeforeOpenFileEvent -= ViewPage_CheckExistBeforeOpenFileEvent;
  217. viewPage.FileChangeEvent -= ViewPage_FileChangeEvent;
  218. viewPage.AfterSaveAsFileEvent -= ViewPage_AfterSaveAsFileEvent;
  219. viewPage.CheckExistBeforeOpenFileEvent += ViewPage_CheckExistBeforeOpenFileEvent;
  220. viewPage.FileChangeEvent += ViewPage_FileChangeEvent;
  221. viewPage.AfterSaveAsFileEvent += ViewPage_AfterSaveAsFileEvent;
  222. passwordViewer = new PDFViewControl();
  223. passwordViewer.InitDocument(filePath);
  224. if (passwordViewer != null && passwordViewer.PDFViewTool != null)
  225. {
  226. CPDFViewer tempViewer = passwordViewer.PDFViewTool.GetCPDFViewer();
  227. CPDFDocument pdfDoc = tempViewer?.GetDocument();
  228. if (pdfDoc == null)
  229. {
  230. MessageBox.Show("Open File Failed");
  231. return;
  232. }
  233. if (pdfDoc.IsLocked)
  234. {
  235. PasswordUI.SetShowText(System.IO.Path.GetFileName(filePath) + " " + LanguageHelper.CommonManager.GetString("Tip_Encrypted"));
  236. PasswordUI.ClearPassword();
  237. PopupBorder.Visibility = Visibility.Visible;
  238. PasswordUI.Visibility = Visibility.Visible;
  239. }
  240. else
  241. {
  242. viewPage.InitWithFilePath(filePath);
  243. tabItem.Content = viewPage;
  244. tabItem.IsSelected = true;
  245. tabItem.FileName = Path.GetFileName(filePath);
  246. tabItem.Tag = filePath;
  247. App.OpenedFilePathList.Add(filePath);
  248. TabControl.Items.Add(tabItem);
  249. viewPage.Loaded += (sender, e) =>
  250. {
  251. viewPage.SetFeatureMode(featureName);
  252. };
  253. }
  254. }
  255. }
  256. private void PasswordUI_Closed(object sender, EventArgs e)
  257. {
  258. PopupBorder.Visibility = Visibility.Collapsed;
  259. PasswordUI.Visibility = Visibility.Collapsed;
  260. }
  261. private void PasswordUI_Canceled(object sender, EventArgs e)
  262. {
  263. PopupBorder.Visibility = Visibility.Collapsed;
  264. PasswordUI.Visibility = Visibility.Collapsed;
  265. }
  266. private void PasswordUI_Confirmed(object sender, string e)
  267. {
  268. if (passwordViewer != null && passwordViewer.PDFViewTool != null)
  269. {
  270. CPDFViewer tempViewer=passwordViewer.PDFViewTool.GetCPDFViewer();
  271. CPDFDocument pdfDoc=tempViewer?.GetDocument();
  272. if(pdfDoc==null)
  273. {
  274. return;
  275. }
  276. pdfDoc.UnlockWithPassword(e);
  277. if (pdfDoc.IsLocked == false)
  278. {
  279. PasswordUI.SetShowError("", Visibility.Collapsed);
  280. PasswordUI.ClearPassword();
  281. PasswordUI.Visibility = Visibility.Collapsed;
  282. PopupBorder.Visibility = Visibility.Collapsed;
  283. string filePath = pdfDoc.FilePath;
  284. TabItemExt tabItem = new TabItemExt();
  285. MainPage viewPage = new MainPage();
  286. tabItem.Content = viewPage;
  287. tabItem.IsSelected = true;
  288. tabItem.FileName = Path.GetFileName(filePath);
  289. tabItem.Tag = filePath;
  290. viewPage.SetPDFViewer(passwordViewer);
  291. App.OpenedFilePathList.Add(filePath);
  292. TabControl.Items.Add(tabItem);
  293. }
  294. else
  295. {
  296. PasswordUI.SetShowError("error", Visibility.Visible);
  297. }
  298. }
  299. }
  300. private void ViewPage_FileChangeEvent(object sender, EventArgs e)
  301. {
  302. for (int i = 0; i < App.Current.Windows.Count; i++)
  303. {
  304. MainWindow win = App.Current.Windows[i] as MainWindow;
  305. if (win != null)
  306. {
  307. var items = win.TabControl.Items;
  308. foreach (TabItemExt item in items)
  309. {
  310. if (item.Tag.ToString().ToLower() == oldAndNewFilePath[1].ToLower())
  311. {
  312. item.IsSelected = true;
  313. item.FileName = Path.GetFileName(oldAndNewFilePath[0]);
  314. item.IsSelected = true;
  315. item.Tag = oldAndNewFilePath[0];
  316. }
  317. }
  318. }
  319. }
  320. for (int i = 0; i < App.OpenedFilePathList.Count; i++)
  321. {
  322. if (oldAndNewFilePath[1].ToLower() == App.OpenedFilePathList[i].ToLower())
  323. {
  324. App.OpenedFilePathList[i] = oldAndNewFilePath[0];
  325. break;
  326. }
  327. }
  328. }
  329. private bool ViewPage_CheckExistBeforeOpenFileEvent(string[] arg)
  330. {
  331. if (App.OpenedFilePathList.Contains(arg[0]))
  332. {
  333. for (int i = 0; i < App.Current.Windows.Count; i++)
  334. {
  335. MainWindow win = App.Current.Windows[i] as MainWindow;
  336. if (win != null)
  337. {
  338. var items = win.TabControl.Items;
  339. foreach (TabItemExt item in items)
  340. {
  341. if (item.Tag.ToString().ToLower() == arg[0].ToLower())
  342. {
  343. win.Activate();
  344. item.IsSelected = true;
  345. return true;
  346. }
  347. }
  348. }
  349. }
  350. }
  351. oldAndNewFilePath = arg;
  352. return false;
  353. }
  354. public class TabItemExt : TabItem, INotifyPropertyChanged
  355. {
  356. public event PropertyChangedEventHandler PropertyChanged;
  357. private string _fileName;
  358. public string FileName
  359. {
  360. get
  361. {
  362. return _fileName;
  363. }
  364. set
  365. {
  366. if (_fileName != value)
  367. {
  368. _fileName = value;
  369. OnPropertyChanged();
  370. }
  371. }
  372. }
  373. public void OnPropertyChanged([CallerMemberName] string name = null)
  374. {
  375. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(name));
  376. }
  377. //protected override void OnSelected(RoutedEventArgs e)
  378. //{
  379. // string filePath = Tag?.ToString();
  380. // if (!string.IsNullOrEmpty(filePath) && File.Exists(filePath))
  381. // {
  382. // this.IsSelected = true;
  383. // }
  384. //}
  385. }
  386. private DragablzItem FindParentDragablzItem(DependencyObject element)
  387. {
  388. while (element != null && !(element is DragablzItem))
  389. {
  390. element = VisualTreeHelper.GetParent(element);
  391. }
  392. return element as DragablzItem;
  393. }
  394. private TabControl FindParentTabControl(DependencyObject element)
  395. {
  396. while (element != null && !(element is TabControl))
  397. {
  398. element = VisualTreeHelper.GetParent(element);
  399. }
  400. return element as TabControl;
  401. }
  402. private void CloseFileButton_Click(object sender, RoutedEventArgs e)
  403. {
  404. var button = sender as Button;
  405. var dragablzItem = FindParentDragablzItem(button);
  406. var tabControl = FindParentTabControl(dragablzItem);
  407. MainPage mainPage = (dragablzItem.Content as TabItemExt).Content as MainPage;
  408. if (mainPage == null)
  409. {
  410. return;
  411. }
  412. if (mainPage.CanSave)
  413. {
  414. string fileName = (dragablzItem.Content as TabItemExt).FileName;
  415. var message = fileName + " " + LanguageHelper.CommonManager.GetString("Warn_NotSave");
  416. var result = MessageBox.Show(message, LanguageHelper.CommonManager.GetString("Caption_Warning"), MessageBoxButton.YesNoCancel, MessageBoxImage.Question);
  417. if (result == MessageBoxResult.Yes)
  418. {
  419. mainPage.SaveFile();
  420. App.OpenedFilePathList.Remove((dragablzItem.Content as TabItemExt).Tag.ToString());
  421. tabControl.Items.Remove(dragablzItem.Content);
  422. }
  423. else if (result == MessageBoxResult.No)
  424. {
  425. App.OpenedFilePathList.Remove((dragablzItem.Content as TabItemExt).Tag.ToString());
  426. tabControl.Items.Remove(dragablzItem.Content);
  427. }
  428. else
  429. {
  430. }
  431. }
  432. else
  433. {
  434. App.OpenedFilePathList.Remove((dragablzItem.Content as TabItemExt).Tag.ToString());
  435. tabControl.Items.Remove(dragablzItem.Content);
  436. }
  437. if (tabControl.Items.Count == 0)
  438. {
  439. HomePageButton.IsToggled = true;
  440. }
  441. }
  442. private void DefaultAddButton_Click(object sender, RoutedEventArgs e)
  443. {
  444. string filePath = CommonHelper.GetExistedPathOrEmpty();
  445. if (filePath != string.Empty)
  446. {
  447. TabControlLoadDocument(filePath);
  448. }
  449. }
  450. private void MinimizeButton_Click(object sender, RoutedEventArgs e)
  451. {
  452. SystemCommands.MinimizeWindow(this);
  453. }
  454. private void MaximizeRestoreButton_Click(object sender, RoutedEventArgs e)
  455. {
  456. if (this.WindowState == WindowState.Maximized)
  457. {
  458. SystemCommands.RestoreWindow(this);
  459. }
  460. else
  461. {
  462. SystemCommands.MaximizeWindow(this);
  463. }
  464. }
  465. private void CloseButton_Click(object sender, RoutedEventArgs e)
  466. {
  467. int count = TabControl.Items.Count;
  468. while (count > 0)
  469. {
  470. TabItemExt item = TabControl.Items[0] as TabItemExt;
  471. MainPage mainPage = item.Content as MainPage;
  472. if (mainPage == null)
  473. {
  474. count--;
  475. continue;
  476. }
  477. if (mainPage.CanSave)
  478. {
  479. string fileName = item.FileName;
  480. var message = fileName + " " + LanguageHelper.CommonManager.GetString("Warn_NotSave");
  481. var result = MessageBox.Show(message, LanguageHelper.CommonManager.GetString("Caption_Warning"), MessageBoxButton.YesNoCancel, MessageBoxImage.Question);
  482. if (result == MessageBoxResult.Yes)
  483. {
  484. mainPage.SaveFile();
  485. App.OpenedFilePathList.Remove(item.Tag.ToString());
  486. TabControl.Items.Remove(item);
  487. count--;
  488. }
  489. else if (result == MessageBoxResult.No)
  490. {
  491. App.OpenedFilePathList.Remove(item.Tag.ToString());
  492. TabControl.Items.Remove(item);
  493. count--;
  494. }
  495. else
  496. {
  497. break;
  498. }
  499. }
  500. else
  501. {
  502. App.OpenedFilePathList.Remove(item.Tag.ToString());
  503. TabControl.Items.Remove(item);
  504. count--;
  505. }
  506. }
  507. if (count == 0)
  508. {
  509. SystemCommands.CloseWindow(this);
  510. }
  511. }
  512. private void SettingsBtn_Click(object sender, RoutedEventArgs e)
  513. {
  514. SettingsDialog settingsDialog = new SettingsDialog();
  515. settingsDialog.LanguageChanged -= SettingsDialog_LanguageChanged;
  516. settingsDialog.LanguageChanged += SettingsDialog_LanguageChanged;
  517. settingsDialog.Owner = this;
  518. settingsDialog.ShowDialog();
  519. }
  520. private void SettingsDialog_LanguageChanged(object sender, string e)
  521. {
  522. foreach (var tab in TabControl.Items)
  523. {
  524. var item = tab as TabItemExt;
  525. if (item?.Content is MainPage mainPage)
  526. {
  527. if (mainPage.CanSave)
  528. {
  529. string fileName = item.FileName;
  530. var message = fileName + " " + LanguageHelper.CommonManager.GetString("Warn_NotSave");
  531. var result = MessageBox.Show(message, LanguageHelper.CommonManager.GetString("Caption_Warning"), MessageBoxButton.YesNoCancel, MessageBoxImage.Question);
  532. if (result == MessageBoxResult.Yes)
  533. {
  534. mainPage.SaveFile();
  535. }
  536. else
  537. {
  538. break;
  539. }
  540. }
  541. mainPage.GetPDFViewControl().GetCPDFViewer().CloseDocument();
  542. }
  543. }
  544. if (Properties.Settings.Default.LastOpenedFiles == null)
  545. {
  546. Properties.Settings.Default.LastOpenedFiles = new System.Collections.Specialized.StringCollection();
  547. }
  548. Properties.Settings.Default.LastOpenedFiles.Clear();
  549. foreach (var filePath in App.OpenedFilePathList)
  550. {
  551. Properties.Settings.Default.LastOpenedFiles.Add(filePath);
  552. }
  553. Properties.Settings.Default.LastSelectedFileIndex = TabControl.SelectedIndex;
  554. Properties.Settings.Default.IsLoadLastFileNeeded = true;
  555. Properties.Settings.Default.Save();
  556. System.Diagnostics.Process.Start(Application.ResourceAssembly.Location);
  557. this.Close();
  558. }
  559. private void HomePageButton_Toggled(object sender, RoutedEventArgs e)
  560. {
  561. if (sender is HomePageButton homePageButton && homePageButton.IsToggled)
  562. {
  563. TabControl.SelectedIndex = -1;
  564. }
  565. else
  566. {
  567. TabControl.SelectedIndex = 0;
  568. }
  569. }
  570. public event PropertyChangedEventHandler PropertyChanged;
  571. protected virtual void OnPropertyChanged(string propertyName = null)
  572. {
  573. PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
  574. }
  575. protected bool UpdateProper<T>(ref T properValue, T newValue, [CallerMemberName] string properName = "")
  576. {
  577. if (object.Equals(properValue, newValue))
  578. return false;
  579. properValue = newValue;
  580. OnPropertyChanged(properName);
  581. return true;
  582. }
  583. private bool HomePageButton_QueryLock(object sender, HomePageButton.QueryLockEventArgs e)
  584. {
  585. return TabControl.Items.Count <= 0;
  586. }
  587. }
  588. }