src/cm/media/js/lib/yui/yui3-3.15.0/build/dataschema-xml/dataschema-xml.js
author ymh <ymh.work@gmail.com>
Fri, 14 Mar 2014 13:16:10 +0100
changeset 611 fa66f4bb1563
parent 602 e16a97fb364a
permissions -rw-r--r--
add some more custom config and put every thing in comment in the custom.yaml template
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
602
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     1
YUI.add('dataschema-xml', function (Y, NAME) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     2
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     3
/**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     4
Provides a DataSchema implementation which can be used to work with XML data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     5
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     6
@module dataschema
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     7
@submodule dataschema-xml
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     8
**/
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
     9
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    10
/**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    11
Provides a DataSchema implementation which can be used to work with XML data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    12
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    13
See the `apply` method for usage.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    14
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    15
@class DataSchema.XML
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    16
@extends DataSchema.Base
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    17
@static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    18
**/
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    19
var Lang = Y.Lang,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    20
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    21
    okNodeType = {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    22
        1 : true,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    23
        9 : true,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    24
        11: true
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    25
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    26
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    27
    SchemaXML;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    28
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    29
SchemaXML = {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    30
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    31
    ////////////////////////////////////////////////////////////////////////////
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    32
    //
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    33
    // DataSchema.XML static methods
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    34
    //
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    35
    ////////////////////////////////////////////////////////////////////////////
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    36
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    37
    Applies a schema to an XML data tree, returning a normalized object with
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    38
    results in the `results` property. Additional information can be parsed out
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    39
    of the XML for inclusion in the `meta` property of the response object.  If
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    40
    an error is encountered during processing, an `error` property will be
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    41
    added.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    42
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    43
    Field data in the nodes captured by the XPath in _schema.resultListLocator_
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    44
    is extracted with the field identifiers described in _schema.resultFields_.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    45
    Field identifiers are objects with the following properties:
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    46
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    47
      * `key`    : <strong>(required)</strong> The desired property name to use
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    48
            store the retrieved value in the result object.  If `locator` is
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    49
            not specified, `key` is also used as the XPath locator (String)
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    50
      * `locator`: The XPath locator to the node or attribute within each
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    51
            result node found by _schema.resultListLocator_ containing the
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    52
            desired field data (String)
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    53
      * `parser` : A function or the name of a function on `Y.Parsers` used
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    54
            to convert the input value into a normalized type.  Parser
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    55
            functions are passed the value as input and are expected to
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    56
            return a value.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    57
      * `schema` : Used to retrieve nested field data into an array for
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    58
            assignment as the result field value.  This object follows the same
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    59
            conventions as _schema_.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    60
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    61
    If no value parsing or nested parsing is needed, you can use XPath locators
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    62
    (strings) instead of field identifiers (objects) -- see example below.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    63
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    64
    `response.results` will contain an array of objects with key:value pairs.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    65
    The keys are the field identifier `key`s, and the values are the data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    66
    values extracted from the nodes or attributes found by the field `locator`
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    67
    (or `key` fallback).
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    68
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    69
    To extract additional information from the XML, include an array of
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    70
    XPath locators in _schema.metaFields_.  The collected values will be
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    71
    stored in `response.meta` with the XPath locator as keys.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    72
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    73
    @example
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    74
        var schema = {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    75
                resultListLocator: '//produce/item',
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    76
                resultFields: [
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    77
                    {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    78
                        locator: 'name',
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    79
                        key: 'name'
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    80
                    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    81
                    {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    82
                        locator: 'color',
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    83
                        key: 'color',
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    84
                        parser: function (val) { return val.toUpperCase(); }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    85
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    86
                ]
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    87
            };
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    88
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    89
        // Assumes data like
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    90
        // <inventory>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    91
        //   <produce>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    92
        //     <item><name>Banana</name><color>yellow</color></item>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    93
        //     <item><name>Orange</name><color>orange</color></item>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    94
        //     <item><name>Eggplant</name><color>purple</color></item>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    95
        //   </produce>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    96
        // </inventory>
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    97
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    98
        var response = Y.DataSchema.JSON.apply(schema, data);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
    99
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   100
        // response.results[0] is { name: "Banana", color: "YELLOW" }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   101
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   102
    @method apply
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   103
    @param {Object} schema Schema to apply.  Supported configuration
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   104
        properties are:
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   105
      @param {String} [schema.resultListLocator] XPath locator for the
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   106
          XML nodes that contain the data to flatten into `response.results`
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   107
      @param {Array} [schema.resultFields] Field identifiers to
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   108
          locate/assign values in the response records. See above for
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   109
          details.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   110
      @param {Array} [schema.metaFields] XPath locators to extract extra
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   111
          non-record related information from the XML data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   112
    @param {XMLDocument} data XML data to parse
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   113
    @return {Object} An Object with properties `results` and `meta`
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   114
    @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   115
    **/
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   116
    apply: function(schema, data) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   117
        var xmldoc = data, // unnecessary variables
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   118
            data_out = { results: [], meta: {} };
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   119
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   120
        if (xmldoc && okNodeType[xmldoc.nodeType] && schema) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   121
            // Parse results data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   122
            data_out = SchemaXML._parseResults(schema, xmldoc, data_out);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   123
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   124
            // Parse meta data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   125
            data_out = SchemaXML._parseMeta(schema.metaFields, xmldoc, data_out);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   126
        } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   127
            data_out.error = new Error("XML schema parse failure");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   128
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   129
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   130
        return data_out;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   131
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   132
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   133
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   134
     * Get an XPath-specified value for a given field from an XML node or document.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   135
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   136
     * @method _getLocationValue
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   137
     * @param field {String | Object} Field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   138
     * @param context {Object} XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   139
     * @return {Object} Data value or null.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   140
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   141
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   142
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   143
    _getLocationValue: function(field, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   144
        var locator = field.locator || field.key || field,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   145
            xmldoc = context.ownerDocument || context,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   146
            result, res, value = null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   147
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   148
        try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   149
            result = SchemaXML._getXPathResult(locator, context, xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   150
            while ((res = result.iterateNext())) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   151
                value = res.textContent || res.value || res.text || res.innerHTML || res.innerText || null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   152
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   153
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   154
            // FIXME: Why defer to a method that is mixed into this object?
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   155
            // DSchema.Base is mixed into DSchema.XML (et al), so
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   156
            // DSchema.XML.parse(...) will work.  This supports the use case
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   157
            // where DSchema.Base.parse is changed, and that change is then
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   158
            // seen by all DSchema.* implementations, but does not support the
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   159
            // case where redefining DSchema.XML.parse changes behavior. In
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   160
            // fact, DSchema.XML.parse is never even called.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   161
            return Y.DataSchema.Base.parse.call(this, value, field);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   162
        } catch (e) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   163
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   164
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   165
        return null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   166
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   167
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   168
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   169
     * Fetches the XPath-specified result for a given location in an XML node
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   170
     * or document.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   171
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   172
     * @method _getXPathResult
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   173
     * @param locator {String} The XPath location.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   174
     * @param context {Object} XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   175
     * @param xmldoc {Object} XML document to resolve namespace.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   176
     * @return {Object} Data collection or null.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   177
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   178
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   179
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   180
    _getXPathResult: function(locator, context, xmldoc) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   181
        // Standards mode
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   182
        if (! Lang.isUndefined(xmldoc.evaluate)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   183
            return xmldoc.evaluate(locator, context, xmldoc.createNSResolver(context.ownerDocument ? context.ownerDocument.documentElement : context.documentElement), 0, null);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   184
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   185
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   186
        // IE mode
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   187
        else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   188
            var values=[], locatorArray = locator.split(/\b\/\b/), i=0, l=locatorArray.length, location, subloc, m, isNth;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   189
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   190
            // XPath is supported
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   191
            try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   192
                // this fixes the IE 5.5+ issue where childnode selectors begin at 0 instead of 1
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   193
                try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   194
                   xmldoc.setProperty("SelectionLanguage", "XPath");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   195
                } catch (e) {}
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   196
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   197
                values = context.selectNodes(locator);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   198
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   199
            // Fallback for DOM nodes and fragments
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   200
            catch (e) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   201
                // Iterate over each locator piece
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   202
                for (; i<l && context; i++) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   203
                    location = locatorArray[i];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   204
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   205
                    // grab nth child []
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   206
                    if ((location.indexOf("[") > -1) && (location.indexOf("]") > -1)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   207
                        subloc = location.slice(location.indexOf("[")+1, location.indexOf("]"));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   208
                        //XPath is 1-based while DOM is 0-based
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   209
                        subloc--;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   210
                        context = context.children[subloc];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   211
                        isNth = true;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   212
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   213
                    // grab attribute value @
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   214
                    else if (location.indexOf("@") > -1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   215
                        subloc = location.substr(location.indexOf("@"));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   216
                        context = subloc ? context.getAttribute(subloc.replace('@', '')) : context;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   217
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   218
                    // grab that last instance of tagName
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   219
                    else if (-1 < location.indexOf("//")) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   220
                        subloc = context.getElementsByTagName(location.substr(2));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   221
                        context = subloc.length ? subloc[subloc.length - 1] : null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   222
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   223
                    // find the last matching location in children
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   224
                    else if (l != i + 1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   225
                        for (m=context.childNodes.length-1; 0 <= m; m-=1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   226
                            if (location === context.childNodes[m].tagName) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   227
                                context = context.childNodes[m];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   228
                                m = -1;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   229
                            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   230
                        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   231
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   232
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   233
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   234
                if (context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   235
                    // attribute
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   236
                    if (Lang.isString(context)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   237
                        values[0] = {value: context};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   238
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   239
                    // nth child
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   240
                    else if (isNth) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   241
                        values[0] = {value: context.innerHTML};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   242
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   243
                    // all children
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   244
                    else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   245
                        values = Y.Array(context.childNodes, 0, true);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   246
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   247
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   248
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   249
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   250
            // returning a mock-standard object for IE
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   251
            return {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   252
                index: 0,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   253
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   254
                iterateNext: function() {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   255
                    if (this.index >= this.values.length) {return undefined;}
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   256
                    var result = this.values[this.index];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   257
                    this.index += 1;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   258
                    return result;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   259
                },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   260
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   261
                values: values
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   262
            };
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   263
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   264
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   265
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   266
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   267
     * Schema-parsed result field.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   268
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   269
     * @method _parseField
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   270
     * @param field {String | Object} Required. Field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   271
     * @param result {Object} Required. Schema parsed data object.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   272
     * @param context {Object} Required. XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   273
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   274
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   275
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   276
    _parseField: function(field, result, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   277
        var key = field.key || field,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   278
            parsed;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   279
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   280
        if (field.schema) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   281
            parsed = { results: [], meta: {} };
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   282
            parsed = SchemaXML._parseResults(field.schema, context, parsed);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   283
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   284
            result[key] = parsed.results;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   285
        } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   286
            result[key] = SchemaXML._getLocationValue(field, context);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   287
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   288
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   289
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   290
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   291
     * Parses results data according to schema
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   292
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   293
     * @method _parseMeta
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   294
     * @param xmldoc_in {Object} XML document parse.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   295
     * @param data_out {Object} In-progress schema-parsed data to update.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   296
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   297
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   298
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   299
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   300
    _parseMeta: function(metaFields, xmldoc_in, data_out) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   301
        if(Lang.isObject(metaFields)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   302
            var key,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   303
                xmldoc = xmldoc_in.ownerDocument || xmldoc_in;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   304
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   305
            for(key in metaFields) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   306
                if (metaFields.hasOwnProperty(key)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   307
                    data_out.meta[key] = SchemaXML._getLocationValue(metaFields[key], xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   308
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   309
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   310
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   311
        return data_out;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   312
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   313
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   314
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   315
     * Schema-parsed result to add to results list.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   316
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   317
     * @method _parseResult
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   318
     * @param fields {Array} Required. A collection of field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   319
     * @param context {Object} Required. XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   320
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   321
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   322
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   323
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   324
    _parseResult: function(fields, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   325
        var result = {}, j;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   326
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   327
        // Find each field value
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   328
        for (j=fields.length-1; 0 <= j; j--) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   329
            SchemaXML._parseField(fields[j], result, context);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   330
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   331
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   332
        return result;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   333
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   334
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   335
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   336
     * Schema-parsed list of results from full data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   337
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   338
     * @method _parseResults
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   339
     * @param schema {Object} Schema to parse against.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   340
     * @param context {Object} XML node or document to parse.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   341
     * @param data_out {Object} In-progress schema-parsed data to update.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   342
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   343
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   344
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   345
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   346
    _parseResults: function(schema, context, data_out) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   347
        if (schema.resultListLocator && Lang.isArray(schema.resultFields)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   348
            var xmldoc = context.ownerDocument || context,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   349
                fields = schema.resultFields,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   350
                results = [],
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   351
                node, nodeList, i=0;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   352
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   353
            if (schema.resultListLocator.match(/^[:\-\w]+$/)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   354
                nodeList = context.getElementsByTagName(schema.resultListLocator);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   355
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   356
                // loop through each result node
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   357
                for (i = nodeList.length - 1; i >= 0; --i) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   358
                    results[i] = SchemaXML._parseResult(fields, nodeList[i]);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   359
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   360
            } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   361
                nodeList = SchemaXML._getXPathResult(schema.resultListLocator, context, xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   362
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   363
                // loop through the nodelist
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   364
                while ((node = nodeList.iterateNext())) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   365
                    results[i] = SchemaXML._parseResult(fields, node);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   366
                    i += 1;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   367
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   368
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   369
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   370
            if (results.length) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   371
                data_out.results = results;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   372
            } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   373
                data_out.error = new Error("XML schema result nodes retrieval failure");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   374
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   375
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   376
        return data_out;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   377
    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   378
};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   379
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   380
Y.DataSchema.XML = Y.mix(SchemaXML, Y.DataSchema.Base);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   381
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   382
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   383
}, '@VERSION@', {"requires": ["dataschema-base"]});