Merge pull request #3460 from FernetMenta/vdpau
[vuplus_xbmc] / xbmc / addons / GUIDialogAddonInfo.cpp
1 /*
2  *      Copyright (C) 2005-2013 Team XBMC
3  *      http://xbmc.org
4  *
5  *  This Program is free software; you can redistribute it and/or modify
6  *  it under the terms of the GNU General Public License as published by
7  *  the Free Software Foundation; either version 2, or (at your option)
8  *  any later version.
9  *
10  *  This Program is distributed in the hope that it will be useful,
11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13  *  GNU General Public License for more details.
14  *
15  *  You should have received a copy of the GNU General Public License
16  *  along with XBMC; see the file COPYING.  If not, see
17  *  <http://www.gnu.org/licenses/>.
18  *
19  */
20
21 #include "GUIDialogAddonInfo.h"
22 #include "dialogs/GUIDialogYesNo.h"
23 #include "dialogs/GUIDialogOK.h"
24 #include "addons/AddonManager.h"
25 #include "AddonDatabase.h"
26 #include "FileItem.h"
27 #include "filesystem/Directory.h"
28 #include "filesystem/SpecialProtocol.h"
29 #include "GUIDialogAddonSettings.h"
30 #include "dialogs/GUIDialogContextMenu.h"
31 #include "dialogs/GUIDialogTextViewer.h"
32 #include "GUIUserMessages.h"
33 #include "guilib/GUIWindowManager.h"
34 #include "guilib/Key.h"
35 #include "utils/JobManager.h"
36 #include "utils/FileOperationJob.h"
37 #include "utils/StringUtils.h"
38 #include "utils/URIUtils.h"
39 #include "addons/AddonInstaller.h"
40 #include "pvr/PVRManager.h"
41
42 #define CONTROL_BTN_INSTALL          6
43 #define CONTROL_BTN_ENABLE           7
44 #define CONTROL_BTN_UPDATE           8
45 #define CONTROL_BTN_SETTINGS         9
46 #define CONTROL_BTN_CHANGELOG       10
47 #define CONTROL_BTN_ROLLBACK        11
48
49 using namespace std;
50 using namespace ADDON;
51 using namespace XFILE;
52
53 CGUIDialogAddonInfo::CGUIDialogAddonInfo(void)
54     : CGUIDialog(WINDOW_DIALOG_ADDON_INFO, "DialogAddonInfo.xml")
55 {
56   m_item = CFileItemPtr(new CFileItem);
57   m_loadType = KEEP_IN_MEMORY;
58 }
59
60 CGUIDialogAddonInfo::~CGUIDialogAddonInfo(void)
61 {
62 }
63
64 bool CGUIDialogAddonInfo::OnMessage(CGUIMessage& message)
65 {
66   switch ( message.GetMessage() )
67   {
68   case GUI_MSG_WINDOW_DEINIT:
69     {
70       if (m_jobid)
71         CJobManager::GetInstance().CancelJob(m_jobid);
72     }
73     break;
74
75   case GUI_MSG_CLICKED:
76     {
77       int iControl = message.GetSenderId();
78       if (iControl == CONTROL_BTN_UPDATE)
79       {
80         OnUpdate();
81         return true;
82       }
83       if (iControl == CONTROL_BTN_INSTALL)
84       {
85         if (!m_localAddon)
86         {
87           OnInstall();
88           return true;
89         }
90         else if (CGUIDialogYesNo::ShowAndGetInput(24037, 750, 0, 0))
91         {
92           OnUninstall();
93           return true;
94         }
95       }
96       else if (iControl == CONTROL_BTN_ENABLE)
97       {
98         OnEnable(!m_item->GetProperty("Addon.Enabled").asBoolean());
99         return true;
100       }
101       else if (iControl == CONTROL_BTN_SETTINGS)
102       {
103         OnSettings();
104         return true;
105       }
106       else if (iControl == CONTROL_BTN_CHANGELOG)
107       {
108         OnChangeLog();
109         return true;
110       }
111       else if (iControl == CONTROL_BTN_ROLLBACK)
112       {
113         OnRollback();
114         return true;
115       }
116     }
117     break;
118 default:
119     break;
120   }
121
122   return CGUIDialog::OnMessage(message);
123 }
124
125 bool CGUIDialogAddonInfo::OnAction(const CAction &action)
126 {
127   if (action.GetID() == ACTION_SHOW_INFO)
128   {
129     Close();
130     return true;
131   }
132   return CGUIDialog::OnAction(action);
133 }
134
135 void CGUIDialogAddonInfo::OnInitWindow()
136 {
137   UpdateControls();
138   CGUIDialog::OnInitWindow();
139   m_changelog = false;
140 }
141
142 void CGUIDialogAddonInfo::UpdateControls()
143 {
144   CStdString xbmcPath = CSpecialProtocol::TranslatePath("special://xbmc/addons");
145   bool isInstalled = NULL != m_localAddon.get();
146   bool isSystem = isInstalled && m_localAddon->Path().Left(xbmcPath.size()).Equals(xbmcPath);
147   bool isEnabled = isInstalled && m_item->GetProperty("Addon.Enabled").asBoolean();
148   bool isUpdatable = isInstalled && m_item->GetProperty("Addon.UpdateAvail").asBoolean();
149   if (isInstalled)
150     GrabRollbackVersions();
151
152   // TODO: System addons should be able to be disabled
153   bool isPVR = isInstalled && m_localAddon->Type() == ADDON_PVRDLL;
154   bool canDisable = isInstalled && (!isSystem || isPVR) && !m_localAddon->IsInUse();
155   bool canInstall = !isInstalled && m_item->GetProperty("Addon.Broken").empty();
156   bool isRepo = (isInstalled && m_localAddon->Type() == ADDON_REPOSITORY) || (m_addon && m_addon->Type() == ADDON_REPOSITORY);
157
158   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_INSTALL, canDisable || canInstall);
159   SET_CONTROL_LABEL(CONTROL_BTN_INSTALL, isInstalled ? 24037 : 24038);
160
161   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_ENABLE, canDisable);
162   SET_CONTROL_LABEL(CONTROL_BTN_ENABLE, isEnabled ? 24021 : 24022);
163
164   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_UPDATE, isUpdatable);
165   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_SETTINGS, isInstalled && m_localAddon->HasSettings());
166   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_CHANGELOG, !isRepo);
167   CONTROL_ENABLE_ON_CONDITION(CONTROL_BTN_ROLLBACK, m_rollbackVersions.size() > 1);
168 }
169
170 void CGUIDialogAddonInfo::OnUpdate()
171 {
172   CStdString referer;
173   referer.Format("Referer=%s-%s.zip",m_localAddon->ID().c_str(),m_localAddon->Version().c_str());
174   CAddonInstaller::Get().Install(m_addon->ID(), true, referer); // force install
175   Close();
176 }
177
178 void CGUIDialogAddonInfo::OnInstall()
179 {
180   CAddonInstaller::Get().Install(m_addon->ID());
181   Close();
182 }
183
184 void CGUIDialogAddonInfo::OnUninstall()
185 {
186   if (!m_localAddon.get())
187     return;
188
189   // ensure the addon is not a dependency of other installed addons
190   VECADDONS addons;
191   CStdStringArray deps;
192   CAddonMgr::Get().GetAllAddons(addons);
193   for (VECADDONS::iterator it  = addons.begin();
194                            it != addons.end();++it)
195   {
196     if ((*it)->GetDeps().find(m_localAddon->ID()) != (*it)->GetDeps().end())
197       deps.push_back((*it)->Name());
198   }
199
200   if (!CAddonInstaller::Get().CheckDependencies(m_localAddon) && deps.size())
201   {
202     CStdString strLine0, strLine1;
203     StringUtils::JoinString(deps, ", ", strLine1);
204     strLine0.Format(g_localizeStrings.Get(24046), m_localAddon->Name().c_str());
205     CGUIDialogOK::ShowAndGetInput(24037, strLine0, strLine1, 24047);
206     return;
207   }
208
209   // ensure the addon isn't disabled in our database
210   CAddonMgr::Get().DisableAddon(m_localAddon->ID(), false);
211
212   CJobManager::GetInstance().AddJob(new CAddonUnInstallJob(m_localAddon),
213                                     &CAddonInstaller::Get());
214   CAddonMgr::Get().RemoveAddon(m_localAddon->ID());
215   Close();
216 }
217
218 void CGUIDialogAddonInfo::OnEnable(bool enable)
219 {
220   if (!m_localAddon.get())
221     return;
222
223   CAddonMgr::Get().DisableAddon(m_localAddon->ID(), !enable);
224   SetItem(m_item);
225   UpdateControls();
226   g_windowManager.SendMessage(GUI_MSG_NOTIFY_ALL, 0, 0, GUI_MSG_UPDATE);
227 }
228
229 void CGUIDialogAddonInfo::OnSettings()
230 {
231   CGUIDialogAddonSettings::ShowAndGetInput(m_localAddon);
232 }
233
234 void CGUIDialogAddonInfo::OnChangeLog()
235 {
236   CGUIDialogTextViewer* pDlgInfo = (CGUIDialogTextViewer*)g_windowManager.GetWindow(WINDOW_DIALOG_TEXT_VIEWER);
237   CStdString name;
238   if (m_addon)
239     name = m_addon->Name();
240   else if (m_localAddon)
241     name = m_localAddon->Name();
242   pDlgInfo->SetHeading(g_localizeStrings.Get(24054)+" - "+name);
243   if (m_item->GetProperty("Addon.Changelog").empty())
244   {
245     pDlgInfo->SetText(g_localizeStrings.Get(13413));
246     CFileItemList items;
247     if (m_localAddon && 
248         !m_item->GetProperty("Addon.UpdateAvail").asBoolean())
249     {
250       items.Add(CFileItemPtr(new CFileItem(m_localAddon->ChangeLog(),false)));
251     }
252     else
253       items.Add(CFileItemPtr(new CFileItem(m_addon->ChangeLog(),false)));
254     items[0]->Select(true);
255     m_jobid = CJobManager::GetInstance().AddJob(
256       new CFileOperationJob(CFileOperationJob::ActionCopy,items,
257                             "special://temp/"),this);
258   }
259   else
260     pDlgInfo->SetText(m_item->GetProperty("Addon.Changelog").asString());
261
262   m_changelog = true;
263   pDlgInfo->DoModal();
264   m_changelog = false;
265 }
266
267 void CGUIDialogAddonInfo::OnRollback()
268 {
269   CGUIDialogContextMenu* dlg = (CGUIDialogContextMenu*)g_windowManager.GetWindow(WINDOW_DIALOG_CONTEXT_MENU);
270   CAddonDatabase database;
271   database.Open();
272
273   CContextButtons buttons;
274   for (unsigned int i=0;i<m_rollbackVersions.size();++i)
275   {
276     CStdString label(m_rollbackVersions[i]);
277     if (m_rollbackVersions[i].Equals(m_localAddon->Version().c_str()))
278      label += " "+g_localizeStrings.Get(24094);
279    if (database.IsAddonBlacklisted(m_localAddon->ID(),label))
280      label += " "+g_localizeStrings.Get(24095);
281
282     buttons.Add(i,label);
283   }
284   int choice;
285   if ((choice=dlg->ShowAndGetChoice(buttons)) > -1)
286   {
287     // blacklist everything newer
288     for (unsigned int j=choice+1;j<m_rollbackVersions.size();++j)
289       database.BlacklistAddon(m_localAddon->ID(),m_rollbackVersions[j]);
290     CStdString path = "special://home/addons/packages/";
291     path += m_localAddon->ID()+"-"+m_rollbackVersions[choice]+".zip";
292     // needed as cpluff won't downgrade
293     if (!m_localAddon->IsType(ADDON_SERVICE))
294       //we will handle this for service addons in CAddonInstallJob::OnPostInstall
295       CAddonMgr::Get().RemoveAddon(m_localAddon->ID());
296     CAddonInstaller::Get().InstallFromZip(path);
297     database.RemoveAddonFromBlacklist(m_localAddon->ID(),m_rollbackVersions[choice]);
298     Close();
299   }
300 }
301
302 bool CGUIDialogAddonInfo::ShowForItem(const CFileItemPtr& item)
303 {
304   CGUIDialogAddonInfo* dialog = (CGUIDialogAddonInfo*)g_windowManager.GetWindow(WINDOW_DIALOG_ADDON_INFO);
305   if (!dialog)
306     return false;
307   if (!dialog->SetItem(item))
308     return false;
309
310   dialog->DoModal(); 
311   return true;
312 }
313
314 bool CGUIDialogAddonInfo::SetItem(const CFileItemPtr& item)
315 {
316   *m_item = *item;
317   m_rollbackVersions.clear();
318
319   // grab the local addon, if it's available
320   m_localAddon.reset();
321   m_addon.reset();
322   if (CAddonMgr::Get().GetAddon(item->GetProperty("Addon.ID").asString(), m_localAddon)) // sets m_addon if installed regardless of enabled state
323     m_item->SetProperty("Addon.Enabled", "true");
324   else
325     m_item->SetProperty("Addon.Enabled", "false");
326   m_item->SetProperty("Addon.Installed", m_addon ? "true" : "false");
327
328   CAddonDatabase database;
329   database.Open();
330   database.GetAddon(item->GetProperty("Addon.ID").asString(),m_addon);
331
332   if (TranslateType(item->GetProperty("Addon.intType").asString()) == ADDON_REPOSITORY)
333   {
334     CAddonDatabase database;
335     database.Open();
336     VECADDONS addons;
337     if (m_addon)
338       database.GetRepository(m_addon->ID(), addons);
339     else if (m_localAddon) // sanity
340       database.GetRepository(m_localAddon->ID(), addons);
341     int tot=0;
342     for (int i = ADDON_UNKNOWN+1;i<ADDON_VIZ_LIBRARY;++i)
343     {
344       int num=0;
345       for (unsigned int j=0;j<addons.size();++j)
346       {
347         if (addons[j]->Type() == (TYPE)i)
348           ++num;
349       }
350       m_item->SetProperty(CStdString("Repo.") + TranslateType((TYPE)i), num);
351       tot += num;
352     }
353     m_item->SetProperty("Repo.Addons", tot);
354   }
355   return true;
356 }
357
358 void CGUIDialogAddonInfo::OnJobComplete(unsigned int jobID, bool success,
359                                         CJob* job)
360 {
361   if (!m_changelog)
362     return;
363
364   CGUIDialogTextViewer* pDlgInfo = (CGUIDialogTextViewer*)g_windowManager.GetWindow(WINDOW_DIALOG_TEXT_VIEWER);
365
366   m_jobid = 0;
367   if (!success)
368   {
369     pDlgInfo->SetText(g_localizeStrings.Get(195));
370   }
371   else
372   {
373     CFile file;
374     if (file.Open("special://temp/"+
375       URIUtils::GetFileName(((CFileOperationJob*)job)->GetItems()[0]->GetPath())))
376     {
377       char* temp = new char[(size_t)file.GetLength()+1];
378       file.Read(temp,file.GetLength());
379       temp[file.GetLength()] = '\0';
380       m_item->SetProperty("Addon.Changelog",temp);
381       pDlgInfo->SetText(temp);
382       delete[] temp;
383     }
384   }
385   CGUIMessage msg(GUI_MSG_NOTIFY_ALL, WINDOW_DIALOG_TEXT_VIEWER, 0, GUI_MSG_UPDATE);
386   g_windowManager.SendThreadMessage(msg);
387 }
388
389 void CGUIDialogAddonInfo::GrabRollbackVersions()
390 {
391   CFileItemList items;
392   XFILE::CDirectory::GetDirectory("special://home/addons/packages/",items,".zip",DIR_FLAG_NO_FILE_DIRS);
393   items.Sort(SortByLabel, SortOrderAscending);
394   for (int i=0;i<items.Size();++i)
395   {
396     if (items[i]->m_bIsFolder)
397       continue;
398     CStdString ID, version;
399     AddonVersion::SplitFileName(ID,version,items[i]->GetLabel());
400     if (ID.Equals(m_localAddon->ID()))
401       m_rollbackVersions.push_back(version);
402   }
403 }