2 * Copyright (C) 2005-2013 Team XBMC
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)
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.
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/>.
21 #include "VideoInfoTag.h"
22 #include "utils/XMLUtils.h"
23 #include "guilib/LocalizeStrings.h"
24 #include "settings/AdvancedSettings.h"
25 #include "utils/log.h"
26 #include "utils/StringUtils.h"
27 #include "utils/Variant.h"
28 #include "TextureCache.h"
29 #include "filesystem/File.h"
35 void CVideoInfoTag::Reset()
38 m_writingCredits.clear();
42 m_strPlotOutline.clear();
44 m_strPictureURL.Clear();
46 m_strShowTitle.clear();
47 m_strOriginalTitle.clear();
48 m_strSortTitle.clear();
56 m_strIMDBNumber.clear();
57 m_strMPAARating.clear();
58 m_strFileNameAndPath.clear();
61 m_strProductionCode.clear();
71 m_strUniqueId.clear();
72 m_iSpecialSortSeason = -1;
73 m_iSpecialSortEpisode = -1;
79 m_fanart.m_xml.clear();
83 m_streamDetails.Reset();
88 m_resumePoint.Reset();
89 m_resumePoint.type = CBookmark::RESUME;
92 m_strShowPath.clear();
97 bool CVideoInfoTag::Save(TiXmlNode *node, const CStdString &tag, bool savePathInfo, const TiXmlElement *additionalNode)
99 if (!node) return false;
101 // we start with a <tag> tag
102 TiXmlElement movieElement(tag.c_str());
103 TiXmlNode *movie = node->InsertEndChild(movieElement);
105 if (!movie) return false;
107 XMLUtils::SetString(movie, "title", m_strTitle);
108 if (!m_strOriginalTitle.IsEmpty())
109 XMLUtils::SetString(movie, "originaltitle", m_strOriginalTitle);
110 if (!m_strShowTitle.IsEmpty())
111 XMLUtils::SetString(movie, "showtitle", m_strShowTitle);
112 if (!m_strSortTitle.IsEmpty())
113 XMLUtils::SetString(movie, "sorttitle", m_strSortTitle);
114 XMLUtils::SetFloat(movie, "rating", m_fRating);
115 XMLUtils::SetFloat(movie, "epbookmark", m_fEpBookmark);
116 XMLUtils::SetInt(movie, "year", m_iYear);
117 XMLUtils::SetInt(movie, "top250", m_iTop250);
118 if (tag == "episodedetails" || tag == "tvshow")
120 XMLUtils::SetInt(movie, "season", m_iSeason);
121 XMLUtils::SetInt(movie, "episode", m_iEpisode);
122 XMLUtils::SetString(movie, "uniqueid", m_strUniqueId);
123 XMLUtils::SetInt(movie, "displayseason",m_iSpecialSortSeason);
124 XMLUtils::SetInt(movie, "displayepisode",m_iSpecialSortEpisode);
126 if (tag == "musicvideo")
128 XMLUtils::SetInt(movie, "track", m_iTrack);
129 XMLUtils::SetString(movie, "album", m_strAlbum);
131 XMLUtils::SetString(movie, "votes", m_strVotes);
132 XMLUtils::SetString(movie, "outline", m_strPlotOutline);
133 XMLUtils::SetString(movie, "plot", m_strPlot);
134 XMLUtils::SetString(movie, "tagline", m_strTagLine);
135 XMLUtils::SetInt(movie, "runtime", GetDuration() / 60);
136 if (!m_strPictureURL.m_xml.empty())
139 doc.Parse(m_strPictureURL.m_xml);
140 const TiXmlNode* thumb = doc.FirstChild("thumb");
143 movie->InsertEndChild(*thumb);
144 thumb = thumb->NextSibling("thumb");
147 if (m_fanart.m_xml.size())
150 doc.Parse(m_fanart.m_xml);
151 movie->InsertEndChild(*doc.RootElement());
153 XMLUtils::SetString(movie, "mpaa", m_strMPAARating);
154 XMLUtils::SetInt(movie, "playcount", m_playCount);
155 XMLUtils::SetDate(movie, "lastplayed", m_lastPlayed);
158 XMLUtils::SetString(movie, "file", m_strFile);
159 XMLUtils::SetString(movie, "path", m_strPath);
160 XMLUtils::SetString(movie, "filenameandpath", m_strFileNameAndPath);
161 XMLUtils::SetString(movie, "basepath", m_basePath);
163 if (!m_strEpisodeGuide.IsEmpty())
166 doc.Parse(m_strEpisodeGuide);
167 if (doc.RootElement())
168 movie->InsertEndChild(*doc.RootElement());
170 XMLUtils::SetString(movie, "episodeguide", m_strEpisodeGuide);
173 XMLUtils::SetString(movie, "id", m_strIMDBNumber);
174 XMLUtils::SetStringArray(movie, "genre", m_genre);
175 XMLUtils::SetStringArray(movie, "country", m_country);
176 XMLUtils::SetString(movie, "set", m_strSet);
177 XMLUtils::SetStringArray(movie, "tag", m_tags);
178 XMLUtils::SetStringArray(movie, "credits", m_writingCredits);
179 XMLUtils::SetStringArray(movie, "director", m_director);
180 XMLUtils::SetDate(movie, "premiered", m_premiered);
181 XMLUtils::SetString(movie, "status", m_strStatus);
182 XMLUtils::SetString(movie, "code", m_strProductionCode);
183 XMLUtils::SetDate(movie, "aired", m_firstAired);
184 XMLUtils::SetStringArray(movie, "studio", m_studio);
185 XMLUtils::SetString(movie, "trailer", m_strTrailer);
187 if (m_streamDetails.HasItems())
189 // it goes fileinfo/streamdetails/[video|audio|subtitle]
190 TiXmlElement fileinfo("fileinfo");
191 TiXmlElement streamdetails("streamdetails");
192 for (int iStream=1; iStream<=m_streamDetails.GetVideoStreamCount(); iStream++)
194 TiXmlElement stream("video");
195 XMLUtils::SetString(&stream, "codec", m_streamDetails.GetVideoCodec(iStream));
196 XMLUtils::SetFloat(&stream, "aspect", m_streamDetails.GetVideoAspect(iStream));
197 XMLUtils::SetInt(&stream, "width", m_streamDetails.GetVideoWidth(iStream));
198 XMLUtils::SetInt(&stream, "height", m_streamDetails.GetVideoHeight(iStream));
199 XMLUtils::SetInt(&stream, "durationinseconds", m_streamDetails.GetVideoDuration(iStream));
200 streamdetails.InsertEndChild(stream);
202 for (int iStream=1; iStream<=m_streamDetails.GetAudioStreamCount(); iStream++)
204 TiXmlElement stream("audio");
205 XMLUtils::SetString(&stream, "codec", m_streamDetails.GetAudioCodec(iStream));
206 XMLUtils::SetString(&stream, "language", m_streamDetails.GetAudioLanguage(iStream));
207 XMLUtils::SetInt(&stream, "channels", m_streamDetails.GetAudioChannels(iStream));
208 streamdetails.InsertEndChild(stream);
210 for (int iStream=1; iStream<=m_streamDetails.GetSubtitleStreamCount(); iStream++)
212 TiXmlElement stream("subtitle");
213 XMLUtils::SetString(&stream, "language", m_streamDetails.GetSubtitleLanguage(iStream));
214 streamdetails.InsertEndChild(stream);
216 fileinfo.InsertEndChild(streamdetails);
217 movie->InsertEndChild(fileinfo);
218 } /* if has stream details */
221 for (iCast it = m_cast.begin(); it != m_cast.end(); ++it)
224 TiXmlElement cast("actor");
225 TiXmlNode *node = movie->InsertEndChild(cast);
226 XMLUtils::SetString(node, "name", it->strName);
227 XMLUtils::SetString(node, "role", it->strRole);
228 XMLUtils::SetInt(node, "order", it->order);
229 XMLUtils::SetString(node, "thumb", it->thumbUrl.GetFirstThumb().m_url);
231 XMLUtils::SetStringArray(movie, "artist", m_artist);
232 XMLUtils::SetStringArray(movie, "showlink", m_showLink);
234 TiXmlElement resume("resume");
235 XMLUtils::SetFloat(&resume, "position", (float)m_resumePoint.timeInSeconds);
236 XMLUtils::SetFloat(&resume, "total", (float)m_resumePoint.totalTimeInSeconds);
237 movie->InsertEndChild(resume);
239 XMLUtils::SetString(movie, "dateadded", m_dateAdded.GetAsDBDateTime());
242 movie->InsertEndChild(*additionalNode);
247 bool CVideoInfoTag::Load(const TiXmlElement *element, bool append, bool prioritise)
253 ParseNative(element, prioritise);
257 void CVideoInfoTag::Archive(CArchive& ar)
262 ar << m_writingCredits;
266 ar << m_strPlotOutline;
268 ar << m_strPictureURL.m_spoof;
269 ar << m_strPictureURL.m_xml;
270 ar << m_fanart.m_xml;
272 ar << m_strSortTitle;
276 ar << (int)m_cast.size();
277 for (unsigned int i=0;i<m_cast.size();++i)
279 ar << m_cast[i].strName;
280 ar << m_cast[i].strRole;
281 ar << m_cast[i].order;
282 ar << m_cast[i].thumb;
283 ar << m_cast[i].thumbUrl.m_xml;
292 ar << m_strIMDBNumber;
293 ar << m_strMPAARating;
294 ar << m_strFileNameAndPath;
295 ar << m_strOriginalTitle;
296 ar << m_strEpisodeGuide;
299 ar << m_strProductionCode;
301 ar << m_strShowTitle;
314 ar << m_iSpecialSortSeason;
315 ar << m_iSpecialSortEpisode;
318 ar << dynamic_cast<IArchivable&>(m_streamDetails);
322 ar << m_parentPathID;
323 ar << m_resumePoint.timeInSeconds;
324 ar << m_resumePoint.totalTimeInSeconds;
327 ar << m_dateAdded.GetAsDBDateTime();
334 ar >> m_writingCredits;
338 ar >> m_strPlotOutline;
340 ar >> m_strPictureURL.m_spoof;
341 ar >> m_strPictureURL.m_xml;
342 ar >> m_fanart.m_xml;
344 ar >> m_strSortTitle;
350 m_cast.reserve(iCastSize);
351 for (int i=0;i<iCastSize;++i)
360 info.thumbUrl.ParseString(strXml);
361 m_cast.push_back(info);
370 ar >> m_strIMDBNumber;
371 ar >> m_strMPAARating;
372 ar >> m_strFileNameAndPath;
373 ar >> m_strOriginalTitle;
374 ar >> m_strEpisodeGuide;
377 ar >> m_strProductionCode;
379 ar >> m_strShowTitle;
392 ar >> m_iSpecialSortSeason;
393 ar >> m_iSpecialSortEpisode;
396 ar >> dynamic_cast<IArchivable&>(m_streamDetails);
400 ar >> m_parentPathID;
401 ar >> m_resumePoint.timeInSeconds;
402 ar >> m_resumePoint.totalTimeInSeconds;
406 CStdString dateAdded;
408 m_dateAdded.SetFromDBDateTime(dateAdded);
414 void CVideoInfoTag::Serialize(CVariant& value) const
416 value["director"] = m_director;
417 value["writer"] = m_writingCredits;
418 value["genre"] = m_genre;
419 value["country"] = m_country;
420 value["tagline"] = m_strTagLine;
421 value["plotoutline"] = m_strPlotOutline;
422 value["plot"] = m_strPlot;
423 value["title"] = m_strTitle;
424 value["votes"] = m_strVotes;
425 value["studio"] = m_studio;
426 value["trailer"] = m_strTrailer;
427 value["cast"] = CVariant(CVariant::VariantTypeArray);
428 for (unsigned int i = 0; i < m_cast.size(); ++i)
431 actor["name"] = m_cast[i].strName;
432 actor["role"] = m_cast[i].strRole;
433 actor["order"] = m_cast[i].order;
434 if (!m_cast[i].thumb.IsEmpty())
435 actor["thumbnail"] = CTextureCache::GetWrappedImageURL(m_cast[i].thumb);
436 value["cast"].push_back(actor);
438 value["set"] = m_strSet;
439 value["setid"] = m_iSetId;
440 value["tag"] = m_tags;
441 value["runtime"] = GetDuration();
442 value["file"] = m_strFile;
443 value["path"] = m_strPath;
444 value["imdbnumber"] = m_strIMDBNumber;
445 value["mpaa"] = m_strMPAARating;
446 value["filenameandpath"] = m_strFileNameAndPath;
447 value["originaltitle"] = m_strOriginalTitle;
448 value["sorttitle"] = m_strSortTitle;
449 value["episodeguide"] = m_strEpisodeGuide;
450 value["premiered"] = m_premiered.IsValid() ? m_premiered.GetAsDBDate() : StringUtils::EmptyString;
451 value["status"] = m_strStatus;
452 value["productioncode"] = m_strProductionCode;
453 value["firstaired"] = m_firstAired.IsValid() ? m_firstAired.GetAsDBDate() : StringUtils::EmptyString;
454 value["showtitle"] = m_strShowTitle;
455 value["album"] = m_strAlbum;
456 value["artist"] = m_artist;
457 value["playcount"] = m_playCount;
458 value["lastplayed"] = m_lastPlayed.IsValid() ? m_lastPlayed.GetAsDBDateTime() : StringUtils::EmptyString;
459 value["top250"] = m_iTop250;
460 value["year"] = m_iYear;
461 value["season"] = m_iSeason;
462 value["episode"] = m_iEpisode;
463 value["uniqueid"]["unknown"] = m_strUniqueId;
464 value["rating"] = m_fRating;
465 value["dbid"] = m_iDbId;
466 value["fileid"] = m_iFileId;
467 value["track"] = m_iTrack;
468 value["showlink"] = m_showLink;
469 m_streamDetails.Serialize(value["streamdetails"]);
470 CVariant resume = CVariant(CVariant::VariantTypeObject);
471 resume["position"] = (float)m_resumePoint.timeInSeconds;
472 resume["total"] = (float)m_resumePoint.totalTimeInSeconds;
473 value["resume"] = resume;
474 value["tvshowid"] = m_iIdShow;
475 value["tvshowpath"] = m_strShowPath;
476 value["dateadded"] = m_dateAdded.IsValid() ? m_dateAdded.GetAsDBDateTime() : StringUtils::EmptyString;
477 value["type"] = m_type;
478 value["seasonid"] = m_iIdSeason;
481 void CVideoInfoTag::ToSortable(SortItem& sortable, Field field) const
485 case FieldDirector: sortable[FieldDirector] = m_director; break;
486 case FieldWriter: sortable[FieldWriter] = m_writingCredits; break;
487 case FieldGenre: sortable[FieldGenre] = m_genre; break;
488 case FieldCountry: sortable[FieldCountry] = m_country; break;
489 case FieldTagline: sortable[FieldTagline] = m_strTagLine; break;
490 case FieldPlotOutline: sortable[FieldPlotOutline] = m_strPlotOutline; break;
491 case FieldPlot: sortable[FieldPlot] = m_strPlot; break;
492 case FieldTitle: sortable[FieldTitle] = m_strTitle; break;
493 case FieldVotes: sortable[FieldVotes] = m_strVotes; break;
494 case FieldStudio: sortable[FieldStudio] = m_studio; break;
495 case FieldTrailer: sortable[FieldTrailer] = m_strTrailer; break;
496 case FieldSet: sortable[FieldSet] = m_strSet; break;
497 case FieldTime: sortable[FieldTime] = GetDuration(); break;
498 case FieldFilename: sortable[FieldFilename] = m_strFile; break;
499 case FieldMPAA: sortable[FieldMPAA] = m_strMPAARating; break;
500 case FieldPath: sortable[FieldPath] = m_strFileNameAndPath; break;
501 case FieldSortTitle: sortable[FieldSortTitle] = m_strSortTitle; break;
502 case FieldTvShowStatus: sortable[FieldTvShowStatus] = m_strStatus; break;
503 case FieldProductionCode: sortable[FieldProductionCode] = m_strProductionCode; break;
504 case FieldAirDate: sortable[FieldAirDate] = m_firstAired.IsValid() ? m_firstAired.GetAsDBDate() : (m_premiered.IsValid() ? m_premiered.GetAsDBDate() : StringUtils::EmptyString); break;
505 case FieldTvShowTitle: sortable[FieldTvShowTitle] = m_strShowTitle; break;
506 case FieldAlbum: sortable[FieldAlbum] = m_strAlbum; break;
507 case FieldArtist: sortable[FieldArtist] = m_artist; break;
508 case FieldPlaycount: sortable[FieldPlaycount] = m_playCount; break;
509 case FieldLastPlayed: sortable[FieldLastPlayed] = m_lastPlayed.IsValid() ? m_lastPlayed.GetAsDBDateTime() : StringUtils::EmptyString; break;
510 case FieldTop250: sortable[FieldTop250] = m_iTop250; break;
511 case FieldYear: sortable[FieldYear] = m_iYear; break;
512 case FieldSeason: sortable[FieldSeason] = m_iSeason; break;
513 case FieldEpisodeNumber: sortable[FieldEpisodeNumber] = m_iEpisode; break;
514 case FieldEpisodeNumberSpecialSort: sortable[FieldEpisodeNumberSpecialSort] = m_iSpecialSortEpisode; break;
515 case FieldSeasonSpecialSort: sortable[FieldSeasonSpecialSort] = m_iSpecialSortSeason; break;
516 case FieldRating: sortable[FieldRating] = m_fRating; break;
517 case FieldId: sortable[FieldId] = m_iDbId; break;
518 case FieldTrackNumber: sortable[FieldTrackNumber] = m_iTrack; break;
519 case FieldTag: sortable[FieldTag] = m_tags; break;
521 case FieldVideoResolution: sortable[FieldVideoResolution] = m_streamDetails.GetVideoHeight(); break;
522 case FieldVideoAspectRatio: sortable[FieldVideoAspectRatio] = m_streamDetails.GetVideoAspect(); break;
523 case FieldVideoCodec: sortable[FieldVideoCodec] = m_streamDetails.GetVideoCodec(); break;
525 case FieldAudioChannels: sortable[FieldAudioChannels] = m_streamDetails.GetAudioChannels(); break;
526 case FieldAudioCodec: sortable[FieldAudioCodec] = m_streamDetails.GetAudioCodec(); break;
527 case FieldAudioLanguage: sortable[FieldAudioLanguage] = m_streamDetails.GetAudioLanguage(); break;
529 case FieldSubtitleLanguage: sortable[FieldSubtitleLanguage] = m_streamDetails.GetSubtitleLanguage(); break;
531 case FieldInProgress: sortable[FieldInProgress] = m_resumePoint.IsPartWay(); break;
532 case FieldDateAdded: sortable[FieldDateAdded] = m_dateAdded.IsValid() ? m_dateAdded.GetAsDBDateTime() : StringUtils::EmptyString; break;
533 case FieldMediaType: sortable[FieldMediaType] = DatabaseUtils::MediaTypeFromString(m_type); break;
538 const CStdString CVideoInfoTag::GetCast(bool bIncludeRole /*= false*/) const
541 for (iCast it = m_cast.begin(); it != m_cast.end(); ++it)
543 CStdString character;
544 if (it->strRole.IsEmpty() || !bIncludeRole)
545 character.Format("%s\n", it->strName.c_str());
547 character.Format("%s %s %s\n", it->strName.c_str(), g_localizeStrings.Get(20347).c_str(), it->strRole.c_str());
548 strLabel += character;
550 return strLabel.TrimRight("\n");
553 void CVideoInfoTag::ParseNative(const TiXmlElement* movie, bool prioritise)
555 XMLUtils::GetString(movie, "title", m_strTitle);
556 XMLUtils::GetString(movie, "originaltitle", m_strOriginalTitle);
557 XMLUtils::GetString(movie, "showtitle", m_strShowTitle);
558 XMLUtils::GetString(movie, "sorttitle", m_strSortTitle);
559 XMLUtils::GetFloat(movie, "rating", m_fRating);
560 XMLUtils::GetFloat(movie, "epbookmark", m_fEpBookmark);
562 const TiXmlElement* rElement = movie->FirstChildElement("rating");
563 if (rElement && (rElement->QueryIntAttribute("max", &max_value) == TIXML_SUCCESS) && max_value>=1)
565 m_fRating = m_fRating / max_value * 10; // Normalise the Movie Rating to between 1 and 10
567 XMLUtils::GetInt(movie, "year", m_iYear);
568 XMLUtils::GetInt(movie, "top250", m_iTop250);
569 XMLUtils::GetInt(movie, "season", m_iSeason);
570 XMLUtils::GetInt(movie, "episode", m_iEpisode);
571 XMLUtils::GetInt(movie, "track", m_iTrack);
572 XMLUtils::GetString(movie, "uniqueid", m_strUniqueId);
573 XMLUtils::GetInt(movie, "displayseason", m_iSpecialSortSeason);
574 XMLUtils::GetInt(movie, "displayepisode", m_iSpecialSortEpisode);
576 XMLUtils::GetInt(movie, "displayafterseason",after);
579 m_iSpecialSortSeason = after;
580 m_iSpecialSortEpisode = 0x1000; // should be more than any realistic episode number
582 XMLUtils::GetString(movie, "votes", m_strVotes);
583 XMLUtils::GetString(movie, "outline", m_strPlotOutline);
584 XMLUtils::GetString(movie, "plot", m_strPlot);
585 XMLUtils::GetString(movie, "tagline", m_strTagLine);
587 if (XMLUtils::GetString(movie, "runtime", runtime) && !runtime.empty())
588 m_duration = GetDurationFromMinuteString(runtime);
589 XMLUtils::GetString(movie, "mpaa", m_strMPAARating);
590 XMLUtils::GetInt(movie, "playcount", m_playCount);
591 XMLUtils::GetDate(movie, "lastplayed", m_lastPlayed);
592 XMLUtils::GetString(movie, "file", m_strFile);
593 XMLUtils::GetString(movie, "path", m_strPath);
594 XMLUtils::GetString(movie, "id", m_strIMDBNumber);
595 XMLUtils::GetString(movie, "filenameandpath", m_strFileNameAndPath);
596 XMLUtils::GetDate(movie, "premiered", m_premiered);
597 XMLUtils::GetString(movie, "status", m_strStatus);
598 XMLUtils::GetString(movie, "code", m_strProductionCode);
599 XMLUtils::GetDate(movie, "aired", m_firstAired);
600 XMLUtils::GetString(movie, "album", m_strAlbum);
601 XMLUtils::GetString(movie, "trailer", m_strTrailer);
602 XMLUtils::GetString(movie, "basepath", m_basePath);
604 size_t iThumbCount = m_strPictureURL.m_url.size();
605 CStdString xmlAdd = m_strPictureURL.m_xml;
607 const TiXmlElement* thumb = movie->FirstChildElement("thumb");
610 m_strPictureURL.ParseElement(thumb);
615 xmlAdd = temp+xmlAdd;
617 thumb = thumb->NextSiblingElement("thumb");
620 // prioritise thumbs from nfos
621 if (prioritise && iThumbCount && iThumbCount != m_strPictureURL.m_url.size())
623 rotate(m_strPictureURL.m_url.begin(),
624 m_strPictureURL.m_url.begin()+iThumbCount,
625 m_strPictureURL.m_url.end());
626 m_strPictureURL.m_xml = xmlAdd;
629 XMLUtils::GetStringArray(movie, "genre", m_genre, prioritise, g_advancedSettings.m_videoItemSeparator);
630 XMLUtils::GetStringArray(movie, "country", m_country, prioritise, g_advancedSettings.m_videoItemSeparator);
631 XMLUtils::GetStringArray(movie, "credits", m_writingCredits, prioritise, g_advancedSettings.m_videoItemSeparator);
632 XMLUtils::GetStringArray(movie, "director", m_director, prioritise, g_advancedSettings.m_videoItemSeparator);
633 XMLUtils::GetStringArray(movie, "showlink", m_showLink, prioritise, g_advancedSettings.m_videoItemSeparator);
636 const TiXmlElement* node = movie->FirstChildElement("actor");
637 if (node && node->FirstChild() && prioritise)
641 const TiXmlNode *actor = node->FirstChild("name");
642 if (actor && actor->FirstChild())
645 info.strName = actor->FirstChild()->Value();
646 XMLUtils::GetString(node, "role", info.strRole);
647 XMLUtils::GetInt(node, "order", info.order);
648 const TiXmlElement* thumb = node->FirstChildElement("thumb");
651 info.thumbUrl.ParseElement(thumb);
652 thumb = thumb->NextSiblingElement("thumb");
654 const char* clear=node->Attribute("clear");
655 if (clear && stricmp(clear,"true"))
657 m_cast.push_back(info);
659 node = node->NextSiblingElement("actor");
662 XMLUtils::GetString(movie, "set", m_strSet);
663 XMLUtils::GetStringArray(movie, "tag", m_tags, prioritise, g_advancedSettings.m_videoItemSeparator);
664 XMLUtils::GetStringArray(movie, "studio", m_studio, prioritise, g_advancedSettings.m_videoItemSeparator);
666 node = movie->FirstChildElement("artist");
667 if (node && node->FirstChild() && prioritise)
671 const TiXmlNode* pNode = node->FirstChild("name");
672 const char* pValue=NULL;
673 if (pNode && pNode->FirstChild())
674 pValue = pNode->FirstChild()->Value();
675 else if (node->FirstChild())
676 pValue = node->FirstChild()->Value();
679 const char* clear=node->Attribute("clear");
680 if (clear && stricmp(clear,"true")==0)
682 vector<string> artists = StringUtils::Split(pValue, g_advancedSettings.m_videoItemSeparator);
683 m_artist.insert(m_artist.end(), artists.begin(), artists.end());
685 node = node->NextSiblingElement("artist");
688 node = movie->FirstChildElement("fileinfo");
691 // Try to pull from fileinfo/streamdetails/[video|audio|subtitle]
692 const TiXmlNode *nodeStreamDetails = node->FirstChild("streamdetails");
693 if (nodeStreamDetails)
695 const TiXmlNode *nodeDetail = NULL;
696 while ((nodeDetail = nodeStreamDetails->IterateChildren("audio", nodeDetail)))
698 CStreamDetailAudio *p = new CStreamDetailAudio();
699 XMLUtils::GetString(nodeDetail, "codec", p->m_strCodec);
700 XMLUtils::GetString(nodeDetail, "language", p->m_strLanguage);
701 XMLUtils::GetInt(nodeDetail, "channels", p->m_iChannels);
702 p->m_strCodec.MakeLower();
703 p->m_strLanguage.MakeLower();
704 m_streamDetails.AddStream(p);
707 while ((nodeDetail = nodeStreamDetails->IterateChildren("video", nodeDetail)))
709 CStreamDetailVideo *p = new CStreamDetailVideo();
710 XMLUtils::GetString(nodeDetail, "codec", p->m_strCodec);
711 XMLUtils::GetFloat(nodeDetail, "aspect", p->m_fAspect);
712 XMLUtils::GetInt(nodeDetail, "width", p->m_iWidth);
713 XMLUtils::GetInt(nodeDetail, "height", p->m_iHeight);
714 XMLUtils::GetInt(nodeDetail, "durationinseconds", p->m_iDuration);
715 p->m_strCodec.MakeLower();
716 m_streamDetails.AddStream(p);
719 while ((nodeDetail = nodeStreamDetails->IterateChildren("subtitle", nodeDetail)))
721 CStreamDetailSubtitle *p = new CStreamDetailSubtitle();
722 XMLUtils::GetString(nodeDetail, "language", p->m_strLanguage);
723 p->m_strLanguage.MakeLower();
724 m_streamDetails.AddStream(p);
727 m_streamDetails.DetermineBestStreams();
730 const TiXmlElement *epguide = movie->FirstChildElement("episodeguide");
733 // DEPRECIATE ME - support for old XML-encoded <episodeguide> blocks.
734 if (epguide->FirstChild() && strnicmp("<episodeguide", epguide->FirstChild()->Value(), 13) == 0)
735 m_strEpisodeGuide = epguide->FirstChild()->Value();
740 m_strEpisodeGuide = stream.str();
745 const TiXmlElement *fanart = movie->FirstChildElement("fanart");
748 // we prioritise mixed-mode nfo's with fanart set
753 m_fanart.m_xml = temp+m_fanart.m_xml;
756 m_fanart.m_xml << *fanart;
761 const TiXmlNode *resume = movie->FirstChild("resume");
764 XMLUtils::GetDouble(resume, "position", m_resumePoint.timeInSeconds);
765 XMLUtils::GetDouble(resume, "total", m_resumePoint.totalTimeInSeconds);
769 CStdString dateAdded;
770 XMLUtils::GetString(movie, "dateadded", dateAdded);
771 m_dateAdded.SetFromDBDateTime(dateAdded);
774 bool CVideoInfoTag::HasStreamDetails() const
776 return m_streamDetails.HasItems();
779 bool CVideoInfoTag::IsEmpty() const
781 return (m_strTitle.IsEmpty() &&
782 m_strFile.IsEmpty() &&
783 m_strPath.IsEmpty());
786 unsigned int CVideoInfoTag::GetDuration() const
789 Prefer the duration from the stream if it isn't too
790 small (60%) compared to the duration from the tag.
792 unsigned int duration = m_streamDetails.GetVideoDuration();
793 if (duration > m_duration * 0.6)
799 unsigned int CVideoInfoTag::GetDurationFromMinuteString(const std::string &runtime)
801 unsigned int duration = (unsigned int)str2uint64(runtime);
803 { // failed for some reason, or zero
804 duration = strtoul(runtime.c_str(), NULL, 10);
805 CLog::Log(LOGWARNING, "%s <runtime> should be in minutes. Interpreting '%s' as %u minutes", __FUNCTION__, runtime.c_str(), duration);