使用Javascript确定JSF库资源

时间:2018-08-03 20:30:58

标签: javascript jsf primefaces omnifaces primefaces-extensions

我是Primefaces Extensions项目的开发人员,并且使用核心的Primefaces JS函数在Javascript中获取DocumentViewer和CKEditor组件的资源位置。 getFacesResource()函数位于PrimeFaces core.js中,看起来像这样。

   /**
     * Builds a resource URL for given parameters.
     *
     * @param {string} name The name of the resource. For example: primefaces.js
     * @param {string} library The library of the resource. For example: primefaces
     * @param {string} version The version of the library. For example: 5.1
     * @returns {string} The resource URL.
     */
getFacesResource : function(name, library, version) {
    // just get sure - name shoudln't start with a slash
    if (name.indexOf('/') === 0)
    {
        name = name.substring(1, name.length);
    }

    var scriptURI = $('script[src*="/' + PrimeFaces.RESOURCE_IDENTIFIER + '/core.js"]').attr('src');
    // portlet
    if (!scriptURI) {
        scriptURI = $('script[src*="' + PrimeFaces.RESOURCE_IDENTIFIER + '=core.js"]').attr('src');
    }

    scriptURI = scriptURI.replace('core.js', name);

它的作用是在DOM中找到“ core.js”脚本,然后删除core.js,以便为您构造的用于查找资源的新URL提供适当的位置。

我们为什么需要这样做?

CKEDitor和DocumentViewer组件是复杂的组件,可加载许多插件,语言文件等。每次升级时,我们都无法修改这些插件在其他地方使用的核心JS文件,因此我们必须编辑其核心源代码。因此,为使DocumentViewer加载语言包,库值具有“ documentviewer / locale / en-GB.locale.txt”,以为DocumentViewer使用的PDF.js加载英语语言包。但是,这需要成为真正的资源,因此PrimeFaces.getFacesResource('documentviewer/locale/en-GB.locale.txt');将URL转换为PDF.JS代码可以找到的适当资源,例如:

https://www.primefaces.org/showcase-ext/javax.faces.resource/documentviewer/locale/en-GB.locale.txt.jsf?ln=primefaces-extensions&v=6.2.5

它处理知道完整的URL,附加库版本以及知道当前服务器是否正在提供.jsf或.xhtml扩展名等信息。

问题:

我正在使用OmniFaces CombinedResourceHandler来获取您的所有JS文件,并从找到的所有JS文件中创建一个脚本。因此10个不同的JS文件现在变成1种性能资源...

/javax.faces.resource/XXX.js.xhtml?ln=omnifaces.combined&v=1532916484000

现在这会破坏核心PrimeFaces代码,因为它无法查找core.js,因为core.js不再存在于页面上。

我尝试使用此开关将core.js排除在组合范围之外。

<context-param>
    <param-name>org.omnifaces.COMBINED_RESOURCE_HANDLER_EXCLUDED_RESOURCES</param-name>
    <param-value>primefaces:core.js</param-value>
</context-param>

在omnifaces.combined.js之后加载PrimeFaces core.js,这会破坏整个页面。确认输出是这个...

<script src="/primeext-showcase/javax.faces.resource/XXX.js.jsf?ln=omnifaces.combined&amp;v=1533319992000"></script>
<script src="/primeext-showcase/javax.faces.resource/core.js.jsf?ln=primefaces&amp;v=6.2">

所以我的问题是,无论OmniFaces是否已合并脚本,我如何才能用纯JavaScript替换/修复PrimeFaces.getFacesResource()JS函数以起作用?

1 个答案:

答案 0 :(得分:1)

我能够重写该方法,以免使用正则表达式在页面上未找到core.js。现在,它在PrimeFaces处于正常模式时可以使用,如果使用OmniFaces CombinedResourceHandler,则可以使用。我将把它作为PrimeFaces的补丁提交。

这是最终的工作方法:

/**
    * Builds a resource URL for given parameters.
    * 
    * @param {string}
    *        name The name of the resource. For example: primefaces.js
    * @param {string}
    *        library The library of the resource. For example: primefaces
    * @param {string}
    *        version The version of the library. For example: 5.1
    * @returns {string} The resource URL.
    */
   getFacesResource : function(name, library, version) {
      // just get sure - name shoudln't start with a slash
      if (name.indexOf('/') === 0) {
         name = name.substring(1, name.length);
      }

      // find any JS served JSF resource
      var scriptURI = $('script[src*="/' + PrimeFaces.RESOURCE_IDENTIFIER + '/"]').first().attr('src');
      // portlet
      if (!scriptURI) {
        scriptURI = $('script[src*="' + PrimeFaces.RESOURCE_IDENTIFIER + '="]').first().attr('src');
      }

      // find script...normal is '/core.js' and portlets are '=core.js'
      var scriptRegex = new RegExp('\\/' + PrimeFaces.RESOURCE_IDENTIFIER + '(\\/|=)(.*?)\\.js');

      // find script to replace e.g. 'core.js'
      var currentScriptName = scriptRegex.exec(scriptURI)[2] + '.js';

      // replace core.js with our custom name
      scriptURI = scriptURI.replace(currentScriptName, name);

      // find the library like ln=primefaces
      var libraryRegex = new RegExp('[?&]([^&=]*)ln=(.*?)(&|$)');

      // find library to replace e.g. 'ln=primefaces'
      var currentLibraryName = 'ln=' + libraryRegex.exec(scriptURI)[2];

      // In a portlet environment, url parameters may be namespaced.
      var namespace = '';
      var urlParametersAreNamespaced = !(scriptURI.indexOf('?' + currentLibraryName) > -1 || scriptURI.indexOf('&'
            + currentLibraryName) > -1);

      if (urlParametersAreNamespaced) {
         namespace = new RegExp('[?&]([^&=]+)' + currentLibraryName + '($|&)').exec(scriptURI)[1];
      }

      // If the parameters are namespaced, the namespace must be included
      // when replacing parameters.
      scriptURI = scriptURI.replace(namespace + currentLibraryName, namespace + 'ln=' + library);

      if (version) {
         var extractedVersion = new RegExp('[?&]' + namespace + 'v=([^&]*)').exec(scriptURI)[1];
         scriptURI = scriptURI.replace(namespace + 'v=' + extractedVersion, namespace + 'v=' + version);
      }

      var prefix = window.location.protocol + '//' + window.location.host;
      return scriptURI.indexOf(prefix) >= 0 ? scriptURI : prefix + scriptURI;
   },