smlparse.cpp
上传用户:zhongxx05
上传日期:2007-06-06
资源大小:33641k
文件大小:879k
源码类别:

Symbian

开发平台:

C/C++

  1.     {
  2. goto exit;
  3.     }
  4. }
  5. // /"systemAudioDesc", "systemOperatingSystem", and "systemCPU" were
  6. // introduced in SMIL 2.0:
  7. // /XXXEH- TODO: We need to coordinate with TLC folks to make this a
  8. // user-settable option:
  9. rc = pNode->m_pValues->GetPropertyCString("systemAudioDesc", pBuf);
  10. if(HXR_OK == rc)
  11. {
  12.     const char* pActualValue = (const char*)pBuf->GetBuffer();
  13.     if(strcmp(pActualValue, "on") == 0)
  14.     {
  15. if(!m_bSystemAudioDescPreference)
  16. {
  17.     bFailed = TRUE;
  18. }
  19.     }
  20.     else
  21.     {
  22. if(m_bSystemAudioDescPreference)
  23. {
  24.     bFailed = TRUE;
  25. }
  26.     }
  27.     pBuf->Release();
  28.     if(bFailed)
  29.     {
  30. goto exit;
  31.     }
  32. }
  33. rc = pNode->m_pValues->GetPropertyCString("systemOperatingSystem",
  34. pBuf);
  35. if(HXR_OK == rc  &&  HX_PLATFORM_UNKNOWN!=m_versionInfo.dwPlatformId)
  36. {
  37. #if defined(_MACINTOSH) || defined(_MAC_UNIX) // /Sanity check on Mac platform id:
  38.     HX_ASSERT(HX_PLATFORM_MACOT == m_versionInfo.dwPlatformId  ||
  39.     HX_PLATFORM_MACTCP == m_versionInfo.dwPlatformId);
  40. #endif
  41.     const char* pOS = (const char*)pBuf->GetBuffer();
  42.     // /First, strip leading whitespace chars:
  43.     while (*pOS  &&  isspace(*pOS))
  44.     {
  45. *pOS++;
  46.     }
  47.     /*fixes PR 64428 (UseSystemOS version)*/
  48.     if (!m_bUseSystemOS)
  49.     {
  50. // /Fixes PR 81522 (systemOperatingSystem version): if user
  51. // says not to use OS info, treat it as unknown per SMIL2 spec:
  52. bFailed = FALSE; // /Setting to FALSE instead of comparing to
  53.  // "unknown" fixes OS part of PR 64428(revisted)
  54. goto doneSysOS;
  55.     }
  56.     // /From the SMIL 2.0 CR spec:
  57.     // "The following list contains the suggested values for this
  58.     //  test attribute (additional names may be supported by an
  59.     //  implementation):
  60.     //     aix, beos, bsdi, dgux, freebsd, hpux, irix, linux,
  61.     //     macos, ncr, nec, netbsd, nextstep, nto, openbsd, openvms,
  62.     //     os2, osf, palmos, qnx, sinix, rhapsody, sco, solaris,
  63.     //     sonly, sunos, unixware, win16, win32, win9x, winnt,
  64.     //     wince, unknown"
  65.     // /NOTE: current Mozilla list can be found here:
  66.     // http://lxr.mozilla.org/seamonkey/search?string=_PR_SI_SYSNAME
  67.     switch (m_versionInfo.dwPlatformId)
  68.     {
  69. case HX_PLATFORM_IRIX:
  70. {
  71.     if (0 != strcmp("irix", pOS))
  72.     {
  73. bFailed = TRUE;
  74.     }
  75. }
  76. break;
  77. case HX_PLATFORM_LINUX:
  78. {
  79.     if (0 != strcmp("linux", pOS))
  80.     {
  81. bFailed = TRUE;
  82.     }
  83. }
  84. break;
  85. case HX_PLATFORM_MACOT:
  86. case HX_PLATFORM_MACTCP:
  87. {
  88.     if (0 != strcmp("macos", pOS))
  89.     {
  90. bFailed = TRUE;
  91. //  _DARWIN, _CARBON, and/or _MAC_UNIX implies Darwin is present on a Mac:
  92. #if defined(_DARWIN)  ||  defined(_CARBON)  ||  defined(_MAC_UNIX)
  93. // /Fixes PR 95625: Also check for darwin (UNIX on Mac):
  94. bFailed = (0 != strcmp("darwin", pOS));
  95. #endif // / _DARWIN || _CARBON || _MAC_UNIX.
  96.     }
  97. }
  98. break;
  99. case HX_PLATFORM_QNXNTO:
  100. {   // /We should assert!  But, hey, it might come back 8-]
  101.     if (0 != strcmp("qnx", pOS))
  102.     {
  103. bFailed = TRUE;
  104.     }
  105. }
  106. break;
  107. case HX_PLATFORM_SOLARIS:
  108. case HX_PLATFORM_SUNOS:
  109. {
  110.     // /These are the same since RN started porting to them:
  111.     if (0 != strcmp("solaris", pOS)  &&
  112.     0 != strcmp("sunos", pOS))
  113.     {
  114. bFailed = TRUE;
  115.     }
  116. }
  117. break;
  118. case HX_PLATFORM_WIN16:
  119. {   // /We should really assert!  But, it might come back 8-]
  120.     if (0 != strcmp("win16", pOS))
  121.     {
  122. bFailed = TRUE;
  123.     }
  124. }
  125. break;
  126. case HX_PLATFORM_WIN32S:// /XXXEH is Win32s always on Win16??
  127. {
  128.     if (0 != strcmp("win16", pOS))
  129.     {
  130. bFailed = TRUE;
  131.     }
  132. }
  133. break;
  134. case HX_PLATFORM_WIN95:
  135. case HX_PLATFORM_WIN98:
  136. {
  137.     if (0 != strcmp("win9x", pOS)  &&
  138.     0 != strcmp("win32", pOS))
  139.     {
  140. bFailed = TRUE;
  141.     }
  142. }
  143. break;
  144. case HX_PLATFORM_WINNT:
  145. {
  146.     if (0 != strcmp("winnt", pOS)  &&
  147.     0 != strcmp("win32", pOS))
  148.     {
  149. bFailed = TRUE;
  150.     }
  151. }
  152. break;
  153. case HX_PLATFORM_SYMBIAN:
  154. {
  155.     if (0 != strcmp("symbian", pOS))
  156.     {
  157. bFailed = TRUE;
  158.     }
  159. }
  160. break;
  161. case HX_PLATFORM_UNKNOWN:
  162. {
  163.     if (0 != strcmp("unknown", pOS))
  164.     {
  165. bFailed = TRUE;
  166.     }
  167. }
  168. break;
  169. default:
  170. {
  171.     // /The following SMIL 2.0 systemOperatingSystem values
  172.     // are not covered in HXGetWinVer as of 12/2000:
  173.     //     XXXaix, XXXbeos, bsdi, dgux, freebsd, XXXhpux, ncr,
  174.     //     nec, netbsd, nextstep, nto, openbsd, openvms,
  175.     //     os2, XXXosf, palmos, sinix, rhapsody, XXXsco,
  176.     //     sonly, unixware, wince
  177. #if defined(_AIX)
  178.     if (0 != strcmp("aix", pOS))
  179.     {
  180. bFailed = TRUE;
  181.     }
  182. #elif defined(_BEOS)
  183.     if (0 != strcmp("beos", pOS))
  184.     {
  185. bFailed = TRUE;
  186.     }
  187. #elif defined(_BSDI)
  188.     if (0 != strcmp("bsdi", pOS))
  189.     {
  190. bFailed = TRUE;
  191.     }
  192. //  _DARWIN, _CARBON, and/or _MAC_UNIX implies Darwin is present on a Mac:
  193. #elif defined(_DARWIN)  ||  defined(_CARBON)  ||  defined(_MAC_UNIX)
  194.     // /Fixes PR 95625 (in case HX_PLATFORM_MAC... is not
  195.     // handled, above):
  196.     if (0 != strcmp("darwin", pOS))
  197.     {
  198. bFailed = TRUE;
  199.     }
  200. #elif defined(_FREEBSD)  ||  defined(_FREEBSD2)
  201.     if (0 != strcmp("freebsd", pOS))
  202.     {
  203. bFailed = TRUE;
  204.     }
  205. #elif defined(_HPUX)
  206.     if (0 != strcmp("hpux", pOS))
  207.     {
  208. bFailed = TRUE;
  209.     }
  210. #elif defined(_NETBSD)  ||  defined(__NetBSD__)
  211.     if (0 != strcmp("netbsd", pOS))
  212.     {
  213. bFailed = TRUE;
  214.     }
  215. #elif defined(_OSF1)
  216.     if (0 != strcmp("osf", pOS))
  217.     {
  218. bFailed = TRUE;
  219.     }
  220. #elif defined(_SCO_UW)
  221.     if (0 != strcmp("sco", pOS))
  222.     {
  223. bFailed = TRUE;
  224.     }
  225. #elif defined(_WINCE)
  226.     if (0 != strcmp("wince", pOS))
  227.     {
  228. bFailed = TRUE;
  229.     }
  230. // /SGI is not in the SMIL 2.0 spec, but the spec says we can add more than
  231. // the OSes it recommends:
  232. #elif defined(_SGI)
  233.     if (0 != strcmp("sgi", pOS))
  234.     {
  235. bFailed = TRUE;
  236.     }
  237. #else // /Catch-all for any other OS; throw an error at compile time:
  238. /* Please add to the above list with your OS's define and match it to
  239. one of the following (or add a string if it's not in the following list and
  240. be sure to tell RN documentation folks about it):
  241. "dgux", "ncr", "nec", "nextstep", "nto", "openbsd", "openvms", "os2",
  242. "palmos", "sinix", "rhapsody", "sonly", "unixware"
  243. */
  244. #endif
  245. }
  246.     break;
  247.     }
  248.     if (bFailed  &&  0 == strcmp("unknown", pOS))
  249.     {
  250. bFailed = FALSE; // /"unknown should play on any OS"
  251.     }
  252. doneSysOS:
  253.     HX_RELEASE(pBuf);
  254.     if(bFailed)
  255.     {
  256. goto exit;
  257.     }
  258. }
  259. rc = pNode->m_pValues->GetPropertyCString("systemCPU", pBuf);
  260. if(HXR_OK == rc  &&  HX_MACHINE_UNKNOWN!=m_versionInfo.dwMachineType)
  261. {
  262.     const char* pCPU = (const char*)pBuf->GetBuffer();
  263.     // /First, strip leading whitespace chars:
  264.     while (*pCPU  &&  isspace(*pCPU))
  265.     {
  266. *pCPU++;
  267.     }
  268.     /*fixes PR 64428 (UseSystemCPU version)*/
  269.     if (!m_bUseSystemCPU)
  270.     {
  271. // /Fixes PR 81522 (systemCPU version): if user says not to
  272. // use their CPU info, treat it as unknown per SMIL2 spec:
  273. bFailed = FALSE; // /Setting to FALSE instead of comparing to
  274.  // "unknown" fixes CPU part of PR 64428(revisted)
  275. goto doneSysCPU;
  276.     }
  277.     // /The SMIL 2.0 draft of 12/2000 says:
  278.     //  "The following list contains the suggested values for this
  279.     //   test attribute (additional names may be supported by an
  280.     //   implementation): alpha, arm, arm32, hppa1.1, m68k, mips,
  281.     //   ppc, rs6000, vax, x86, unknown."
  282.     switch (m_versionInfo.dwMachineType)
  283.     {
  284. case HX_MACHINE_ALPHA:
  285. {
  286.     if (0 != strcmp("alpha", pCPU))
  287.     {
  288. bFailed = TRUE;
  289.     }
  290. }
  291. break;
  292.                 case HX_MACHINE_SYMEMULATOR:
  293.                 case HX_MACHINE_ARM:
  294.                 {
  295.                     if (0 != strcmp("arm", pCPU))
  296.                     {
  297.                         if (0 != strcmp("arm32", pCPU))
  298.                         {
  299.                             bFailed = TRUE;
  300.                         }
  301.                     }
  302.                 }
  303.                 break;
  304. /* /XXX- these aren't supported in HXGetWinVer():
  305. case :
  306. {
  307.     if (0 != strcmp("hppa1.1", pCPU))
  308.     {
  309. bFailed = TRUE;
  310.     }
  311. }
  312. break;
  313. */
  314. case HX_MACHINE_68K:
  315. {
  316.     if (0 != strcmp("m68k", pCPU))
  317.     {
  318. bFailed = TRUE;
  319.     }
  320. }
  321. break;
  322. case HX_MACHINE_MIPS:
  323. {
  324.     if (0 != strcmp("mips", pCPU))
  325.     {
  326. bFailed = TRUE;
  327.     }
  328. }
  329. break;
  330. case HX_MACHINE_PPC:
  331. {
  332.     if (0 != strcmp("ppc", pCPU))
  333.     {
  334. bFailed = TRUE;
  335.     }
  336. }
  337. break;
  338. /* /XXX- this isn't supported in HXGetWinVer():
  339. case :
  340. {
  341.     if (0 != strcmp("rs6000", pCPU))
  342.     {
  343. bFailed = TRUE;
  344.     }
  345. }
  346. break;
  347. */
  348. // /This one is not in the SMIL 2.0 spec; I added it as
  349. // permitted by the spec:
  350. case HX_MACHINE_SPARC:
  351. {
  352.     if (0 != strcmp("sparc", pCPU))
  353.     {
  354. bFailed = TRUE;
  355.     }
  356. }
  357. break;
  358. /* /XXX- this isn't supported in HXGetWinVer():
  359. case :
  360. {
  361.     if (0 != strcmp("vax", pCPU))
  362.     {
  363. bFailed = TRUE;
  364.     }
  365. }
  366. break;
  367. */
  368. case HX_MACHINE_486:
  369. case HX_MACHINE_586:
  370. case HX_MACHINE_686:
  371. {
  372.     if (0 != strcmp("x86", pCPU))
  373.     {
  374. bFailed = TRUE;
  375.     }
  376. }
  377. break;
  378. default:
  379. {
  380.     // /Need to fill in others here and in HXGetWinVer().
  381. }
  382. break;
  383.     }
  384.     if (bFailed  &&  0 == strcmp("unknown", pCPU))
  385.     {
  386. bFailed = FALSE; // /"unknown should play on any CPU"
  387.     }
  388. doneSysCPU:
  389.     HX_RELEASE(pBuf);
  390.     if(bFailed)
  391.     {
  392. goto exit;
  393.     }
  394. }
  395. if (m_pActiveNamespaceMap)
  396. {
  397.     // look through name spaces for systemComponent name space.  If it's there
  398.     // append the prefix and look for systemComponent test attribute
  399.     CHXMapStringToOb::Iterator ndxBuffer = m_pActiveNamespaceMap->Begin();
  400.     for (; ndxBuffer != m_pActiveNamespaceMap->End(); ++ndxBuffer)
  401.     {
  402. IHXBuffer* pBuffer = (IHXBuffer*)(*ndxBuffer);
  403. if (strcmp((const char*) SYSTEM_COMPONENT_NAMESPACE, (const char*)pBuffer->GetBuffer()) == 0)
  404. {
  405.     const char* pPrefix = (const char*)ndxBuffer.get_key();
  406.     // +1 for the : in the namespace and 1 for the 
  407.     char* pAttrName = new char[strlen(pPrefix) + strlen((const char*) SYSTEM_COMPONENT) + 2];
  408.     pAttrName[0] = '';
  409.     strcat(pAttrName, pPrefix); /* Flawfinder: ignore */
  410.     strcat(pAttrName, ":"); /* Flawfinder: ignore */
  411.     strcat(pAttrName, (const char*) SYSTEM_COMPONENT); /* Flawfinder: ignore */
  412.     rc = pNode->m_pValues->GetPropertyCString(
  413. pAttrName, pBuf);
  414.     HX_VECTOR_DELETE(pAttrName);
  415.     if(HXR_OK == rc)
  416.     {
  417. bFailed = systemComponentFailed(pBuf);
  418. pBuf->Release();
  419. // If I get in here, I found the right prefix
  420. if (bFailed)
  421. {
  422.     goto exit;
  423. }
  424.     }
  425. }
  426.     }
  427. }
  428.     }
  429. exit:
  430.     return bFailed;
  431. }
  432. BOOL
  433. CSmilParser::customTestFailed(SMILNode* pNode)
  434. {
  435.     BOOL bFailed = FALSE; // /If no "customTest" attribute, we don't fail.
  436.     IHXBuffer* pBuf = NULL;
  437.     if(pNode->m_pValues)
  438.     {
  439. HX_RESULT rc = pNode->m_pValues->GetPropertyCString("customTest",
  440.     pBuf);
  441. if(HXR_OK == rc)
  442. {
  443.     const char* pCustTestKey = (const char*)pBuf->GetBuffer();
  444.     CSmilCustomTest* pCustTest = (CSmilCustomTest*)
  445.     ((*m_pCustomTestMap)[pCustTestKey]);
  446.     if (pCustTest)
  447.     {
  448. // /Look for uid in registry; if found and set to 1, then
  449. // we've succeeded, else if found and set to 0, we've failed,
  450. // else if not found and m_defaultState==TRUE, we've succeed,
  451. // else we've failed:
  452. if (pCustTest->m_uid.GetLength())
  453. {
  454.     IHXPreferences* pPrefs = 0;
  455.     IHXRegistry* pRegistry = NULL;
  456.     // /If there is a uid attribute, then we fail unless
  457.     // we can find a match in the registry (that's set to
  458.     // a number that isn't 0 or a string that isn't a
  459.     // case-insensitive version of the "false"):
  460.     bFailed = TRUE;
  461.     if(HXR_OK == m_pContext->QueryInterface(
  462.     IID_IHXPreferences, (void**)&pPrefs))
  463.     {
  464. IHXBuffer* pBuf = 0;
  465. CHXString regKey = "customTests\";
  466. regKey += pCustTest->m_uid;
  467. if (HXR_OK == pPrefs->ReadPref(
  468. (const char*)regKey, pBuf))
  469. {
  470.     const char* pszBuf =
  471.     (const char*)pBuf->GetBuffer();
  472.     HX_ASSERT(pszBuf);
  473.     if (pszBuf)
  474.     {
  475. UINT32 ulPrevNumericalVal =
  476. ((UINT32)atol(pszBuf));
  477. if (ulPrevNumericalVal != 0)
  478. {
  479.     bFailed = FALSE;
  480. }
  481. // /atol returns 0 if it fails, but
  482. // 0 is a valid value:
  483. else if (0==strcmp("0", pszBuf))
  484. {
  485.     bFailed = TRUE;
  486. }
  487. else // /Might be a string like "True":
  488. {
  489.     // /Look for any value that is not a
  490.     // case-insensitive version of "False":
  491.     bFailed = !stricmp("false", pszBuf);
  492. }
  493.     }
  494.     HX_RELEASE(pBuf);
  495. }
  496. else // /not in registry, so use defaultState:
  497. {
  498.     bFailed = !pCustTest->m_bDefaultState;
  499. }
  500. HX_RELEASE(pPrefs);
  501.     }
  502. }
  503. else
  504. {
  505.     bFailed = !pCustTest->m_bDefaultState;
  506. }
  507.     }
  508.     HX_RELEASE(pBuf);
  509. }
  510.     }
  511.     return bFailed;
  512. }
  513. HX_RESULT
  514. CSmilParser::selectSwitchNodes(SMILNode* pSwitchNode)
  515. {
  516.     HX_RESULT rc = HXR_OK;
  517.     SMILNodeList* pNodeList = pSwitchNode->m_pNodeList;
  518.     if(!pNodeList)
  519.     {
  520. return rc;
  521.     }
  522.     SMILNode* pSelectedNode = 0;
  523.     CHXSimpleList* pRejectedNodeList = new CHXSimpleList;
  524.     CHXSimpleList::Iterator i;
  525.     for(i=pNodeList->Begin();i!=pNodeList->End();++i)
  526.     {
  527. SMILNode* pNode = (SMILNode*)(*i);
  528. if(!pNode->m_bDelete)
  529. {
  530.     if (testAttributeFailed(pNode)  ||
  531.     customTestFailed(pNode))
  532.     {
  533. pRejectedNodeList->AddTail(pNode);
  534.     }
  535.     else
  536.     {
  537. pSelectedNode = pNode;
  538. //[SMIL 1.0 Compliance] Fixes PR 26732:
  539. //Now, update selected node's id to that of the switch so
  540. // that anchors pointing to the switch id will reference
  541. // the node of the switch that was selected.  Note that
  542. // we can't fix this the other way around, namely to rename
  543. // all anchors' id strings to that of the selected node
  544. // since not all of them have been parsed/set yet:
  545. HX_ASSERT(pSelectedNode->m_id.GetLength() > 0  &&
  546. pSwitchNode->m_id.GetLength() > 0);
  547. if (pSelectedNode->m_id.GetLength() > 0  &&
  548. pSwitchNode->m_id.GetLength() > 0)
  549. {
  550.     pSelectedNode->m_id = pSwitchNode->m_id;
  551.     (*m_pIDMap)[(const char*)pSelectedNode->m_id] =
  552.     pSelectedNode;
  553.     pSwitchNode->m_id = assignID("switch");
  554.     (*m_pIDMap)[(const char*)pSwitchNode->m_id] =
  555.     pSwitchNode;
  556. }
  557. break;
  558.     }
  559. }
  560.     }
  561.     HX_DELETE(pRejectedNodeList);
  562.     // delete non-selected nodes
  563.     for(i=pNodeList->Begin();i!=pNodeList->End();++i)
  564.     {
  565. SMILNode* pNode = (SMILNode*)(*i);
  566. if(pNode != pSelectedNode)
  567. {
  568.     pNode->m_bDelete = TRUE;
  569. }
  570.     }
  571.     return rc;
  572. }
  573. HX_RESULT
  574. CSmilParser::markTestAttributeNodes(SMILNodeList* pNodeList)
  575. {
  576.     HX_RESULT rc = HXR_OK;
  577.     if(!pNodeList)
  578.     {
  579. return rc;
  580.     }
  581.     CHXSimpleList::Iterator i;
  582.     for(i=pNodeList->Begin();i!=pNodeList->End() && SUCCEEDED(rc);++i)
  583.     {
  584. SMILNode* pNode = (SMILNode*)(*i);
  585. rc = addToNamespaceScope(pNode);
  586. if (FAILED(rc))
  587. {
  588.     return rc;
  589. }
  590. if(pNode->m_tag == SMILSwitch)
  591. {
  592.     selectSwitchNodes(pNode);
  593. }
  594. else
  595. {
  596.     if(testAttributeFailed(pNode)  ||
  597.     customTestFailed(pNode))
  598.     {
  599. pNode->m_bDelete = TRUE;
  600.     }
  601. }
  602. //Fix for PR 21303: we need to keep drilling down
  603. // through the tree even if we've just called
  604. // selectSwitchNodes() to delete all but one
  605. // child of the switch; the descendants of that
  606. // child may have attributes as well that need
  607. // to be checked:
  608. //(For efficiency, only do the following code if
  609. // the node is not already slated for deletion
  610. // because a deletable node's children will be
  611. // deleted anyway):
  612. if (!pNode->m_bDelete)
  613. {
  614.     rc = markTestAttributeNodes(pNode->m_pNodeList);
  615. }
  616. if (SUCCEEDED(rc))
  617. {
  618.     rc = removeFromNamespaceScope(pNode);
  619. }
  620.     }
  621.     return rc;
  622. }
  623. BOOL
  624. CSmilParser::firstDependentChild(SMILNode* pNode)
  625. {
  626.     if(getFirstNodeChild(pNode->m_pDependency) == pNode)
  627.     {
  628. return TRUE;
  629.     }
  630.     return FALSE;
  631. }
  632. // /Walk up parent chain until we find an ancestor of the specified tag type:
  633. SMILNode*
  634. CSmilParser::getSpecificAncestor(SMILNodeTag ancestorTag, SMILNode* pNode)
  635. {
  636.     SMILNode* pRet = NULL;
  637.     while(pNode)
  638.     {
  639. pNode = pNode->m_pParent;
  640. if(pNode  &&  (ancestorTag == pNode->m_tag) )
  641. {
  642.     pRet = pNode;
  643.     break;
  644. }
  645.     }
  646.     return pRet;
  647. }
  648. SMILNode*
  649. CSmilParser::findActiveChildOfAncestorExcl(SMILNode* pInterruptingNode,
  650.    LONG32 lCurTime)
  651. {
  652.     SMILNode* pChild = NULL;
  653.     // /If we have an ancestor excl, call it X, then we need to
  654.     // find the descendant of X that is currently playing (if any):
  655.     SMILNode* pClosestExclAncestor = getSpecificAncestor(SMILExcl,
  656.     pInterruptingNode);
  657.     // /Now, we need to dig down and find an active element at lCurTime:
  658.     if (pClosestExclAncestor)
  659.     {
  660. pChild = (findAnyActiveDescendant(pClosestExclAncestor,
  661. lCurTime, pInterruptingNode));
  662.     }
  663.     return pChild;
  664. }
  665. // /This recurses down through tree under pNode to see if there is any media
  666. // element that is playing at lCurTime; if so, returns the first one it finds
  667. SMILNode*
  668. CSmilParser::findAnyActiveDescendant(SMILNode* pNode, LONG32 lCurTime,
  669.      SMILNode* pButNotThisNode)
  670. {
  671.     SMILNode* pChild = NULL;
  672.     // /Now, we need to dig down and find an active element at lCurTime:
  673.     if (pNode)
  674.     {
  675. pChild = pNode->getFirstChild();
  676. while (pChild)
  677. {
  678.     if (pChild->m_bDelete  ||  pChild == pButNotThisNode)
  679.     {
  680. pChild = pNode->getNextChild();
  681. continue;
  682.     }
  683.     else if (isMediaObject(pChild))
  684.     {
  685. if (pChild->m_pElement->m_bInsertedIntoTimeline)
  686. {
  687.     ULONG32 ulActualStartTime = 0;
  688.     ULONG32 ulActualStopTime = 0;
  689.     HX_RESULT pnrStart =
  690.     pChild->m_pElement->getCurrentScheduledStartTime(
  691.     ulActualStartTime);
  692.     LONG32 lElementCurBeginTime = (LONG32)ulActualStartTime;
  693.     // /Using this function instead of delay+dur fixes
  694.     // PR 56760 because dur already has begin-offset-from-
  695.     // syncbase in it so we were counting it twice:
  696.     HX_RESULT pnrStop =
  697.     pChild->m_pElement->getCurrentScheduledStopTime(
  698.     ulActualStopTime);
  699.     LONG32 lElementCurEndTime = (LONG32)ulActualStopTime;
  700.     if (lElementCurBeginTime <= lCurTime  &&
  701.     lElementCurEndTime > lCurTime)
  702.     {
  703. if (!pChild->m_pElement->isPausedInExcl())
  704. {
  705.     break; // /Is currently playing; we found one.
  706. }
  707.     }
  708. }
  709.     }
  710.     else
  711.     {
  712. SMILNode* pDepthChild =
  713. findAnyActiveDescendant(pChild, lCurTime,
  714. pButNotThisNode);
  715. if (pDepthChild)
  716. {
  717.     pChild = pDepthChild; // /We found one.
  718.     break;
  719. }
  720.     }
  721.     pChild = pNode->getNextChild();
  722. }
  723.     }
  724.     return pChild;
  725. }
  726. SMILNode*
  727. CSmilParser::findLastDeferredChildOfAncestorExcl(
  728. SMILNode* pInterruptingNode, LONG32 lCurTime)
  729. {
  730.     SMILNode* pLastDeferred = NULL;
  731.     // /If we have an ancestor excl, call it X, then we need to
  732.     // find the descendant of X that is currently playing (if any):
  733.     SMILNode* pClosestExclAncestor = getSpecificAncestor(SMILExcl,
  734.     pInterruptingNode);
  735.     // /Now, we need to dig down and find the active or deferred
  736.     // element at lCurTime:
  737.     if (pClosestExclAncestor)
  738.     {
  739. pLastDeferred = (findLastDeferredDescendant(pClosestExclAncestor,
  740. lCurTime, pInterruptingNode, pLastDeferred));
  741.     }
  742.     return pLastDeferred;
  743. }
  744. SMILNode*
  745. CSmilParser::findLastDeferredDescendant(SMILNode* pNode,
  746. LONG32 lCurTime, SMILNode* pButNotThisNode, SMILNode* pLastDeferredChild)
  747. {
  748.     SMILNode* pChild = NULL;
  749.     // /Now, we need to dig down and find an active element at lCurTime:
  750.     if (pNode)
  751.     {
  752. pChild = pNode->getFirstChild();
  753. while (pChild)
  754. {
  755.     if (pChild->m_bDelete  ||  pChild == pButNotThisNode)
  756.     {
  757. pChild = pNode->getNextChild();
  758. continue;
  759.     }
  760.     // /Time container, media, or animate|set|...etc. can be deferred:
  761.     else if (isTimelineObject(pChild))
  762.     {
  763. if (pChild->m_pElement->isDeferredInExcl())
  764. {
  765.     if (!pLastDeferredChild)
  766.     {
  767. pLastDeferredChild = pChild;
  768.     }
  769.     else if (pChild->m_pElement->m_ulTimeDeferralOccurred <
  770.     pLastDeferredChild->
  771.     m_pElement->m_ulTimeDeferralOccurred)
  772.     {
  773. pLastDeferredChild = pChild;
  774.     }
  775. }
  776.     }
  777.     else
  778.     {
  779. SMILNode* pDepthChild =
  780. findLastDeferredDescendant(pChild, lCurTime,
  781. pButNotThisNode, pLastDeferredChild);
  782. if (pDepthChild)
  783. {
  784.     if (!pLastDeferredChild)
  785.     {
  786. pLastDeferredChild = pDepthChild;
  787.     }
  788.     else if (pDepthChild->m_pElement->m_ulTimeDeferralOccurred <
  789.     pLastDeferredChild->
  790.     m_pElement->m_ulTimeDeferralOccurred)
  791.     {
  792. pLastDeferredChild = pDepthChild;
  793.     }
  794. }
  795.     }
  796.     pChild = pNode->getNextChild();
  797. }
  798.     }
  799.     return pLastDeferredChild;
  800. }
  801. BOOL
  802. CSmilParser::hasAncestor(SMILNodeTag ancestor, SMILNode* pNode)
  803. {
  804.     // walk up parent chain until we find a parent of that tag type:
  805.     BOOL rc = FALSE;
  806.     while(pNode)
  807.     {
  808. pNode = pNode->m_pParent;
  809. if(pNode  &&  (ancestor == pNode->m_tag) )
  810. {
  811.     rc = TRUE;
  812.     break;
  813. }
  814.     }
  815.     return rc;
  816. }
  817. SMILNodeTag
  818. CSmilParser::getSyncTag(SMILNode* pNode)
  819. {
  820.     // walk up parent chain until we either find
  821.     // a PAR, a SEQ, or an EXCL
  822.     SMILNodeTag tag = SMILSeq; // assume a SEQ
  823.     BOOL bIsAnimate = FALSE;
  824.     if (pNode->m_tag == SMILAnimate      ||
  825.         pNode->m_tag == SMILSet          ||
  826.         pNode->m_tag == SMILAnimateColor ||
  827.         pNode->m_tag == SMILAnimateMotion)
  828.     {
  829.         bIsAnimate = TRUE;
  830.     }
  831.     while(pNode)
  832.     {
  833. pNode = pNode->m_pParent;
  834. if(pNode && pNode->m_tag == SMILPar)
  835. {
  836.             // XXXMEH - TODO - we should add the ability
  837.             // for media elements to act as time containers
  838.     tag = SMILPar;
  839.     break;
  840. }
  841. else if(pNode && pNode->m_tag == SMILSeq)
  842. {
  843.     tag = SMILSeq;
  844.     break;
  845. }
  846. else if(pNode && pNode->m_tag == SMILExcl)
  847. {
  848.     tag = SMILExcl;
  849.     break;
  850. }
  851.         else if (bIsAnimate &&
  852.                  isMediaObject(pNode))
  853.         {
  854.             tag = SMILPar;
  855.             break;
  856.         }
  857.     }
  858.     return tag;
  859. }
  860. SMILNode*
  861. CSmilParser::getSyncAncestor(SMILNode* pNode)
  862. {
  863.     // /Walk up parent chain until we either find
  864.     // a PAR, a SEQ, or an EXCL
  865.     SMILNode* pReturnNode = NULL;
  866.     SMILNode* pAncestorNode = pNode;
  867.     while(pAncestorNode)
  868.     {
  869. pAncestorNode = pAncestorNode->m_pParent;
  870. if(pAncestorNode &&
  871.     (pAncestorNode->m_tag == SMILPar ||
  872.     pAncestorNode->m_tag == SMILExcl  ||
  873.     pAncestorNode->m_tag == SMILSeq))
  874. {
  875.     pReturnNode = pAncestorNode;
  876.     break;
  877. }
  878. // /Media elements can act as time containers for area/anchor and for
  879. // animateXYZ and set elements:
  880. else if (isMediaObject(pAncestorNode)
  881. &&  (SMILArea   == pNode->m_tag       ||
  882.      SMILAnchor == pNode->m_tag       ||
  883.                      SMILAnimate == pNode->m_tag      ||
  884.                      SMILSet     == pNode->m_tag      ||
  885.                      SMILAnimateColor == pNode->m_tag ||
  886.                      SMILAnimateMotion == pNode->m_tag)
  887. )
  888. {
  889.     pReturnNode = pAncestorNode;
  890.     break;
  891. }
  892.     }
  893.     return pReturnNode;
  894. }
  895. // /Returns in ulSyncBaseTime the begin time of the sync base unless the
  896. // sync base is a prior sibling in a seq, in which case it returns that
  897. // node's element's end time.  If ulSyncBaseTime can't be obtained,
  898. // (UINT32)-1 is returned:
  899. ULONG32
  900. CSmilParser::getSyncBaseTimeInGroupTimeCoords(SMILNode* pNode)
  901. {
  902.     SMILNode* pSyncBaseNode = NULL;
  903.     SMILNode* pSyncAncestorNode = NULL;
  904.     ULONG32 ulSyncBaseTime = (UINT32)-1;
  905.     if (pNode)
  906.     {
  907. pSyncAncestorNode = getSyncAncestor(pNode);
  908.     }
  909.     if (pNode->m_pElement  &&  pNode->m_pElement->m_pTimelineElement  &&
  910.     pSyncAncestorNode  &&  pSyncAncestorNode->m_pElement)
  911.     {
  912. CSmilElement* pSyncBaseElement = pSyncAncestorNode->m_pElement;
  913. if (pSyncAncestorNode  &&  SMILSeq == pSyncAncestorNode->m_tag)
  914. {
  915.     CHXSimpleList* pSyncAncestorChildList = NULL;
  916.     // /SyncBase of seq child is end of prior sibling, not parent's
  917.     // begin, unless no prior sibling exists:
  918.     if (pSyncBaseElement->m_pTimelineElement  &&  NULL !=
  919.     (pSyncAncestorChildList =
  920.     pSyncBaseElement->m_pTimelineElement->getChildren()) )
  921.     {
  922. CHXSimpleList::Iterator i = pSyncAncestorChildList->Begin();
  923. CSmilTimelineElement* pPrevTimelineElement = NULL;
  924. for(; i != pSyncAncestorChildList->End(); ++i)
  925. {
  926.     CSmilTimelineElement* pTimelineElement =
  927.     (CSmilTimelineElement*)(*i);
  928.     if (pTimelineElement ==
  929.     pNode->m_pElement->m_pTimelineElement)
  930.     {
  931. // /Found prev timed sibling (or it is NULL):
  932. if (pPrevTimelineElement)
  933. {
  934.     pSyncBaseElement =
  935.     pPrevTimelineElement->getSourceElement();
  936.     HX_ASSERT(pSyncBaseElement);
  937. }
  938. else
  939. {
  940.     pSyncBaseElement = pSyncAncestorNode->m_pElement;
  941. }
  942. break;
  943.     }
  944.     pPrevTimelineElement = pTimelineElement;
  945. }
  946.     }
  947.     // /If pNode is a timeline element, its sync ancestor should at
  948.     // least have pNode's timeline element in its child list:
  949.     HX_ASSERT(pSyncAncestorChildList);
  950. }
  951. if (pSyncBaseElement)
  952. {
  953.     HX_RESULT pnrStart = HXR_OK;
  954.     // /If syncBaseElement is actually the parent seq, then use its
  955.     // begin time, not its end time, as the sync base:
  956.     if (pSyncBaseElement == pSyncAncestorNode->m_pElement)
  957.     {
  958. pnrStart = pSyncBaseElement->getCurrentScheduledStartTime(
  959. ulSyncBaseTime);
  960.     }
  961.     else // /Use end of prior seq sibling as sync base:
  962.     {
  963. pnrStart = pSyncBaseElement->getCurrentScheduledStopTime(
  964. ulSyncBaseTime);
  965.     }
  966. }
  967. else
  968. {
  969.     ulSyncBaseTime = (UINT32)-1;
  970. }
  971.     }
  972.     return(ulSyncBaseTime);
  973. }
  974. BOOL
  975. CSmilParser::inSeq(SMILNode* pNode)
  976. {
  977.     SMILNodeTag tag = getSyncTag(pNode);
  978.     return tag == SMILSeq;
  979. }
  980. HX_RESULT CSmilParser::createHeadElements(SMILNodeList* pNodeList)
  981. {
  982.     HX_RESULT retVal = HXR_OK;
  983.     if (pNodeList)
  984.     {
  985.         LISTPOSITION pos = pNodeList->GetHeadPosition();
  986.         while (pos && SUCCEEDED(retVal))
  987.         {
  988.             SMILNode* pNode = (SMILNode*) pNodeList->GetAt(pos);
  989.             if (pNode && !pNode->m_bDelete)
  990.             {
  991.                 retVal = addToNamespaceScope(pNode);
  992.                 if (SUCCEEDED(retVal))
  993.                 {
  994.                     // Get the tag of the parent
  995.                     SMILNodeTag eParentTag = SMILUnknown;
  996.                     if (pNode->m_pParent)
  997.                     {
  998.                         eParentTag = pNode->m_pParent->m_tag;
  999.                     }
  1000.     // /XXXEH- TODO: do what makeTransition() and
  1001.     // makeCustomTest() do, below, and use their actual
  1002.     // failed retVals rather than just HXR_FAIL if anything
  1003.     // goes wrong:
  1004.                     // Init the return value to failure
  1005.                     retVal = HXR_FAIL;
  1006.                     // Switch based on tag type
  1007.                     switch(pNode->m_tag)
  1008.                     {
  1009.                         case SMILRegion:
  1010.                         {
  1011.                             if (eParentTag == SMILBasicLayout ||
  1012.                                 eParentTag == SMILRegion      ||
  1013.                                 eParentTag == SMILViewport)
  1014.                             {
  1015.                                 CSmilRegion* pRegion = makeRegion(pNode);
  1016.                                 if(pRegion)
  1017.                                 {
  1018.                                     pNode->m_pElement      = pRegion;
  1019.                                     pRegion->m_ulDelay     = 0;
  1020.                                     pRegion->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1021.                                     insertElementByTimestamp(pRegion);
  1022.                                     retVal           = HXR_OK;
  1023.                                 }
  1024.                             }
  1025.                         }
  1026.                         break;
  1027.                         case SMILTransition:
  1028.                         {
  1029. #if defined(HELIX_FEATURE_SMIL2_TRANSITIONS)
  1030.                             if (eParentTag == SMILHead)
  1031.                             {
  1032.                                 CSmilTransition* pTrans = makeTransition(pNode, retVal);
  1033.                                 if (pTrans && SUCCEEDED(retVal))
  1034.                                 {
  1035.                                     pNode->m_pElement     = pTrans;
  1036.                                     pTrans->m_ulDelay     = 0;
  1037.                                     pTrans->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1038.                                     insertElementByTimestamp(pTrans);
  1039.                                 }
  1040.                             }
  1041. #else
  1042.                             pNode->m_bDelete = TRUE;
  1043.                             retVal           = HXR_OK;
  1044. #endif /* #if defined(HELIX_FEATURE_SMIL2_TRANSITIONS) */
  1045.                         }
  1046.                         break;
  1047.                         case SMILMeta:
  1048.                         {
  1049.                             if (eParentTag == SMILHead)
  1050.                             {
  1051.                                 CSmilMeta* pMeta = makeMeta(pNode);
  1052.                                 if(pMeta)
  1053.                                 {
  1054.                                     pNode->m_pElement    = pMeta;
  1055.                                     pMeta->m_ulDelay     = 0;
  1056.                                     pMeta->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1057.                                     insertElementByTimestamp(pMeta);
  1058.                                     retVal = HXR_OK;
  1059.                                 }
  1060.                             }
  1061.                         }
  1062.                         break;
  1063.                         case SMILMetadata:
  1064.                         {
  1065.                             if (eParentTag == SMILHead)
  1066.                             {
  1067.                                 CSmilMetadata* pMetadata=makeMetadata(pNode);
  1068.                                 if(pMetadata)
  1069.                                 {
  1070.                                     pNode->m_pElement    = pMetadata;
  1071.                                     pMetadata->m_ulDelay     = 0;
  1072.                                     pMetadata->m_ulTimestamp =
  1073.     INITIAL_STREAM0_TIMESTAMP;
  1074.                                     insertElementByTimestamp(pMetadata);
  1075.                                     retVal = HXR_OK;
  1076.                                 }
  1077.                             }
  1078.                         }
  1079.                         break;
  1080.                         case SMILRendererPreFetch:
  1081.                         {
  1082.                             if (eParentTag == SMILRNRendererList)
  1083.                             {
  1084.                                 CSmilRendererPreFetch* pRenderer =
  1085.                                     makeRendererPreFetch(pNode);
  1086.                                 pNode->m_pElement         = pRenderer;
  1087.                                 pRenderer->m_ulDelay      = 0;
  1088.                                 pRenderer->m_ulTimestamp  = INITIAL_STREAM0_TIMESTAMP;
  1089.                                 insertElementByTimestamp(pRenderer);
  1090.                                 retVal = HXR_OK;
  1091.                             }
  1092.                         }
  1093.                         break;
  1094.                         case SMILRootLayout:
  1095.                         {
  1096.                             if (eParentTag == SMILBasicLayout)
  1097.                             {
  1098.                                 CSmilRootLayout* pRoot = makeRootLayout(pNode);
  1099.                                 if(pRoot)
  1100.                                 {
  1101.                                     pNode->m_pElement    = pRoot;
  1102.                                     pRoot->m_ulDelay     = 0;
  1103.                                     pRoot->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1104.                                     insertElementByTimestamp(pRoot);
  1105.                                     retVal           = HXR_OK;
  1106.                                 }
  1107.                             }
  1108.                         }
  1109.                         break;
  1110.                         case SMILRegPoint:
  1111.                         {
  1112.                             if (eParentTag == SMILBasicLayout)
  1113.                             {
  1114.                                 CSmilRegPoint* pRegPt = makeRegPoint(pNode);
  1115.                                 if(pRegPt)
  1116.                                 {
  1117.                                     pNode->m_pElement     = pRegPt;
  1118.                                     pRegPt->m_ulDelay     = 0;
  1119.                                     pRegPt->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1120.                                     insertElementByTimestamp(pRegPt);
  1121.                                     retVal           = HXR_OK;
  1122.                                 }
  1123.                             }
  1124.                         }
  1125.                         break;
  1126.                         case SMILViewport:
  1127.                         {
  1128. #if defined(HELIX_FEATURE_SMIL2_MULTIWINDOWLAYOUT)
  1129.                             if (eParentTag == SMILBasicLayout)
  1130.                             {
  1131.                                 CSmilViewport* pView = makeViewport(pNode);
  1132.                                 if(pView)
  1133.                                 {
  1134.                                     pNode->m_pElement       = pView;
  1135.                                     pView->m_ulDelay        = 0;
  1136.                                     pView->m_ulTimestamp    = INITIAL_STREAM0_TIMESTAMP;
  1137.                                     insertElementByTimestamp(pView);
  1138.                                     retVal           = HXR_OK;
  1139.                                 }
  1140.                             }
  1141. #else
  1142.                             // topLayout is not supported, so we
  1143.                             // delete the node
  1144.                             pNode->m_bDelete = TRUE;
  1145.                             // Clear the return value
  1146.                             retVal = HXR_OK;
  1147. #endif /* #if defined(HELIX_FEATURE_SMIL2_MULTIWINDOWLAYOUT) */
  1148.                         }
  1149.                         break;
  1150.                         case SMILBasicLayout:
  1151.                         {
  1152.     retVal = HXR_OK;
  1153.                         }
  1154.                         break;
  1155. case SMILCustomTest:
  1156. {
  1157.                             if (eParentTag == SMILCustomAttributes)
  1158.                             {
  1159.                                 CSmilCustomTest* pCustomTest =
  1160. makeCustomTest(pNode, retVal);
  1161.                                 if (pCustomTest  &&  SUCCEEDED(retVal))
  1162.                                 {
  1163.                                     pNode->m_pElement        = pCustomTest;
  1164.                                     pCustomTest->m_ulDelay     = 0;
  1165.                                     pCustomTest->m_ulTimestamp =
  1166.     INITIAL_STREAM0_TIMESTAMP;
  1167.                                     insertElementByTimestamp(pCustomTest);
  1168.                                     retVal = HXR_OK;
  1169.     if(!m_pCustomTestMap)
  1170.     {
  1171. m_pCustomTestMap =
  1172. new CHXMapStringToOb;
  1173.     }
  1174.     if (m_pCustomTestMap)
  1175.     {
  1176. (*m_pCustomTestMap)[pNode->m_id] =
  1177. pCustomTest;
  1178.     }
  1179.     else
  1180.     {
  1181. retVal = HXR_OUTOFMEMORY;
  1182.     }
  1183.                                 }
  1184.                             }
  1185. }
  1186. break;
  1187.                         case SMILUnknown:
  1188.                         {
  1189.                             // If this is a </layout>, then we
  1190.                             // generate a close layout if we
  1191.                             // have had any regions
  1192.                             if (pNode->m_id == "CLOSE-layout")
  1193.                             {
  1194.                                 CSmilEndLayout* pEndLayout = new CSmilEndLayout();
  1195.                                 if (pEndLayout)
  1196.                                 {
  1197.                                     pNode->m_pElement         = pEndLayout;
  1198.                                     pEndLayout->m_ulDelay     = 0;
  1199.                                     pEndLayout->m_ulTimestamp = INITIAL_STREAM0_TIMESTAMP;
  1200.                                     insertElementByTimestamp(pEndLayout);
  1201.                                     retVal = HXR_OK;
  1202.                                 }
  1203.                             }
  1204.                             else
  1205.                             {
  1206.                                 // Clear the return value
  1207.                                 retVal = HXR_OK;
  1208.                             }
  1209.                         }
  1210.                         break;
  1211.                         default:
  1212.                             retVal = HXR_OK;
  1213.                     }
  1214.                     // Now examine the children
  1215.                     if (SUCCEEDED(retVal))
  1216.                     {
  1217.                         retVal = createHeadElements(pNode->m_pNodeList);
  1218.                         if (SUCCEEDED(retVal))
  1219.                         {
  1220.                             retVal = removeFromNamespaceScope(pNode);
  1221.                         }
  1222.                     }
  1223.                 }
  1224.             }
  1225.             // Advance to the next node
  1226.             if (SUCCEEDED(retVal))
  1227.             {
  1228.                 pNodeList->GetNext(pos);
  1229.             }
  1230.         }
  1231.     }
  1232.     return retVal;
  1233. }
  1234. HX_RESULT
  1235. CSmilParser::setInitialDelays(SMILNodeList* pNodeList)
  1236. {
  1237.     HX_RESULT rc = HXR_OK;
  1238.     UINT16 nGroup = (UINT16)-1;
  1239.     if(!pNodeList)
  1240.     {
  1241. return rc;
  1242.     }
  1243.     CHXSimpleList::Iterator i = pNodeList->Begin();
  1244.     for(;i!=pNodeList->End();++i)
  1245.     {
  1246. if(HXR_OK != rc)
  1247. {
  1248.     return rc;
  1249. }
  1250. SMILNode* pNode = (SMILNode*)(*i);
  1251. if(pNode->m_bDelete) // skip this puppy
  1252. {
  1253.     continue;
  1254. }
  1255. else if(pNode->m_tag == SMILAAnchor ||
  1256.         pNode->m_tag == SMILPriorityClass  ||
  1257.         pNode->m_tag == SMILSwitch)
  1258. {
  1259.     pNode = getTimelineDescendent(pNode);
  1260.     if(!pNode)
  1261.     {
  1262. continue;
  1263.     }
  1264. }
  1265. setInitialDelay(pNode);
  1266. switch(pNode->m_tag)
  1267. {
  1268.     case SMILSeq:
  1269.     {
  1270. setInitialDelayOnSeq(pNode);
  1271.     }
  1272.     break;
  1273.     case SMILExcl:
  1274.     {
  1275. HX_ASSERT(0 && "Excl can't be outer time container");
  1276.     }
  1277.     // /fall on through to SMILPar code:
  1278.     case SMILPar:
  1279.     {
  1280. SMILNode* pThisNode =
  1281.     getTimelineDescendent(pNode, NULL);
  1282. while(pThisNode)
  1283. {
  1284.     setInitialDelay(pThisNode);
  1285.     pThisNode =
  1286. getTimelineDescendent(pNode, pThisNode);
  1287. }
  1288.     }
  1289.     break;
  1290. }
  1291.     }
  1292.     return rc;
  1293. }
  1294. void
  1295. CSmilParser::setInitialDelay(SMILNode* pNode)
  1296. {
  1297.     if(pNode->m_pElement &&
  1298. pNode->m_pElement->m_pTimelineElement &&
  1299. !pNode->m_pElement->m_pTimelineElement->initialDelaySet())
  1300.     {
  1301. ULONG32 ulInitialDelay = 0;
  1302. // /This fixes default begin of excl children (which is not 0 but
  1303. // rather indefinite):
  1304. if (pNode->m_pElement->m_bBeginOffsetSet  ||
  1305. !hasAncestor(SMILExcl, pNode))
  1306. {
  1307.     // HP m_ulPersistentComponentDelay is 0 except in nested meta
  1308.     pNode->m_pElement->m_pTimelineElement->setDelay(m_ulPersistentComponentDelay, FALSE);
  1309. }
  1310.     }
  1311.     if(pNode->m_pNodeList)
  1312.     {
  1313. if(pNode->m_tag == SMILSeq ||
  1314.     pNode->m_tag == SMILExcl  ||
  1315.     pNode->m_tag == SMILPar)
  1316. {
  1317.     if(pNode->m_pElement &&
  1318. pNode->m_pElement->m_pTimelineElement &&
  1319. pNode->m_pElement->m_ulDuration != (UINT32)-1)
  1320.     {
  1321. pNode->m_pElement->m_pTimelineElement->setDuration(
  1322.     pNode->m_pElement->m_ulDuration);
  1323.     }
  1324. }
  1325. if(pNode->m_tag == SMILSeq)
  1326. {
  1327.     setInitialDelayOnSeq(pNode);
  1328. }
  1329. //If the element doesn't exist or it does but the timelineElement
  1330. // doesn't exist, or it does but there is no delay event, or there is
  1331. // one but the delay is already set, then it's ok to set call
  1332. // setInitialDelay():
  1333. else if ((!pNode->m_pElement  ||
  1334. !pNode->m_pElement->m_pTimelineElement)  ||
  1335. //[SMIL 1.0 Compliance] Helps fix PR 14420:
  1336. //Don't call this if pNode has an unresolved delay event:
  1337. (!pNode->m_pElement->m_pTimelineElement->delayEvent()  ||
  1338. pNode->m_pElement->m_pTimelineElement->initialDelaySet() ) )
  1339. {
  1340.     SMILNode* pThisNode =
  1341. getTimelineDescendent(pNode, NULL);
  1342.     while(pThisNode)
  1343.     {
  1344. setInitialDelay(pThisNode);
  1345. pThisNode = getTimelineDescendent(pNode, pThisNode);
  1346.     }
  1347. }
  1348.     }
  1349. }
  1350. void
  1351. CSmilParser::setInitialDelayOnSeq(SMILNode* pNode)
  1352. {
  1353.     UINT16 nGroup = (UINT16)-1;
  1354.     SMILNode* pThisNode = getTimelineDescendent(pNode, NULL);
  1355.     while(pThisNode)
  1356.     {
  1357. if(pThisNode->m_nGroup != nGroup)
  1358. {
  1359.     nGroup = pThisNode->m_nGroup;
  1360.     setInitialDelay(pThisNode);
  1361. }
  1362. pThisNode = getTimelineDescendent(pNode, pThisNode);
  1363.     }
  1364. }
  1365. HX_RESULT
  1366. CSmilParser::assignGroupIndexes(SMILNodeList* pNodeList)
  1367. {
  1368.     HX_RESULT rc = HXR_OK;
  1369.     if(!pNodeList)
  1370.     {
  1371. return rc;
  1372.     }
  1373.     UINT16 nGroup = 0;
  1374.     UINT16 nNum = 0;
  1375.     CHXSimpleList::Iterator i = pNodeList->Begin();
  1376.     for(; i != pNodeList->End(); ++i)
  1377.     {
  1378. if(HXR_OK != rc)
  1379. {
  1380.     return rc;
  1381. }
  1382. SMILNode* pNode = (SMILNode*)(*i);
  1383. if(pNode->m_bDelete) // skip this puppy
  1384. {
  1385.     continue;
  1386. }
  1387. else if(pNode->m_tag == SMILAAnchor ||
  1388.         pNode->m_tag == SMILSwitch)
  1389. {
  1390.     pNode = getTimelineDescendent(pNode);
  1391.     if(!pNode)
  1392.     {
  1393. continue;
  1394.     }
  1395. }
  1396. switch(pNode->m_tag)
  1397. {
  1398.     case SMILSeq:
  1399.     {
  1400. SMILNode* pThisNode =
  1401.     getTimelineDescendent(pNode, NULL);
  1402. while(pThisNode)
  1403. {
  1404.     rc = assignGroupIndexOnSeq(pThisNode, nGroup);
  1405.     if(HXR_OK == rc)
  1406.     {
  1407. pThisNode =
  1408.     getTimelineDescendent(pNode, pThisNode);
  1409. nGroup++;
  1410.     }
  1411.     else
  1412.     {
  1413. break;
  1414.     }
  1415. }
  1416.     }
  1417.     break;
  1418.     case SMILExcl:
  1419.     case SMILPar:
  1420.     {
  1421. // top-level par, each child has group 0
  1422. SMILNode* pThisNode =
  1423.     getTimelineDescendent(pNode, NULL);
  1424. while(pThisNode)
  1425. {
  1426.     rc = assignGroupIndexOnPar(pThisNode, nGroup);
  1427.     if(HXR_OK == rc)
  1428.     {
  1429. pThisNode =
  1430.     getTimelineDescendent(pNode, pThisNode);
  1431.     }
  1432.     else
  1433.     {
  1434. break;
  1435.     }
  1436. }
  1437.     }
  1438.     break;
  1439.     default:
  1440.     break;
  1441. }
  1442.     }
  1443.     // Assign the number of groups
  1444.     m_usNumGroups = nGroup;
  1445.     return rc;
  1446. }
  1447. HX_RESULT
  1448. CSmilParser::assignGroupIndexOnPar(SMILNode* pNode,
  1449.    UINT16 nGroup)
  1450. {
  1451.     HX_RESULT rc = HXR_OK;
  1452.     if(pNode->m_bDelete)
  1453.     {
  1454. return rc;
  1455.     }
  1456.     pNode->m_nGroup = nGroup;
  1457.     pNode->m_repeatid.AppendULONG(nGroup);
  1458.     // /Added check for isNonMediaPlayableObject since animate objects, for
  1459.     // instance, can exist in a group without any media siblings, and we
  1460.     // want them to be able to "play" (e.g., animate a region attribute):
  1461.     if ((isMediaObject(pNode)  ||  isNonMediaPlayableObject(pNode))
  1462.     &&  !pNode->m_bDelete)
  1463.     {
  1464. // /XXXEH- NOT SURE YET HOW TO HANDLE THIS; we should also assign
  1465. // entire tree underneath this if it has children since they will
  1466. // be in same player (except the ones that explicitly declare that
  1467. // they are independent):
  1468. // /Don't do this if sync behavior is "independent":
  1469. if (SmilSyncBehaviorIndependent != pNode->m_pElement->m_syncBehavior)
  1470. {
  1471.     addGroup(pNode);
  1472.     // XXXMEH - we also have to look at
  1473.     // children of media objects, and assign
  1474.     // their group indices as well.
  1475.     SMILNode* pThisNode = getTimelineDescendent(pNode, NULL);
  1476.     UINT16 nNum = 0;
  1477.     while(pThisNode)
  1478.     {
  1479. rc = assignGroupIndexOnPar(pThisNode, nGroup);
  1480. pThisNode->m_num = nNum++;
  1481. if(HXR_OK == rc)
  1482. {
  1483.     pThisNode =
  1484. getTimelineDescendent(pNode, pThisNode);
  1485. }
  1486. else
  1487. {
  1488.     break;
  1489. }
  1490.     }
  1491. }
  1492. else
  1493. {
  1494.     // /XXXEH- handle independent and canSlip player groups here:
  1495.     HX_ASSERT(0  &&  "ehodge: handle addGroup for indep player");
  1496. }
  1497.     }
  1498.     else if (SMILSeq == pNode->m_tag ||
  1499.     SMILExcl == pNode->m_tag  ||
  1500.            SMILPar == pNode->m_tag)
  1501.     {
  1502. // top-level par, each child has group 0
  1503. SMILNode* pThisNode =
  1504. getTimelineDescendent(pNode, NULL);
  1505. UINT16 nNum = 0;
  1506. while(pThisNode)
  1507. {
  1508.     rc = assignGroupIndexOnPar(pThisNode, nGroup);
  1509.     pThisNode->m_num = nNum++;
  1510.     if(HXR_OK == rc)
  1511.     {
  1512. pThisNode =
  1513.     getTimelineDescendent(pNode, pThisNode);
  1514.     }
  1515.     else
  1516.     {
  1517. break;
  1518.     }
  1519. }
  1520.     }
  1521.     return rc;
  1522. }
  1523. HX_RESULT
  1524. CSmilParser::assignGroupIndexOnSeq(SMILNode* pNode,
  1525.    UINT16& nGroup)
  1526. {
  1527.     HX_RESULT rc = HXR_OK;
  1528.     if(pNode->m_bDelete)
  1529.     {
  1530.       return rc;
  1531.     }
  1532.     pNode->m_nGroup = nGroup;
  1533.     pNode->m_repeatid.AppendULONG(nGroup);
  1534.     // /Added check for isNonMediaPlayableObject since animate objects, for
  1535.     // instance, can exist in a group without any media siblings, and we
  1536.     // want them to be able to "play" (e.g., animate a region attribute):
  1537.     if ((isMediaObject(pNode)  ||  isNonMediaPlayableObject(pNode))
  1538.     &&  !pNode->m_bDelete)
  1539.     {
  1540.         addGroup(pNode);
  1541.         // XXXMEH - we also need to look at
  1542.         // children of media elements and assign
  1543.         // their group indices as well. Treat time
  1544.         // containers like <par>'s.
  1545.         SMILNode* pThisNode = getTimelineDescendent(pNode, NULL);
  1546.         while(pThisNode)
  1547.         {
  1548.             rc = assignGroupIndexOnPar(pThisNode, nGroup);
  1549.             if(HXR_OK == rc)
  1550.             {
  1551.                 pThisNode = getTimelineDescendent(pNode, pThisNode);
  1552.             }
  1553.             else
  1554.             {
  1555.                 break;
  1556.             }
  1557.         }
  1558.     }
  1559.     else if (SMILSeq == pNode->m_tag)
  1560.     {
  1561.       SMILNode* pThisNode =
  1562.           getTimelineDescendent(pNode, NULL);
  1563.       while(pThisNode)
  1564.       {
  1565.           rc = assignGroupIndexOnSeq(pThisNode, nGroup);
  1566.           if(HXR_OK == rc)
  1567.           {
  1568.               pThisNode = getTimelineDescendent(pNode, pThisNode);
  1569.               if (pThisNode)
  1570.               {
  1571.                   if (!pThisNode->m_bDelete  &&
  1572.       (isMediaObject(pThisNode)  ||
  1573.                       isTimeContainerObject(pThisNode))  &&
  1574.       // /Fixes PR 54268: if pNode isn't the outer wrapper seq
  1575.       // (in a file where the body has multiple children so
  1576.       // we'll play it as a clip list) then we don't want to
  1577.       // put each of pNode's children into a separate group:
  1578.       pNode->m_bIsOuterWrapperTimeContainer)
  1579.                   {
  1580.                       nGroup++;
  1581.                   }
  1582.               }
  1583.           }
  1584.           else
  1585.           {
  1586.               break;
  1587.           }
  1588.       }
  1589.     }
  1590.     else if (SMILPar == pNode->m_tag  ||  SMILExcl == pNode->m_tag)
  1591.     {
  1592.       SMILNode* pThisNode =
  1593.           getTimelineDescendent(pNode, NULL);
  1594.       while(pThisNode)
  1595.       {
  1596.           rc = assignGroupIndexOnPar(pThisNode, nGroup);
  1597.           if(HXR_OK == rc)
  1598.           {
  1599.               pThisNode =
  1600.                   getTimelineDescendent(pNode, pThisNode);
  1601.           }
  1602.           else
  1603.           {
  1604.               break;
  1605.           }
  1606.       }
  1607.     }
  1608.     return rc;
  1609. }
  1610. BOOL
  1611. CSmilParser::isTimelineObject(SMILNode* pNode)
  1612. {
  1613.     BOOL bResult = isMediaObject(pNode)  ||
  1614.     isTimeContainerObject(pNode);
  1615.     if (!bResult)
  1616.     {
  1617. switch (pNode->m_tag)
  1618. {
  1619.     case SMILTransition:
  1620.     case SMILAnimate:
  1621.     case SMILAnimateMotion:
  1622.     case SMILAnimateColor:
  1623.     {
  1624.       bResult = TRUE;
  1625.     }
  1626.     break;
  1627.     default:
  1628.     break;
  1629. }
  1630.     }
  1631.     return bResult;
  1632. }
  1633. BOOL
  1634. CSmilParser::isMediaObject(SMILNode* pNode)
  1635. {
  1636.     BOOL bResult = FALSE;
  1637.     if (NULL == pNode)
  1638.     {
  1639.       goto cleanup;
  1640.     }
  1641.     switch(pNode->m_tag)
  1642.     {
  1643.       case SMILRef:
  1644.       case SMILText:
  1645.       case SMILImg:
  1646.       case SMILAudio:
  1647.       case SMILVideo:
  1648.       case SMILAnimation:
  1649.       case SMILTextstream:
  1650.       case SMILBrush:
  1651.       case SMILPrefetch:
  1652.         {
  1653.           bResult = TRUE;
  1654.         }
  1655.       break;
  1656.       default:
  1657.       break;
  1658.       }
  1659. cleanup:
  1660.     return bResult;
  1661. }
  1662. // /This is for handling cases where only animate... element(s) are inside
  1663. // the first child of the body element and we want a group to be created so
  1664. // a timeline longer than zero will play.  This is usefull if you want to
  1665. // have some regions (that get altered over time) be the entire presentation:
  1666. BOOL
  1667. CSmilParser::isNonMediaPlayableObject(SMILNode* pNode)
  1668. {
  1669.     BOOL bResult = FALSE;
  1670.     if (NULL == pNode)
  1671.     {
  1672.       goto cleanup;
  1673.     }
  1674.     switch(pNode->m_tag)
  1675.     {
  1676. case SMILAnimate:
  1677. case SMILAnimateColor:
  1678. case SMILAnimateMotion:
  1679. {
  1680.     bResult = TRUE;
  1681.         }
  1682. break;
  1683. default:
  1684. break;
  1685.     }
  1686. cleanup:
  1687.     return bResult;
  1688. }
  1689. BOOL
  1690. CSmilParser::isTimeContainerObject(SMILNode* pNode)
  1691. {
  1692.     BOOL bResult = FALSE;
  1693.     if (NULL == pNode)
  1694.     {
  1695. goto cleanup;
  1696.     }
  1697.     switch(pNode->m_tag)
  1698.     {
  1699. case SMILSeq:
  1700. case SMILExcl:
  1701. case SMILPar:
  1702. {
  1703.     bResult = TRUE;
  1704. }
  1705. break;
  1706. default:
  1707. break;
  1708.     }
  1709. cleanup:
  1710.     return bResult;
  1711. }
  1712. BOOL
  1713. CSmilParser::isEndTagObject(SMILNode* pNode)
  1714. {
  1715.     BOOL bResult = FALSE;
  1716.     if (NULL == pNode)
  1717.     {
  1718. goto cleanup;
  1719.     }
  1720.     switch(pNode->m_tag)
  1721.     {
  1722. case SMILEndSeq:
  1723. case SMILEndExcl:
  1724. case SMILEndPar:
  1725. case SMILEndAAnchor:
  1726. case SMILEndPriorityClass:
  1727. {
  1728.     bResult = TRUE;
  1729. }
  1730. break;
  1731. default:
  1732. break;
  1733.     }
  1734. cleanup:
  1735.     return bResult;
  1736. }
  1737. HX_RESULT
  1738. CSmilParser::addGroup(SMILNode* pNode)
  1739. {
  1740.     HX_RESULT rc = HXR_OK;
  1741.     if (pNode &&
  1742.         (pNode->m_tag != SMILAnimate &&
  1743.          pNode->m_tag != SMILAnimateMotion &&
  1744.          pNode->m_tag != SMILSet &&
  1745.          pNode->m_tag != SMILAnimateColor))
  1746.     {
  1747.         // set group info information
  1748.         CSmilAddGroup* pAddGroup = 0;
  1749.         if(m_pAddGroupMap->Lookup(pNode->m_nGroup, (void*&)pAddGroup))
  1750.         {
  1751.     // we don't count self-replicated elements as they
  1752.     // share the same track ID
  1753.     if (pNode->m_repeatTag == RepeatUnknown)
  1754.     {
  1755.         pAddGroup->m_nTotalTracks++;
  1756.         if(pNode->m_pElement &&
  1757.            pNode->m_pElement->m_ulDelay == 0)
  1758.         {
  1759.     pAddGroup->m_nInitTracks++;
  1760.         }
  1761.     }
  1762.         }
  1763.         else
  1764.         {
  1765.     CSmilAddGroup* pAddGroup = new CSmilAddGroup;
  1766.     pAddGroup->m_nGroup = pNode->m_nGroup;
  1767.     // reference PAR or EXCL properties in group
  1768.     if(pNode->m_pDependency &&
  1769.            (pNode->m_pDependency->m_tag == SMILPar  ||
  1770.            pNode->m_pDependency->m_tag == SMILExcl)  &&
  1771.            // /See if it has a par and/or excl ancestor:
  1772.            (hasAncestor(SMILPar, pNode)  ||
  1773.            hasAncestor(SMILExcl, pNode)) )
  1774.     {
  1775.         SMILNode* pDepNode = pNode->m_pDependency;
  1776. #if defined(DONT_ALLOW_PLAYLIST_STYLE_SEQ_IF_EXPLICIT_OUTER_TIMECONTAINER)
  1777. // /Fixes case 1 of PR 56364: clip-list file's first
  1778. // clip's explicit dur was being used for 2nd clip's dur
  1779. // (since, in that case, the m_pDependency node is not a
  1780. // parent but rather a prior sibling in a seq):
  1781. if (!m_bAllowPlaylistBehavior)
  1782. {
  1783.     // /The following fixes par-in-seq version of PR 50715 and 50724:
  1784.     // (fix Interop Timing #9.5 &#9.+{7,11,22,24,27,29,33,34,41,42},
  1785.     // some of which are fixed for seq-in-seq, further down.)
  1786.     // e.g.,
  1787.     // <body>
  1788.     //   <seq>
  1789.     //     <par dur="10s">
  1790.     //       <ref src="...">
  1791.     //     </par>
  1792.     //     <ref dur="4s">
  1793.     //   </seq>
  1794.     // </body>
  1795.     // Used to end at 10s (second clip didn't play) but now we take
  1796.     //.into account the fact that pNode's parent may not contain
  1797.     // the entire group's sources:
  1798.     // /Step up through ancestors until you hit body or smil, then
  1799.     // back down one and use that element's explicit dur, if any:
  1800.     // We need to make sure that we're not a child of a group that
  1801.     // has other children:
  1802.     SMILNode* pNextDepNode = (pDepNode?
  1803.     pDepNode->m_pDependency : NULL);
  1804.     while (pDepNode->m_pDependency  &&
  1805.     !(pNextDepNode->m_tag == SMILBody  ||
  1806.     (pNextDepNode->m_tag == SMILSeq  &&
  1807.     pNextDepNode->m_id == "body")  ||
  1808.     pNextDepNode->m_bIsOuterWrapperTimeContainer) )
  1809.     {
  1810. pDepNode = pNextDepNode;
  1811. pNextDepNode = (pDepNode?
  1812.     pDepNode->m_pDependency : NULL);
  1813.     }
  1814. }
  1815.         HX_ASSERT(pDepNode);
  1816. #endif
  1817.         IHXValues* pValues = pDepNode->m_pValues;
  1818.         if(pValues)
  1819.         {
  1820.     pAddGroup->m_pValues = pValues;
  1821.     pAddGroup->m_pValues->AddRef();
  1822.         }
  1823. // /Once core is fixed so that group duration can be changed on
  1824. // the fly, we'll just always set the group dur to unresolved,
  1825. // and let resolveGroupDurToOuterTimeContainerDur() take care of
  1826. // setting the overall duration.  I tested this with over 200
  1827. // files and they all worked as expected:
  1828. #define PR79703_WORK_AROUND
  1829. #if defined(PR79703_WORK_AROUND)
  1830. // /"Fixes" PR 79703 (until group dur can be changed in core):
  1831. pAddGroup->m_ulDuration = 0xFFFFFFFF;
  1832. #else
  1833. pAddGroup->m_ulDuration = pDepNode->m_pElement->m_ulDuration;
  1834. #endif
  1835.         // XXX HP remove the following if when XXXEH fix the same for
  1836.         // body duration
  1837.         if (m_ulPersistentComponentDuration)
  1838.         {
  1839.     pAddGroup->m_ulDuration = m_ulPersistentComponentDuration;
  1840.         }
  1841.         // /Fixes PR 52452: <body><par begin="5s" end="30s">...
  1842.         // was playing as 25s presentation instead of 30s (even
  1843.         // though video would play beyond that for 5s):
  1844.         if (pDepNode->m_pElement->m_bCurBeginIsOffsetFromSyncBase  &&
  1845.         (UINT32)-1 != pAddGroup->m_ulDuration)
  1846.         {
  1847.     pAddGroup->m_ulDuration +=
  1848.     pDepNode->m_pElement->m_ulBeginOffsetFromSyncBase;
  1849.         }
  1850.     }
  1851.     // XXXJHUG  -- If the group is from a Seq, we still want
  1852.     // to pass along the Title, author, copyright & abstract from
  1853.     // the dependant parrent sequence group.
  1854.     if (pNode->m_pDependency &&
  1855.         pNode->m_pDependency->m_tag == SMILSeq)
  1856.     {
  1857.         // the duration of a sequence is dependent on the elements
  1858.         // within the sequence.
  1859.         // ...Ah, but not always; if there is an explicit dur or end for the
  1860.         // seq, then we want to use that dur.  The rest of this code in this
  1861.         // block was added to account for this, fixing many SMIL2 Timing
  1862.         // interop bugs including #9.38 "fill_freeze_with_repeat.smil":
  1863.         // /Step up through ancestors until you hit body element or an
  1864.         // outer seq or par wrapping time container (if there is one);
  1865.         // Then, back down one and use that element's explicit dur, if
  1866.         // any:  We need to make sure that we're not a child of a group
  1867.         // that has other children:
  1868.         SMILNode* pDepNode = pNode->m_pDependency;
  1869. #if defined(DONT_ALLOW_PLAYLIST_STYLE_SEQ_IF_EXPLICIT_OUTER_TIMECONTAINER)
  1870.         // /The following fixes seq-in-seq version of PR 50715 and 50724:
  1871.         // (fix Interop Timing #9.5 &#9.+{7,11,22,24,27,29,33,34,41,42},
  1872.         // some of which are fixed for seq-in-seq, further down.)
  1873.         SMILNode* pNextDepNode = pDepNode?
  1874.         pDepNode->m_pDependency : NULL;
  1875.         while (pDepNode->m_pDependency  &&
  1876.         !(pNextDepNode->m_tag == SMILBody  ||
  1877.         (pNextDepNode->m_tag == SMILSeq  &&
  1878.         pNextDepNode->m_id == "body")  ||
  1879.         pNextDepNode->m_bIsOuterWrapperTimeContainer) )
  1880.         {
  1881.     // /Fixes problem in: <body><media /><seq>...</seq></body>
  1882.     // where seq was getting dur of media as dependent (which
  1883.     // it's not because it's in a prior group):
  1884.     if (pNode->m_nGroup != pNextDepNode->m_nGroup)
  1885.     {
  1886. break;
  1887.     }
  1888.     pDepNode = pNextDepNode;
  1889.     pNextDepNode = pDepNode?
  1890.         pDepNode->m_pDependency : NULL;
  1891.         }
  1892.         HX_ASSERT(pDepNode);
  1893. #endif
  1894.         IHXValues* pValues = pDepNode->m_pValues;
  1895.         if(pValues)
  1896.         {
  1897.     pAddGroup->m_pValues = pValues;
  1898.     pAddGroup->m_pValues->AddRef();
  1899.     // XXXJH should use CCF
  1900.     pAddGroup->m_pValues = new CHXHeader();
  1901.     IHXBuffer* pBuf = NULL;
  1902.     const char* name = NULL;
  1903.     pAddGroup->m_pValues->AddRef();
  1904.     if (SUCCEEDED(pValues->GetFirstPropertyCString(name, pBuf)))
  1905.     {
  1906.         do
  1907.         {
  1908.     if (strcmp("title",name)==0 ||
  1909.         strcmp("author",name)==0 ||
  1910.         strcmp("abstract",name)==0 ||
  1911.         strcmp("copyright",name)==0)
  1912.     {
  1913.         pAddGroup->m_pValues->SetPropertyCString(name, pBuf);
  1914.     }
  1915.     HX_RELEASE(pBuf);
  1916.         }
  1917.         while(SUCCEEDED(pValues->GetNextPropertyCString(name, pBuf)));
  1918.     }
  1919.         }
  1920.         pAddGroup->m_ulDuration = pDepNode->m_pElement->m_ulDuration;
  1921.         // /Fixes outer-seq bug related to PR 52452:
  1922.         // <body><seq begin="5s" end="30s"><vid ...
  1923.         // was playing as 25s presentation instead of 30s (even
  1924.         // though video would play beyond that for 5s):
  1925.         if (pDepNode->m_pElement->m_bCurBeginIsOffsetFromSyncBase  &&
  1926.         (UINT32)-1 != pAddGroup->m_ulDuration)
  1927.         {
  1928.     pAddGroup->m_ulDuration +=
  1929.     pDepNode->m_pElement->m_ulBeginOffsetFromSyncBase;
  1930.         }
  1931. #if !defined(DONT_ALLOW_PLAYLIST_STYLE_SEQ_IF_EXPLICIT_OUTER_TIMECONTAINER)
  1932.         // /And, if a virtual wrapper seq was created in
  1933.         // expandRepeatElements(), then we want to use our parent's dur:
  1934.         if (pNode->m_pDependency->m_bIsSeqWrapperForRepeatElement)
  1935.         {
  1936.     // /Get sync ancestor, not just parent in case parent is an
  1937.     // <a>, <switch>, ...etc. that shouldn't impose timing
  1938.     // constraints on its children:
  1939.     SMILNode* pSyncAncestor =
  1940.     getSyncAncestor(pNode->m_pDependency);
  1941.     if (pSyncAncestor  &&  pSyncAncestor->m_pElement)
  1942.     {
  1943.         pAddGroup->m_ulDuration =
  1944.         pSyncAncestor->m_pElement->m_ulDuration;
  1945.     }
  1946.         }
  1947. #endif
  1948.     }
  1949.     (*m_pAddGroupMap)[pNode->m_nGroup] = pAddGroup;
  1950.     pAddGroup->m_nTotalTracks = 1;
  1951.     if(pNode->m_pElement &&
  1952.         pNode->m_pElement->m_ulDelay == 0)
  1953.     {
  1954.         pAddGroup->m_nInitTracks = 1;
  1955.     }
  1956.         }
  1957.     }
  1958.     return rc;
  1959. }
  1960. HX_RESULT
  1961. CSmilParser::createBodyElements(SMILNodeList* pNodeList)
  1962. {
  1963.     HX_RESULT rc = HXR_OK;
  1964.     if(!pNodeList)
  1965.     {
  1966. return rc;
  1967.     }
  1968.     if(!m_pNodeDependencies)
  1969.     {
  1970. m_pNodeDependencies = new CHXStack;
  1971.     }
  1972.     if(!m_pAnchorStack)
  1973.     {
  1974. m_pAnchorStack = new CHXStack;
  1975.     }
  1976.     if(!m_pTrackHintList)
  1977.     {
  1978. m_pTrackHintList = new CHXSimpleList;
  1979.     }
  1980.     UINT32 ulTrackHint = 1;
  1981.     CHXSimpleList::Iterator i;
  1982.     for(i=pNodeList->Begin();i!=pNodeList->End();++i)
  1983.     {
  1984. if(HXR_OK != rc)
  1985. {
  1986.     return rc;
  1987. }
  1988. SMILNode* pNode = (SMILNode*)(*i);
  1989. if(pNode->m_bDelete) // skip this puppy
  1990. {
  1991.     continue;
  1992. }
  1993. rc = addToNamespaceScope(pNode);
  1994. if (FAILED(rc))
  1995. {
  1996.     return rc;
  1997. }
  1998. switch(pNode->m_tag)
  1999. {
  2000.     case SMILSeq:
  2001.     {
  2002. CSmilSeqElement* pElement = makeSeqElement(pNode);
  2003. if(!pElement)
  2004. {
  2005.     return HXR_FAIL;
  2006. }
  2007. pNode->m_pElement = pElement;
  2008. CSmilTimelineSeq* pTimelineSeq =
  2009.     new CSmilTimelineSeq(pElement, this);
  2010. pElement->m_pTimelineElement = pTimelineSeq;
  2011.                 // Check if this element depends on an external media
  2012.                 // marker file. If so, then add it to a list to be
  2013.                 // processed later
  2014.                 checkForXMMFDependency(pElement);
  2015. if(!m_pCurrentDependentNode) // at the top
  2016. {
  2017.     //pNode->m_nGroup = (UINT16)-1;   // top level
  2018. }
  2019. else
  2020. {
  2021.     pNode->m_pDependency = m_pCurrentDependentNode;
  2022.     if(!hasAncestor(SMILPar, pNode)  &&
  2023.     !hasAncestor(SMILExcl, pNode))
  2024.     {
  2025. //pNode->m_nGroup =
  2026. //    m_pCurrentDependentNode->m_nGroup + 1;
  2027.     }
  2028.     else
  2029.     {
  2030. //pNode->m_nGroup =
  2031. //    m_pCurrentDependentNode->m_nGroup;
  2032.     }
  2033. }
  2034. m_pCurrentDependentNode = pNode;
  2035. m_pNodeDependencies->Push(m_pCurrentDependentNode);
  2036. ulTrackHint = 1;
  2037. m_pTrackHintList->AddTail((void*)ulTrackHint);
  2038.     }
  2039.     break;
  2040.     case SMILPar:
  2041.     {
  2042. CSmilParElement* pElement = makeParElement(pNode);
  2043. if(!pElement)
  2044. {
  2045.     return HXR_FAIL;
  2046. }
  2047. pNode->m_pElement = pElement;
  2048. CSmilTimelinePar* pTimelinePar =
  2049.     new CSmilTimelinePar(pElement, this);
  2050. pElement->m_pTimelineElement = pTimelinePar;
  2051.                 // Check if this element depends on an external media
  2052.                 // marker file. If so, then add it to a list to be
  2053.                 // processed later
  2054.                 checkForXMMFDependency(pElement);
  2055. if(!m_pCurrentDependentNode) // at the top
  2056. {
  2057.     // pNode->m_nGroup = 0;
  2058. }
  2059. else
  2060. {
  2061.     pNode->m_pDependency = m_pCurrentDependentNode;
  2062.     if(!hasAncestor(SMILPar, pNode)  &&
  2063.     !hasAncestor(SMILExcl, pNode))
  2064.     {
  2065. //pNode->m_nGroup =
  2066. //    m_pCurrentDependentNode->m_nGroup + 1;
  2067.     }
  2068.     else
  2069.     {
  2070. //pNode->m_nGroup = m_pCurrentDependentNode->m_nGroup;
  2071.     }
  2072. }
  2073. if(firstDependentChild(pNode) || m_pCurrentDependentNode == 0)
  2074. {
  2075.     ulTrackHint = 1;
  2076.     m_pTrackHintList->AddTail((void*)ulTrackHint);
  2077. }
  2078. else
  2079. {
  2080.     if(m_pTrackHintList->GetCount() > 0)
  2081.     {
  2082. ulTrackHint = (UINT32)m_pTrackHintList->RemoveTail();
  2083.     }
  2084.     m_pTrackHintList->AddTail((void*)++ulTrackHint);
  2085. }
  2086. m_pCurrentDependentNode = pNode;
  2087. m_pNodeDependencies->Push(m_pCurrentDependentNode);
  2088.     }
  2089.     break;
  2090.     case SMILExcl:
  2091.     {
  2092. CSmilExclElement* pElement = makeExclElement(pNode);
  2093. if(!pElement)
  2094. {
  2095.     return HXR_FAIL;
  2096. }
  2097. pNode->m_pElement = pElement;
  2098. CSmilTimelineExcl* pTimelineExcl =
  2099.     new CSmilTimelineExcl(pElement, this);
  2100. pElement->m_pTimelineElement = pTimelineExcl;
  2101.                 // Check if this element depends on an external media
  2102.                 // marker file. If so, then add it to a list to be
  2103.                 // processed later
  2104.                 checkForXMMFDependency(pElement);
  2105. if(!m_pCurrentDependentNode) // at the top
  2106. {
  2107.     // pNode->m_nGroup = 0;
  2108. }
  2109. else
  2110. {
  2111.     pNode->m_pDependency = m_pCurrentDependentNode;
  2112.     if(!hasAncestor(SMILPar, pNode)  &&
  2113.     !hasAncestor(SMILExcl, pNode))
  2114.     {
  2115. //pNode->m_nGroup =
  2116. //    m_pCurrentDependentNode->m_nGroup + 1;
  2117.     }
  2118.     else
  2119.     {
  2120. //pNode->m_nGroup = m_pCurrentDependentNode->m_nGroup;
  2121.     }
  2122. }
  2123. // /XXXEH- should we do this for SMILExcl (as for SMILPar)?:
  2124. if(firstDependentChild(pNode) || m_pCurrentDependentNode == 0)
  2125. {
  2126.     ulTrackHint = 1;
  2127.     m_pTrackHintList->AddTail((void*)ulTrackHint);
  2128. }
  2129. else
  2130. {
  2131.     if(m_pTrackHintList->GetCount() > 0)
  2132.     {
  2133. ulTrackHint = (UINT32)m_pTrackHintList->RemoveTail();
  2134.     }
  2135.     m_pTrackHintList->AddTail((void*)++ulTrackHint);
  2136. }
  2137. m_pCurrentDependentNode = pNode;
  2138. m_pNodeDependencies->Push(m_pCurrentDependentNode);
  2139.     }
  2140.     break;
  2141.     case SMILPriorityClass:
  2142.     {
  2143. // /Make sure parent is SMILExcl; a priorityClass element only
  2144. // is allowed an excl for a parent:
  2145. HX_ASSERT(pNode->m_pParent);
  2146. if (pNode->m_pParent)
  2147. {
  2148.     if (SMILExcl !=pNode->m_pParent->m_tag)
  2149.     {
  2150. #if defined(_DEBUG)
  2151. // /XXXEH- need to add informative error message for this:
  2152. BOOL NEED_ERROR__priorityClass_must_be_child_of_excl = 0;
  2153. HX_ASSERT(NEED_ERROR__priorityClass_must_be_child_of_excl);
  2154. #endif
  2155. CSmilSMILSyntaxErrorHandler errHandler(m_pContext);
  2156. errHandler.ReportError(SMILErrorUnexpectedTag,
  2157.        (const char*) pNode->m_name,
  2158.        pNode->m_ulTagStartLine);
  2159. rc = HXR_FAIL;
  2160. // /XXXEH- TODO: handle out-of-spec error condition.
  2161. break;
  2162.     }
  2163.     // /Next, make sure all of parent's other children, if any,
  2164.     // are priorityClass elements as well:
  2165.     CHXSimpleList* pNodeList = pNode->m_pParent->m_pNodeList;
  2166.     HX_ASSERT(pNodeList);
  2167.     if (pNodeList)
  2168.     {
  2169. CHXSimpleList::Iterator i;
  2170. for (i=pNodeList->Begin();i!=pNodeList->End();++i)
  2171. {
  2172.     SMILNode* pChildNode = (SMILNode*)(*i);
  2173.     if (pChildNode->m_bDelete)
  2174.     {
  2175. continue;
  2176.     }
  2177.     // /If an excl element has a priorityClass child,
  2178.     // *ALL* children must be priorityClass elements:
  2179.     if(SMILPriorityClass != pChildNode->m_tag  &&
  2180.     SMILEndExcl != pChildNode->m_tag)
  2181.     {
  2182. CHXString errMsg =
  2183. (const char*)pChildNode->m_name;
  2184. const char* pTmp = " with ";
  2185. errMsg += pTmp;
  2186. errMsg += pNode->m_name;
  2187. CSmilSMILSyntaxErrorHandler
  2188. errHandler(m_pContext);
  2189. errHandler.ReportError(
  2190. SMILErrorUnexpectedTag,
  2191. (const char*)errMsg,
  2192. pNode->m_ulTagStartLine);
  2193. rc = HXR_FAIL;
  2194. break;
  2195.     }
  2196. }
  2197.     }
  2198. }
  2199. CSmilPriorityClassElement* pElement =
  2200. makePriorityClassElement(pNode);
  2201. if(pElement)
  2202. {
  2203.     pNode->m_pElement = pElement;
  2204. }
  2205. else
  2206. {
  2207.     rc = HXR_INVALID_PARAMETER;
  2208. }
  2209.     }
  2210.     break;
  2211.     case SMILEndPriorityClass:
  2212.     break;
  2213.     case SMILEndSeq:
  2214.     {
  2215. m_pCurrentDependentNode =
  2216.     (SMILNode*)m_pNodeDependencies->Pop();
  2217. if(m_pTrackHintList->GetCount() > 0)
  2218. {
  2219.     m_pTrackHintList->RemoveTail();
  2220. }
  2221.     }
  2222.     break;
  2223.     case SMILEndExcl:
  2224.     case SMILEndPar:
  2225.     {
  2226. m_pCurrentDependentNode =
  2227.     (SMILNode*)m_pNodeDependencies->Pop();
  2228. if(m_pTrackHintList->GetCount() > 0)
  2229. {
  2230.     m_pTrackHintList->RemoveTail();
  2231. }
  2232.     }
  2233.     break;
  2234.     case SMILAAnchor:
  2235.     {
  2236. CSmilAAnchorElement* pElement = makeAAnchorElement(pNode);
  2237. if(pElement)
  2238. {
  2239.     pNode->m_pElement = pElement;
  2240.     m_pAnchorStack->Push(m_pCurrentAnchor);
  2241.     m_pCurrentAnchor = pElement;
  2242. }
  2243. else
  2244. {
  2245.     rc = HXR_INVALID_PARAMETER;
  2246. }
  2247.     }
  2248.     break;
  2249.     case SMILEndAAnchor:
  2250.     {
  2251. m_pCurrentAnchor = (CSmilAAnchorElement*)m_pAnchorStack->Pop();
  2252.     }
  2253.     break;
  2254.     case SMILArea:
  2255.     case SMILAnchor:
  2256.     {
  2257. CSmilAnchorElement* pElement = makeAnchorElement(pNode);
  2258. if(pElement)
  2259. {
  2260.     pNode->m_pElement = pElement;
  2261.     //[SMIL 1.0 compliance] Helps fix PR 26471:
  2262.     // In order to get a notification that we have a resolved
  2263.     // event-based begin time, we need to be a
  2264.     // timeline element:
  2265.     CSmilTimelineAnchor* pTimelineAnchor =
  2266. new CSmilTimelineAnchor(pElement, this);
  2267.     pElement->m_pTimelineElement = pTimelineAnchor;
  2268.                     // Check if this element depends on an external media
  2269.                     // marker file. If so, then add it to a list to be
  2270.                     // processed later
  2271.                     checkForXMMFDependency(pElement);
  2272.     // add link to parent
  2273.     SMILNode* pParent = pNode->m_pParent;
  2274.     if(pParent->m_pElement)
  2275.     {
  2276. pParent->m_pElement->m_pHyperlinks->AddTail(pElement);
  2277.     }
  2278.     if(!m_pCurrentDependentNode)
  2279.     {
  2280. //anchor should NEVER be top element:
  2281. HX_ASSERT(m_pCurrentDependentNode);
  2282.     }
  2283.     else
  2284.     {
  2285. pNode->m_pDependency = m_pCurrentDependentNode;
  2286.     }
  2287.     if(firstDependentChild(pNode) || m_pCurrentDependentNode == 0)
  2288.     {
  2289. ulTrackHint = 1;
  2290. m_pTrackHintList->AddTail((void*)ulTrackHint);
  2291.     }
  2292.     else
  2293.     {
  2294. if(m_pTrackHintList->GetCount() > 0)
  2295. {
  2296.     ulTrackHint = (UINT32)m_pTrackHintList->RemoveTail();
  2297. }
  2298. m_pTrackHintList->AddTail((void*)++ulTrackHint);
  2299.     }
  2300.     m_pCurrentDependentNode = pNode;
  2301.     m_pNodeDependencies->Push(m_pCurrentDependentNode);
  2302.     //end of part of fix for PR 26471.
  2303. }
  2304.     }
  2305.     break;
  2306.     case SMILBrush:
  2307. #if !defined(HELIX_FEATURE_SMIL2_BRUSH)
  2308.                 pNode->m_bDelete = TRUE;
  2309.                 break;
  2310. #endif
  2311.     case SMILRef:
  2312.     case SMILText:
  2313.     case SMILImg:
  2314.     case SMILAudio:
  2315.     case SMILVideo:
  2316.     case SMILAnimation:
  2317.     case SMILTextstream:
  2318.     case SMILPrefetch:
  2319.     {
  2320. HX_ASSERT(isMediaObject(pNode));
  2321. CSmilSource* pSource = makeSource(pNode);
  2322. if(!pSource)
  2323. {
  2324.     return HXR_FAIL;
  2325. }
  2326. m_bContainsSource = TRUE;
  2327. pNode->m_pElement = pSource;
  2328. CSmilTimelineElement* pTimelineElement =
  2329.     new CSmilTimelineElement(pSource, this);
  2330. pNode->m_pElement->m_pTimelineElement = pTimelineElement;
  2331.                 // Check if this element depends on an external media
  2332.                 // marker file. If so, then add it to a list to be
  2333.                 // processed later
  2334.                 checkForXMMFDependency(pSource);
  2335. // attach any links
  2336. if(m_pCurrentAnchor)
  2337. {
  2338.     pSource->m_pHyperlinks->AddTail(m_pCurrentAnchor);
  2339. }
  2340. if(!m_pCurrentDependentNode)
  2341. {
  2342.     // make it behave like it's in a SEQ
  2343.     pNode->m_pDependency = 0;
  2344.     //pNode->m_nGroup = 0;
  2345.     m_pCurrentDependentNode = pNode;
  2346.     ulTrackHint = 1;
  2347.     m_pTrackHintList->AddTail((void*)ulTrackHint);
  2348. }
  2349. else
  2350. {
  2351.     pNode->m_pDependency = m_pCurrentDependentNode;
  2352.     if(inSeq(pNode))
  2353.     {
  2354. if(firstDependentChild(pNode))
  2355. {
  2356.     ulTrackHint = 1;
  2357.     m_pTrackHintList->AddTail((void*)ulTrackHint);
  2358. }
  2359. else
  2360. {
  2361.     if(m_pTrackHintList->GetCount() > 0)
  2362.     {
  2363. ulTrackHint =
  2364.     (UINT32)m_pTrackHintList->RemoveTail();
  2365.     }
  2366.     m_pTrackHintList->AddTail((void*)++ulTrackHint);
  2367. }
  2368.     }
  2369.     else
  2370.     {
  2371. if(firstDependentChild(pNode))
  2372. {
  2373.     ulTrackHint = 1;
  2374.     m_pTrackHintList->AddTail((void*)ulTrackHint);
  2375. }
  2376.     }
  2377.     if(inSeq(pNode)  &&  (!hasAncestor(SMILPar, pNode)  &&
  2378.     !hasAncestor(SMILExcl, pNode)) )
  2379.     {
  2380. if(firstDependentChild(pNode))
  2381. {
  2382.     //pNode->m_nGroup = m_pCurrentDependentNode->m_nGroup;
  2383. }
  2384. else
  2385. {
  2386.     //pNode->m_nGroup =
  2387. //m_pCurrentDependentNode->m_nGroup + 1;
  2388. }
  2389. m_pCurrentDependentNode = pNode;
  2390.     }
  2391.     else if(inSeq(pNode))
  2392.     {
  2393. //pNode->m_nGroup = m_pCurrentDependentNode->m_nGroup;
  2394. m_pCurrentDependentNode = pNode;
  2395.     }
  2396.     else // in PAR or EXCL:
  2397.     {
  2398. // pNode->m_nGroup = m_pCurrentDependentNode->m_nGroup;
  2399.     }
  2400. }
  2401. // /Helps fix PR 58294: using this variable fixes
  2402. // <par end="30s"> version, but doesn't completely fix
  2403. // <par endsync="all"> version:
  2404. if (pNode->m_pElement->m_bHasAtLeastOneNonEventBasedBegin)
  2405. {
  2406.     m_bContainsInitiallyScheduledTrack = TRUE;
  2407. }
  2408. // set track hint info
  2409. BOOL bFirstHint = TRUE;
  2410. CHXSimpleList::Iterator hintIter = m_pTrackHintList->Begin();
  2411. for(; hintIter != m_pTrackHintList->End(); ++hintIter)
  2412. {
  2413.     char tmpBuf[20]; /* Flawfinder: ignore */
  2414.     UINT32 ulTrackHint = (UINT32)(*hintIter);
  2415.     if(bFirstHint)
  2416.     {
  2417. sprintf(tmpBuf, "%d", ulTrackHint); /* Flawfinder: ignore */
  2418. bFirstHint = FALSE;
  2419.     }
  2420.     else
  2421.     {
  2422. sprintf(tmpBuf, ".%d", ulTrackHint); /* Flawfinder: ignore */
  2423.     }
  2424.     pNode->m_trackHint += tmpBuf;
  2425. }
  2426.     }
  2427.     break;
  2428.     case SMILAnimate:
  2429.     case SMILSet:
  2430.     case SMILAnimateMotion:
  2431.     case SMILAnimateColor:
  2432.     {
  2433. #if defined(HELIX_FEATURE_SMIL2_ANIMATION)
  2434.                 // Create the animate element object
  2435.                 CSmilAnimateElement* pElement = makeAnimateElement(pNode);
  2436.                 if(pElement)
  2437.                 {
  2438.                     // Add this to the animate element list
  2439.                     if (!m_pAnimateElementList)
  2440.                     {
  2441.                         m_pAnimateElementList = new CHXSimpleList();
  2442.                     }
  2443.                     if (m_pAnimateElementList)
  2444.                     {
  2445.                         m_pAnimateElementList->AddTail((void*) pElement);
  2446.                     }
  2447.                     // Link the animate element object to the node
  2448.                     pNode->m_pElement = pElement;
  2449.                     // Create an animate timeline element - we
  2450.                     // need this to get notification of resolved times
  2451.                     CSmilTimelineAnimate* pTimelineElement =
  2452.                         new CSmilTimelineAnimate(pElement, this);
  2453.                     if (pTimelineElement)
  2454.                     {
  2455.                         // Link the timeline element to the element
  2456.                         pElement->m_pTimelineElement = pTimelineElement;
  2457.                         // Check if this element depends on an external media
  2458.                         // marker file. If so, then add it to a list to be
  2459.                         // processed later
  2460.                         checkForXMMFDependency(pElement);
  2461.         if(!m_pCurrentDependentNode)
  2462.         {
  2463.             // make it behave like it's in a SEQ
  2464.             pNode->m_pDependency = 0;
  2465.             m_pCurrentDependentNode = pNode;
  2466.         }
  2467.         else
  2468.         {
  2469.             pNode->m_pDependency = m_pCurrentDependentNode;
  2470.             if(inSeq(pNode)  &&  (!hasAncestor(SMILPar, pNode)  &&
  2471.             !hasAncestor(SMILExcl, pNode)) )
  2472.             {
  2473.         m_pCurrentDependentNode = pNode;
  2474.             }
  2475.             else if(inSeq(pNode))
  2476.             {
  2477.         m_pCurrentDependentNode = pNode;
  2478.             }
  2479.         }
  2480.                     }
  2481.                     else
  2482.                     {
  2483.                         rc = HXR_OUTOFMEMORY;
  2484.                     }
  2485.                 }
  2486.                 else
  2487.                 {
  2488.                     // XXXMEH - we would get here if the animation
  2489.                     // failed parsing for some reason. We won't throw
  2490.                     // an error (although I think we should) - we will
  2491.                     // just not do the animation.
  2492. //                    rc = HXR_OUTOFMEMORY;
  2493.                     // XXXMEH - what we *will* do is mark the node
  2494.                     // for deletion.
  2495.                     pNode->m_bDelete = TRUE;
  2496.                 }
  2497. #else
  2498.                 // Animation is not supported so delete element
  2499.                 pNode->m_bDelete = TRUE;
  2500. #endif /* #if defined(HELIX_FEATURE_SMIL2_ANIMATION) */
  2501.     }
  2502.     break;
  2503.             case SMILParam:
  2504.             {
  2505.                 // Create the animate element object
  2506.                 CSmilParamElement* pElement = makeParamElement(pNode);
  2507.                 if(pElement)
  2508.                 {
  2509.                     // Link the animate element object to the node
  2510.                     pNode->m_pElement = pElement;
  2511.                 }
  2512.             }
  2513.             break;
  2514.     default:
  2515.     break;
  2516. }
  2517.         if (SUCCEEDED(rc))
  2518.         {
  2519.     // /This fixes case where there is nothing to play yet there
  2520.     // is a duration on the time container; we want to play "nothing"
  2521.     // for that duration:
  2522.     if (isTimeContainerObject(pNode)  &&
  2523.     // /Note: m_bContainsSource only tells if 1 or more sources
  2524.     // *might* play, but this var says that 1+ *will* play:
  2525.     !m_bContainsInitiallyScheduledTrack  &&
  2526.     pNode->m_pElement  &&
  2527.     DEFAULT_DUR_IF_NO_SOURCES_SCHEDULED ==
  2528.     m_ulDurIfNoInitialTracksScheduled)
  2529.     {
  2530. if ((UINT32)-1 != pNode->m_pElement->m_ulDuration)
  2531. {
  2532.     m_ulDurIfNoInitialTracksScheduled =
  2533.     pNode->m_pElement->m_ulDuration;
  2534. }
  2535. // /Helps fix PR 58294 (original version w/endsync="all"):
  2536. // if endsync="all" and children exist, then treat duration as
  2537. // unresolved in case all children are event-begun (but
  2538. // children haven't been created yet):
  2539. else if (SMILEventSourceAll ==
  2540. pNode->m_pElement->m_nEndsyncEventSourceTag  &&
  2541. pNode->m_pNodeList->GetCount() > 0  &&
  2542. pNode->m_pElement->m_pTimelineElement  &&
  2543. !pNode->m_pElement->m_pTimelineElement->getNumChildren())
  2544. {
  2545.     m_ulDurIfNoInitialTracksScheduled = WAY_IN_THE_FUTURE;
  2546. }
  2547. // /ELSE DO NOTHING!
  2548.     }
  2549.             rc = createBodyElements(pNode->m_pNodeList);
  2550.     if (SUCCEEDED(rc))
  2551.             {
  2552.                 rc = removeFromNamespaceScope(pNode);
  2553.             }
  2554.         }
  2555.     }
  2556.     return rc;
  2557. }
  2558. HX_RESULT
  2559. CSmilParser::expandRepeatElements(SMILNodeList* pNodeList)
  2560. {
  2561.     // walk through tree and expand any "repeat" attributes
  2562.     HX_RESULT rc = HXR_OK;
  2563.     if(!pNodeList)
  2564.     {
  2565. return rc;
  2566.     }
  2567.     LISTPOSITION lPos = pNodeList->GetHeadPosition();
  2568.     while (lPos && HXR_OK == rc)
  2569.     {
  2570. SMILNode* pNode = (SMILNode*)pNodeList->GetAt(lPos);
  2571. if(pNode->m_bDelete) // skip this puppy
  2572. {
  2573.     pNodeList->GetNext(lPos);
  2574.     continue;
  2575. }
  2576.         // XXXMEH - don't handle repeats in <animate>, <set>,
  2577.         // <animateMotion>, and <animateColor> here - we
  2578.         // will handle them in the animation calculation
  2579.         // code.
  2580.         if (pNode->m_tag == SMILAnimate       ||
  2581.             pNode->m_tag == SMILSet           ||
  2582.             pNode->m_tag == SMILAnimateMotion ||
  2583.             pNode->m_tag == SMILAnimateColor)
  2584.         {
  2585.             pNodeList->GetNext(lPos);
  2586.             continue;
  2587.         }
  2588. if(pNode->m_pValues)
  2589. {
  2590.     // /Handle case where there are multiple repeatX attributes:
  2591.     IHXBuffer* pBufSMIL1Repeat = NULL;
  2592.     IHXBuffer* pBufRepeatCount = NULL;
  2593.     IHXBuffer* pBufRepeatDur = NULL;
  2594.     BOOL bHasSMIL1Repeat = (HXR_OK ==
  2595.     pNode->m_pValues->GetPropertyCString("repeat",
  2596.     pBufSMIL1Repeat));
  2597.     BOOL bHasRepeatCount = (HXR_OK ==
  2598.     pNode->m_pValues->GetPropertyCString("repeatCount",
  2599.     pBufRepeatCount));
  2600.     BOOL bHasRepeatDur = (HXR_OK ==
  2601.     pNode->m_pValues->GetPropertyCString("repeatDur",
  2602.     pBufRepeatDur));
  2603.     // /If both repeat and repeatDur are specified, use repeatDur:
  2604.     if (bHasSMIL1Repeat  &&  bHasRepeatCount)
  2605.     {
  2606. bHasSMIL1Repeat = FALSE;
  2607.     }
  2608.     // /XXXEH- TODO: figure out how to handle this:
  2609.     //    repeat="..." repeatDur="..."
  2610.     // Just use repeatDur and ignore repeat??  That's what I'm doing:
  2611.     if (bHasSMIL1Repeat  &&  bHasRepeatDur)
  2612.     {
  2613. bHasSMIL1Repeat = FALSE;
  2614.     }
  2615.     if (!pNode->m_bRepeatHandled  &&
  2616.     (bHasSMIL1Repeat  ||  bHasRepeatCount  ||  bHasRepeatDur) )
  2617.     {
  2618. if (bHasRepeatDur  ||  bHasRepeatCount)
  2619. {
  2620.     // /SMIL 1.0-VS-2.0 repeat conflict should have been
  2621.     // solved already above so assert if not:
  2622.     HX_ASSERT(!bHasSMIL1Repeat);
  2623. }
  2624. BOOL bSMIL1RepeatIndefinite = FALSE;
  2625. BOOL bRepeatCountIndefinite = FALSE;
  2626. BOOL bRepeatDurIndefinite = FALSE;
  2627. const char* pSMIL1RepeatVal = NULL;
  2628. const char* pRepeatCountVal = NULL;
  2629. const char* pRepeatDurVal = NULL;
  2630. if (bHasSMIL1Repeat)
  2631. {
  2632.     pSMIL1RepeatVal = (const char*)pBufSMIL1Repeat->GetBuffer();
  2633.     if(strcmp(pSMIL1RepeatVal, "indefinite") == 0)
  2634.     {
  2635. bSMIL1RepeatIndefinite = TRUE;
  2636.     }
  2637. }
  2638. if (bHasRepeatCount)
  2639. {
  2640.     HX_ASSERT(!bHasSMIL1Repeat);
  2641.     pRepeatCountVal =
  2642.     (const char*)pBufRepeatCount->GetBuffer();
  2643.     if(strcmp(pRepeatCountVal, "indefinite") == 0)
  2644.     {
  2645. bRepeatCountIndefinite = TRUE;
  2646.     }
  2647. }
  2648. if (bHasRepeatDur)
  2649. {
  2650.     HX_ASSERT(!bHasSMIL1Repeat);
  2651.     pRepeatDurVal = (const char*)pBufRepeatDur->GetBuffer();
  2652.     if(strcmp(pRepeatDurVal, "indefinite") == 0)
  2653.     {
  2654. bRepeatDurIndefinite = TRUE;
  2655.     }
  2656. }
  2657. if (bHasRepeatDur  &&  bHasRepeatCount)
  2658. {
  2659.     // /Find "MIN(p1,p2,indefinite)" per the SMIL 2.0 Timing
  2660.     // spec; If exactly one is indefinite, throw it away.  If
  2661.     // both are, then just use repeatCount of indefinite:
  2662.     if (bRepeatCountIndefinite  &&  !bRepeatDurIndefinite)
  2663.     {
  2664. bHasRepeatCount = FALSE;
  2665.     }
  2666.     else if (bRepeatDurIndefinite  &&  !bRepeatCountIndefinite)
  2667.     {
  2668. bHasRepeatDur = FALSE;
  2669.     }
  2670.     else if (bRepeatDurIndefinite  &&  bRepeatCountIndefinite)
  2671.     {
  2672. bHasRepeatDur = FALSE;
  2673.     }
  2674.     // /We need to know which will play for less time, but we
  2675.     // can't know how long the total repeatCount duration
  2676.     // will be until we resolve the dur of the element, so we
  2677.     // must calc this when trackDurationSet() gets called:
  2678.     else // /neither is indefinite:
  2679.     {
  2680. pNode->m_bNeedToResolveRepeatDurVSRepeatCount = TRUE;
  2681. // /Just pick one for now (by removing the other one):
  2682. bHasRepeatDur = FALSE;
  2683.     }
  2684. }
  2685. // /Only 1 of these three should be TRUE at this point:
  2686. HX_ASSERT(1 == (UINT32)bHasRepeatCount +
  2687. (UINT32)bHasRepeatDur + (UINT32)bHasSMIL1Repeat);
  2688. // /Now, settle on which one won:
  2689. const char* pRepeatVal = NULL;
  2690. BOOL bRepeatIndefinite = FALSE;
  2691. if (bHasRepeatCount)
  2692. {
  2693.     pRepeatVal = pRepeatCountVal;
  2694.     bRepeatIndefinite = bRepeatCountIndefinite;
  2695. }
  2696. else if (bHasRepeatDur)
  2697. {
  2698.     pRepeatVal = pRepeatDurVal;
  2699.     bRepeatIndefinite = bRepeatDurIndefinite;
  2700. }
  2701. else
  2702. {
  2703.     pRepeatVal = pSMIL1RepeatVal;
  2704.     bRepeatIndefinite = bSMIL1RepeatIndefinite;
  2705. }
  2706. if (pRepeatVal)
  2707. {
  2708.     INT32 lRepeatCount = 0;
  2709.     double fRepeatCount = 0.0;
  2710.     ULONG32 ulRepeatDur = 0;
  2711.     if(bRepeatIndefinite)
  2712.     {
  2713. bRepeatIndefinite = TRUE;
  2714. lRepeatCount = 2;
  2715. fRepeatCount = 2.0;
  2716.     }
  2717.     else
  2718.     {
  2719. if (bHasSMIL1Repeat)
  2720. {
  2721.     lRepeatCount = atol(pRepeatVal);
  2722.     fRepeatCount = double(float(lRepeatCount));
  2723. }
  2724. else if (bHasRepeatCount)
  2725. {
  2726.     lRepeatCount = atol(pRepeatVal);
  2727.     fRepeatCount = atof(pRepeatVal);
  2728. }
  2729. else // / bHasRepeatDur so handle dur:
  2730. {
  2731.     HX_RESULT pnrslt = parseClockValue(pRepeatVal,
  2732.     ulRepeatDur);
  2733.     if (HXR_OK == pnrslt)
  2734.     {
  2735. // /We'll wrap it in a seq with a dur of
  2736. // ulRepeatDur and then repeat it underneath
  2737. // indefinitely, ending when the seq does:
  2738. bRepeatIndefinite = TRUE;
  2739. lRepeatCount = 2;
  2740. fRepeatCount = 2.0;
  2741.     }
  2742.     else
  2743.     {
  2744. // /Forget repeating since time was invalid:
  2745. bHasRepeatDur = FALSE;
  2746.     }
  2747. }
  2748.     }
  2749.     double fPartialPlayFactor =
  2750.     fRepeatCount - float(double(lRepeatCount));
  2751.     if(fRepeatCount == 0.0)
  2752.     {
  2753. pNode->m_bDelete = TRUE;
  2754.     }
  2755.     else if (fRepeatCount < 1.0)
  2756.     {
  2757. HX_ASSERT(fPartialPlayFactor > 0.0);
  2758. pNode->m_fPartialPlayFactor = fPartialPlayFactor;
  2759.     }
  2760.     else if (fRepeatCount > 1.0)
  2761.     {
  2762. // build a <seq>/</seq> around the repeated element...
  2763. SMILNode* pSeqNode = NULL;
  2764. SMILNode* pSeqEndNode = NULL;
  2765. createParent(pNode, SMILSeq, pSeqNode, pSeqEndNode);
  2766. // /Set this to TRUE so we can later decide whether
  2767. // or not to use explicit dur|end on parent as dur
  2768. // of this:
  2769. pSeqNode->m_bIsSeqWrapperForRepeatElement = TRUE;
  2770. // /For element that has any attributes that affect
  2771. // its "active duration" (see SMIL 2.0 spec) and not
  2772. // just its "simple duration", we put these into the
  2773. // wrapper seq's values giving us the desired result.
  2774. // These attributes are: begin, end, min, and max.
  2775. // (We also will "remove" these values from pNode by
  2776. // setting pNodes appropriate "HandledBy" flags):
  2777. IHXBuffer* pBufTemp = NULL;
  2778. BOOL bHasEndDurOrMinTimingAttribute = FALSE;
  2779. IHXValues* pValues = 0;
  2780. if (m_pClassFactory  &&
  2781. (HXR_OK == m_pClassFactory->CreateInstance(
  2782.  CLSID_IHXValues, (void**)&pValues)) )
  2783. {
  2784.     pSeqNode->m_pValues = pValues;
  2785. }
  2786. if (HXR_OK == pNode->m_pValues->
  2787. GetPropertyCString("end", pBufTemp) )
  2788. {
  2789.     bHasEndDurOrMinTimingAttribute = TRUE;
  2790.     if (pSeqNode->m_pValues)
  2791.     {
  2792. pSeqNode->m_pValues->SetPropertyCString(
  2793. "end", pBufTemp);
  2794. pNode->m_bEndHandledByWrapperParent =TRUE;
  2795.     }
  2796.     HX_RELEASE(pBufTemp);
  2797. }
  2798. if (HXR_OK == pNode->m_pValues->
  2799. GetPropertyCString("min", pBufTemp) )
  2800. {
  2801.     bHasEndDurOrMinTimingAttribute = TRUE;
  2802.     if (pSeqNode->m_pValues)
  2803.     {
  2804. // /XXXEH- TODO: handle min="media"
  2805. pSeqNode->m_pValues->SetPropertyCString(
  2806. "min", pBufTemp);
  2807. pNode->m_bMinHandledByWrapperParent =TRUE;
  2808.     }
  2809.     HX_RELEASE(pBufTemp);
  2810. }
  2811. if (HXR_OK == pNode->m_pValues->
  2812. GetPropertyCString("max", pBufTemp) )
  2813. {
  2814.     if (pSeqNode->m_pValues)
  2815.     {
  2816. // /XXXEH- TODO: handle max="media"
  2817. pSeqNode->m_pValues->SetPropertyCString(
  2818. "max", pBufTemp);
  2819. pNode->m_bMaxHandledByWrapperParent =TRUE;
  2820.     }
  2821.     HX_RELEASE(pBufTemp);
  2822. }
  2823. #if defined(XXXEH_NEED_TO_HANDLE_NON_CLOCKVALUE_BEGINS)
  2824. // /XXXEH- TODO: handle begin offset in wrapping seq, but we'll need to then
  2825. // properly handle multiple begin conditions on an element as well as things
  2826. // like begin="activateEvent" (which should work whenever ANY iteration of
  2827. // the repeating media gets clicked):
  2828. if (HXR_OK == pNode->m_pValues->
  2829. GetPropertyCString("begin", pBufTemp) )
  2830. {
  2831.     if (pSeqNode->m_pValues)
  2832.     {
  2833. pSeqNode->m_pValues->SetPropertyCString(
  2834. "begin", pBufTemp);
  2835. pNode->m_bBeginHandledByWrapperParent =TRUE;
  2836.     }
  2837.     HX_RELEASE(pBufTemp);
  2838. }
  2839. #endif
  2840. // /For repeatDur="t", give the outer seq a dur of t:
  2841. if (bHasRepeatDur)
  2842. {
  2843.     // /NOTE: in case of discrete media this should
  2844.     // have a dur of 0, but we don't know if it's
  2845.     // discrete media until
  2846.     // CSmilDocumentRenderer::TrackDurationSet()
  2847.     // gets called; we have to guess if it is, for
  2848.     // now:
  2849.     if (!bHasEndDurOrMinTimingAttribute)
  2850.     {
  2851. bHasEndDurOrMinTimingAttribute =
  2852. (HXR_OK == pNode->m_pValues->
  2853. GetPropertyCString("dur", pBufTemp));
  2854. HX_RELEASE(pBufTemp);
  2855.     }
  2856.     // /XXXEH- TODO: find a better way to get around
  2857.     // this discrete-media problem: at this point, we
  2858.     // need the media's header to tell us whether it
  2859.     // is discrete media, but we have to decide *now*
  2860.     // whether or not to honor its repeatDur...
  2861.     BOOL bIsImageOrPlainTextOrHTML = FALSE;
  2862.     if (!bHasEndDurOrMinTimingAttribute)
  2863.     {
  2864. if (HXR_OK == pNode->m_pValues->
  2865.     GetPropertyCString("src",
  2866.     pBufTemp)  &&
  2867.     pBufTemp->GetSize()>0 )
  2868. {
  2869.     const char* pszBuf =
  2870.     (const char*)pBufTemp->GetBuffer();
  2871.     const char* pszExten = NULL;
  2872.     UINT32 ulBufLen = strlen(pszBuf);
  2873.     const char* pszWalkBuf = (ulBufLen > 1 ?
  2874.     &pszBuf[ulBufLen-2] : NULL);
  2875.     while (pszWalkBuf  &&  pszWalkBuf>pszBuf)
  2876.     {
  2877. if ('.' == *pszWalkBuf)
  2878. {
  2879.     pszExten = pszWalkBuf+1;
  2880.     break;
  2881. }
  2882. pszWalkBuf--;
  2883.     }
  2884.     if (pszExten)
  2885.     {
  2886. if (0==stricmp("jpg",pszExten)  ||
  2887.     0==stricmp("png",pszExten)  ||
  2888.     0==stricmp("gif",pszExten)  ||
  2889.     0==stricmp("bmp",pszExten)  ||
  2890.     0==stricmp("html",pszExten)  ||
  2891.     0==stricmp("txt",pszExten) )
  2892.     // /XXX ...etc.
  2893. {
  2894.     bIsImageOrPlainTextOrHTML = TRUE;
  2895. }
  2896.     }
  2897. }
  2898.     }
  2899.     if (!bIsImageOrPlainTextOrHTML)
  2900.     {
  2901. // /Fixes SMIL 2.0 Interop Timing #4.18;
  2902. // don't add invalid ulRepeatDur (which
  2903. // it is when "indefinite"):
  2904. if (!bRepeatDurIndefinite)
  2905. {
  2906.     pSeqNode->m_ulRepeatDur = ulRepeatDur;
  2907. }
  2908. else
  2909. {
  2910.     HX_ASSERT(0 == ulRepeatDur);
  2911. }
  2912.     }
  2913. }
  2914. SMILNode* pNodeCopy = NULL;
  2915. BOOL   bOverWrite = TRUE;
  2916. INT32 lNumRepeats = fPartialPlayFactor > 0.0 ?
  2917. lRepeatCount+1 : lRepeatCount;
  2918. for (INT32 lCount = 0; lCount < lNumRepeats; ++lCount)
  2919. {
  2920.     // we want to keep the original ids for the
  2921.     // 1st repeat so that we don't break any links
  2922.     if (0 == lCount)
  2923.     {
  2924. pNodeCopy = new SMILNode(*pNode, TRUE, this);
  2925. bOverWrite = TRUE;
  2926.     }
  2927.     else
  2928.     {
  2929. #if defined(XXXEH_REPEAT_VALUE_TIMING_SHOULD_BE_EVENT_BASED)
  2930. pNodeCopy = new SMILNode(*pNode, FALSE, this);
  2931. #else
  2932. pNodeCopy = new SMILNode(*pNode, FALSE, this,
  2933. lCount);
  2934. #endif
  2935. bOverWrite = FALSE;
  2936.     }
  2937.     pNodeCopy->m_pParent = pSeqNode;
  2938.     pNodeCopy->m_pNodeList->m_pParentNode = pNodeCopy;
  2939.     pNodeCopy->m_bRepeatHandled = TRUE;
  2940.     // after the 1st repeat, mark the rest of repeats
  2941.     // as replica(used in counting tracks etc.)
  2942.     if (lCount > 0)
  2943.     {
  2944. if (bRepeatIndefinite)
  2945. {
  2946.     if (pNodeCopy->m_tag == SMILSeq ||
  2947. pNodeCopy->m_tag == SMILExcl  ||
  2948. pNodeCopy->m_tag == SMILPar)
  2949.     {
  2950. // set indefinite repeat tag on seq
  2951. // will be used in constructing timeline elements
  2952. pSeqNode->m_repeatTag = RepeatIndefiniteOnMe;
  2953. pNodeCopy->m_repeatTag = RepeatIndefiniteOnGroup;
  2954. markRepeatReplica(pNodeCopy->m_pNodeList, RepeatIndefiniteOnGroup);
  2955.     }
  2956.     else
  2957.     {
  2958. pNodeCopy->m_repeatTag = RepeatIndefiniteOnMe;
  2959.     }
  2960. }
  2961. else
  2962. {
  2963.     if (lCount == lNumRepeats-1  &&
  2964.     fPartialPlayFactor > 0.0)
  2965.     {
  2966. pNodeCopy->m_fPartialPlayFactor =
  2967. fPartialPlayFactor;
  2968.     }
  2969.     pNodeCopy->m_repeatTag=RepeatReplica;
  2970.     if (pNodeCopy->m_tag == SMILSeq ||
  2971. pNodeCopy->m_tag == SMILExcl  ||
  2972. pNodeCopy->m_tag == SMILPar)
  2973.     {
  2974. markRepeatReplica(
  2975.     pNodeCopy->m_pNodeList,
  2976.     RepeatReplica);
  2977.     }
  2978. }
  2979.     }
  2980.     mapID(pNodeCopy, bOverWrite);
  2981.     //Fix for PR 13119: internal elements (source
  2982.     // elements, especially) of a repeated <seq> or
  2983.     // <par> tag were not getting mapped and thus
  2984.     // never played, so we need to map the IDs of
  2985.     // all in the m_pNodeList of the pNodeCopy:
  2986.     mapChildrenIDs(pNodeCopy->m_pNodeList, bOverWrite);
  2987.     pSeqNode->m_pNodeList->AddTail(pNodeCopy);
  2988. }
  2989. pSeqNode->m_pNodeList->AddTail(pSeqEndNode);
  2990. // special case to ensure all repeated elements
  2991. // are within the same group(share the same timeline)
  2992. if (!hasAncestor(SMILPar, pSeqNode)  &&
  2993.     !hasAncestor(SMILExcl, pSeqNode))
  2994. {
  2995.     SMILNode* pParNode = NULL;
  2996.     SMILNode* pParEndNode = NULL;
  2997.     createParent(pSeqNode, SMILPar, pParNode, pParEndNode);
  2998.     pParNode->m_repeatTag = pSeqNode->m_repeatTag;
  2999.     pParNode->m_pNodeList->AddTail(pSeqNode);
  3000.     pParNode->m_pNodeList->AddTail(pParEndNode);
  3001.     // now add the list to the parent...
  3002.     pNodeList->InsertBefore(lPos, pParNode);
  3003.     pNode->m_bDelete = TRUE; // delete original node
  3004.     pNode = pParNode;
  3005. }
  3006. else
  3007. {
  3008.     // now add the list to the parent...
  3009.     pNodeList->InsertBefore(lPos, pSeqNode);
  3010.     pNode->m_bDelete = TRUE; // delete original node
  3011.     pNode = pSeqNode;
  3012. }
  3013.     }
  3014. }
  3015.     }
  3016.     HX_RELEASE(pBufSMIL1Repeat);
  3017.     HX_RELEASE(pBufRepeatCount);
  3018.     HX_RELEASE(pBufRepeatDur);
  3019. }
  3020. // breadth first
  3021. rc = expandRepeatElements(pNode->m_pNodeList);
  3022. pNodeList->GetNext(lPos);
  3023.     }
  3024.     return rc;
  3025. }
  3026. HX_RESULT
  3027. CSmilParser::createParent (SMILNode* pChildNode,
  3028.    SMILNodeTag tag,
  3029.    SMILNode*& pParent,
  3030.    SMILNode*& pParentEnd)
  3031. {
  3032.     HX_RESULT hr = HXR_OK;
  3033.     pParent = NULL;
  3034.     pParentEnd = NULL;
  3035.     if (!pChildNode)
  3036.     {
  3037. hr = HXR_FAILED;
  3038. goto cleanup;
  3039.     }
  3040.     pParent = new SMILNode;
  3041.     pParent->m_pParent = pChildNode->m_pParent;
  3042.     pParent->m_tag = tag;
  3043.     pParent->m_pNodeList = new SMILNodeList;
  3044.     pParent->m_pNodeList->m_pParentNode = pParent;
  3045.     pParentEnd = new SMILNode;
  3046.     pParentEnd->m_pParent = pParent;
  3047.     if (SMILPar == tag)
  3048.     {
  3049. pParent->m_name = "par";
  3050. pParent->m_id = assignID("parCreated");
  3051. pParentEnd->m_name = "par";
  3052. pParentEnd->m_id = assignID("CLOSE-par");
  3053. pParentEnd->m_tag = SMILEndPar;
  3054.     }
  3055.     else if (SMILSeq == tag)
  3056.     {
  3057. pParent->m_name = "seq";
  3058. pParent->m_id = assignID("seqCreated");
  3059. pParentEnd->m_name = "seq";
  3060. pParentEnd->m_id = assignID("CLOSE-seq");
  3061. pParentEnd->m_tag = SMILEndSeq;
  3062.     }
  3063.     else if (SMILExcl == tag)
  3064.     {
  3065. pParent->m_name = "excl";
  3066. pParent->m_id = assignID("exclCreated");
  3067. pParentEnd->m_name = "excl";
  3068. pParentEnd->m_id = assignID("CLOSE-excl");
  3069. pParentEnd->m_tag = SMILEndExcl;
  3070.     }
  3071.     else
  3072.     {
  3073. HX_ASSERT(FALSE);
  3074. hr = HXR_FAILED;
  3075. goto cleanup;
  3076.     }
  3077.     mapID(pParent);
  3078.     mapID(pParentEnd);
  3079. cleanup:
  3080.     return hr;
  3081. }
  3082. void
  3083. CSmilParser::resetTimeline()
  3084. {
  3085.     m_pTimelineElementManager->resetTimeline();
  3086. }
  3087. HX_RESULT
  3088. CSmilParser::prepForSeek(UINT32 ulOldTime, UINT32 ulNewTime)
  3089. {
  3090.     HX_RESULT rslt = HXR_OK;
  3091.     // /Helps Fix PR 67170: reload the onLoad url list with the original list.
  3092.     // findNextPendingOnLoadURL() now handles removing all but the last of
  3093.     // simultaneous hurls to the same rn:sendTo+target, so a long seek back
  3094.     // or a long seek forward will skip all but the last of each rn-sendTo
  3095.     // link whose time is less than the seeked-to time.  Otherwise we'd be
  3096.     // simultaneously hurling possibly many links that would be immediately
  3097.     // replaced.  The end result is a better user experience, but will not
  3098.     // necessarily leave the sendTo app in the state it would have been had
  3099.     // normal (seek-less) playback arrived at that time in the presentation
  3100.     // (especially if a "back" button exists in the app).
  3101.     // If the seek-from/seek-to times are identical, leave the list as is:
  3102.     if (ulOldTime != ulNewTime  &&
  3103.     m_pOnLoadURLList  &&  m_pOnLoadURLListCopyForPostSeek)
  3104.     {
  3105. // /TRUE for special post-seek handling in FindNextPendingOnLoadURL:
  3106. m_bHandlePostSeekOnLoadURLs = TRUE;
  3107. // /Clear the old, incomplete list:
  3108.  m_pOnLoadURLList->RemoveAll();
  3109. // /Now, copy the complete original list into the active list:
  3110. LISTPOSITION lPos = m_pOnLoadURLListCopyForPostSeek->GetHeadPosition();
  3111. while (lPos)
  3112. {
  3113.     CSmilAAnchorElement* pCurAnchor = (CSmilAAnchorElement*)
  3114.     m_pOnLoadURLListCopyForPostSeek->GetNext(lPos);
  3115.     if (!pCurAnchor)
  3116.     {
  3117. HX_ASSERT(pCurAnchor); // /List shouldn't have an empty node.
  3118. continue;
  3119.     }
  3120.     // /Move from head to tail in each list, so add tail:
  3121.     m_pOnLoadURLList->AddTail((void*) pCurAnchor);
  3122. }
  3123.     }
  3124.     return rslt;
  3125. }
  3126. // /Smil doc renderer needs to get at prefetch element from prefetch ID to
  3127. // adjust element's duration when the prefetching is done (if no explicit
  3128. // duration|end was put on the prefetch element).  This does nothing if
  3129. // duration is already a valid amount.
  3130. // Returns HXR_FAILED if resetDuration() is not called on the element.
  3131. HX_RESULT
  3132. CSmilParser::handlePrefetchFinished(const char* pID, UINT32 ulTimeFinished)
  3133. {
  3134.     HX_RESULT pnr = HXR_FAILED;
  3135.     if ((UINT32)-1 != ulTimeFinished)
  3136.     {
  3137. CSmilTimelineElement* pTmLnElement = m_pTimelineElementManager->getTimelineElement(pID);
  3138. if (pTmLnElement)
  3139. {
  3140.     pnr = pTmLnElement->handlePrefetchFinished(ulTimeFinished);
  3141. }
  3142.     }
  3143.     return pnr;
  3144. }
  3145. SMILNode*
  3146. CSmilParser::getTimelineDescendent(SMILNode* pParentNode)
  3147. {
  3148.     SMILNode* pDescendentNode = NULL;
  3149.     CHXSimpleList* pNodeList = pParentNode->m_pNodeList;
  3150.     if(!pNodeList)
  3151.     {
  3152. return pDescendentNode;
  3153.     }
  3154.     CHXSimpleList::Iterator i;
  3155.     for(i=pNodeList->Begin();i!=pNodeList->End();++i)
  3156.     {
  3157. SMILNode* pNode = (SMILNode*)(*i);
  3158. if(pNode->m_bDelete)
  3159. {
  3160.     continue;
  3161. }
  3162. // /<a>, <priorityClass>, and <switch> are not timeline elements:
  3163. if(pNode->m_tag == SMILAAnchor ||
  3164. pNode->m_tag == SMILPriorityClass  ||
  3165. pNode->m_tag == SMILSwitch)
  3166. {
  3167.     pNode = getTimelineDescendent(pNode);
  3168.     if(pNode)
  3169.     {
  3170. pDescendentNode = pNode;
  3171. break;
  3172.     }
  3173. }
  3174. else
  3175. {
  3176.             if(pNode->m_tag == SMILSeq           ||
  3177.                pNode->m_tag == SMILPar           ||
  3178.                pNode->m_tag == SMILExcl          ||
  3179.                pNode->m_tag == SMILRef           ||
  3180.                pNode->m_tag == SMILText          ||
  3181.                pNode->m_tag == SMILImg           ||
  3182.                pNode->m_tag == SMILAudio         ||
  3183.                pNode->m_tag == SMILVideo         ||
  3184.                pNode->m_tag == SMILAnimation     ||
  3185.                pNode->m_tag == SMILTextstream    ||
  3186.                pNode->m_tag == SMILBrush         ||
  3187.                pNode->m_tag == SMILPrefetch      ||
  3188.                pNode->m_tag == SMILArea          ||
  3189.                pNode->m_tag == SMILAnchor        ||
  3190.                pNode->m_tag == SMILAnimate       ||
  3191.                pNode->m_tag == SMILSet           ||
  3192.                pNode->m_tag == SMILAnimateMotion ||
  3193.                pNode->m_tag == SMILAnimateColor)
  3194.     {
  3195. pDescendentNode = pNode;
  3196. break;
  3197.     }
  3198. }
  3199.     }
  3200.     return pDescendentNode;
  3201. }
  3202. SMILNode*
  3203. CSmilParser::getPrevNode(SMILNode* pCurrentNode)
  3204. {
  3205.     SMILNode* pPrevNode = NULL;
  3206.     CHXSimpleList* pNodeList = pCurrentNode->m_pParent->m_pNodeList;
  3207.     if(!pNodeList)
  3208.     {
  3209. return pPrevNode;
  3210.     }
  3211.     BOOL bFound = FALSE;
  3212.     CHXSimpleList::Iterator i;
  3213.     for(i=pNodeList->Begin();i!=pNodeList->End();++i)
  3214.     {
  3215. SMILNode* pNode = (SMILNode*)(*i);
  3216. if(pNode->m_bDelete)
  3217. {
  3218.     continue;
  3219. }
  3220. if (pNode->m_id == pCurrentNode->m_id)
  3221. {
  3222.     bFound = TRUE;
  3223.     break;
  3224. }
  3225. pPrevNode = pNode;
  3226.     }
  3227.     if (bFound)
  3228.     {
  3229. return pPrevNode;
  3230.     }
  3231.     else
  3232.     {
  3233. return NULL;
  3234.     }
  3235. }
  3236. SMILNode*
  3237. CSmilParser::getTimelineDescendent(SMILNode* pParentNode, SMILNode* pSiblingNode)
  3238. {
  3239.     SMILNode* pFirstDescendent = getTimelineDescendent(pParentNode);
  3240.     if(!pSiblingNode)
  3241.     {
  3242. return pFirstDescendent;
  3243.     }
  3244.     // first, find parent of sibling
  3245.     SMILNode* pSiblingParent = pSiblingNode->m_pParent;
  3246.     // /Fixes a bug where a non-timed parent has more than one timed
  3247.     // child, e.g., "<par><a><ref1/><ref2/></a><ref3/></par>" in which
  3248.     // case we want to find ref2, not ref3.  We want to check the actual
  3249.     // parent's children first, then, if actual parent is not a timeline
  3250.     // element, go up to its parent, p2, and look for timeline
  3251.     // sibling, else if p2 is not a timeline element then go up to its
  3252.     // parent, p3, and do the same, ...etc. until a timed child is found
  3253.     // or until pX is a timeline element. Has been a bug since SMIL 1:
  3254.     SMILNode* pEffectiveSiblingNode = pSiblingNode;
  3255.     SMILNode* pFoundSibling = NULL;
  3256.     do
  3257.     {
  3258. BOOL bParentIsTimedObject = isTimeContainerObject(pSiblingParent)  ||
  3259. isMediaObject(pSiblingParent);
  3260. SMILNode* pNextSibling = findNextSibling(pEffectiveSiblingNode);
  3261. if (!pNextSibling)
  3262. {
  3263.     // /No sibling found; if parent is a timed element, then
  3264.     // we're done because there is no next sibling:
  3265.     if (bParentIsTimedObject)
  3266.     {
  3267. pFoundSibling = NULL;
  3268. break;
  3269.     }
  3270.     else
  3271.     {
  3272. // /Go up and find non-timed parent's next sibling:
  3273. pEffectiveSiblingNode = pSiblingParent;
  3274. pSiblingParent = pSiblingParent->m_pParent;
  3275.     }
  3276. }
  3277. else if (isTimeContainerObject(pNextSibling)  ||
  3278. isMediaObject(pNextSibling))
  3279. {
  3280.     pFoundSibling = pNextSibling;
  3281.     break;
  3282. }
  3283. // /Sibling is not a timed element; return its first timed descendant
  3284. // if one exists, otherwise return its next sibling (and let the
  3285. // caller of this determine what to do from there):
  3286. else
  3287. {
  3288.     SMILNode* pNextSiblingTimelineDescendant =
  3289.     getTimelineDescendent(pNextSibling, NULL);
  3290.     if (!pNextSiblingTimelineDescendant)
  3291.     {
  3292. pFoundSibling = pNextSibling;
  3293. break;
  3294.     }
  3295.     else
  3296.     {
  3297. pFoundSibling = pNextSiblingTimelineDescendant;
  3298. break;
  3299.     }
  3300. }
  3301.     } while (pSiblingParent);
  3302.     return pFoundSibling;
  3303. }
  3304. // /Finds pNode's parent's next child, if any.  DOES NOT CARE IF NODE IS
  3305. // A TIMED ELEMENT OR NOT; this is strictly a document tree thing:
  3306. SMILNode*
  3307. CSmilParser::findNextSibling(SMILNode* pNode)
  3308. {
  3309.     SMILNode* pNextSibling = NULL;
  3310.     CHXSimpleList::Iterator i;
  3311.     if(!pNode  ||  !pNode->m_pParent)
  3312.     {
  3313. goto cleanup;
  3314.     }
  3315.     // /Go through parent's list until we find pNode, then return next one:
  3316.     i = pNode->m_pParent->m_pNodeList->Begin();
  3317.     for (; i != pNode->m_pParent->m_pNodeList->End(); ++i)
  3318.     {
  3319. SMILNode* pThisNode = (SMILNode*)(*i);
  3320. // /(NOTE: even if pThisNode->m_bDelete, we still want its sibling;
  3321. // this happens when repeat of x results in: seq/x,x1,x2.../seq,x
  3322. // where the last x has m_bDelete set.)
  3323. if (pThisNode == pNode)
  3324. {
  3325.     // /We found pNode so return the next one, if any:
  3326.     if (++i != pNode->m_pParent->m_pNodeList->End())
  3327.     {
  3328. pNextSibling = (SMILNode*)(*(i));
  3329. // /However, if the node is an end tag (SMILEndXYZ) and is
  3330. // the last child, then we want to ignore it:
  3331. if (pNode->m_pParent->m_pNodeList->End() == ++i  &&
  3332. (pNextSibling  &&  isEndTagObject(pNextSibling)))
  3333. {
  3334.     pNextSibling = NULL;
  3335. }
  3336.     }
  3337.     break; // /We either found it or it doesn't exist.
  3338. }