src/cm/media/js/lib/yui/yui3-3.15.0/build/dataschema-xml/dataschema-xml-debug.js
author gibus
Mon, 10 Mar 2014 15:19:48 +0100
changeset 602 e16a97fb364a
permissions -rw-r--r--
Use YUI 3.15
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
            Y.log("XML data could not be schema-parsed: " + Y.dump(data) + " " + Y.dump(data), "error", "dataschema-xml");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   128
            data_out.error = new Error("XML schema parse failure");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   129
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   130
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   131
        return data_out;
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
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   135
     * Get an XPath-specified value for a given field from an XML node or document.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   136
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   137
     * @method _getLocationValue
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   138
     * @param field {String | Object} Field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   139
     * @param context {Object} XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   140
     * @return {Object} Data value or null.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   141
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   142
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   143
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   144
    _getLocationValue: function(field, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   145
        var locator = field.locator || field.key || field,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   146
            xmldoc = context.ownerDocument || context,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   147
            result, res, value = null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   148
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   149
        try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   150
            result = SchemaXML._getXPathResult(locator, context, xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   151
            while ((res = result.iterateNext())) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   152
                value = res.textContent || res.value || res.text || res.innerHTML || res.innerText || null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   153
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   154
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   155
            // FIXME: Why defer to a method that is mixed into this object?
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   156
            // DSchema.Base is mixed into DSchema.XML (et al), so
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   157
            // DSchema.XML.parse(...) will work.  This supports the use case
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   158
            // where DSchema.Base.parse is changed, and that change is then
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   159
            // seen by all DSchema.* implementations, but does not support the
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   160
            // case where redefining DSchema.XML.parse changes behavior. In
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   161
            // fact, DSchema.XML.parse is never even called.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   162
            return Y.DataSchema.Base.parse.call(this, value, field);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   163
        } catch (e) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   164
            Y.log('SchemaXML._getLocationValue failed: ' + e.message);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   165
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   166
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   167
        return null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   168
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   169
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   170
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   171
     * Fetches the XPath-specified result for a given location in an XML node
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   172
     * or document.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   173
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   174
     * @method _getXPathResult
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   175
     * @param locator {String} The XPath location.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   176
     * @param context {Object} XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   177
     * @param xmldoc {Object} XML document to resolve namespace.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   178
     * @return {Object} Data collection or null.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   179
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   180
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   181
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   182
    _getXPathResult: function(locator, context, xmldoc) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   183
        // Standards mode
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   184
        if (! Lang.isUndefined(xmldoc.evaluate)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   185
            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
   186
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   187
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   188
        // IE mode
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   189
        else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   190
            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
   191
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   192
            // XPath is supported
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   193
            try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   194
                // 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
   195
                try {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   196
                   xmldoc.setProperty("SelectionLanguage", "XPath");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   197
                } catch (e) {}
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   198
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   199
                values = context.selectNodes(locator);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   200
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   201
            // Fallback for DOM nodes and fragments
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   202
            catch (e) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   203
                // Iterate over each locator piece
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   204
                for (; i<l && context; i++) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   205
                    location = locatorArray[i];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   206
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   207
                    // grab nth child []
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   208
                    if ((location.indexOf("[") > -1) && (location.indexOf("]") > -1)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   209
                        subloc = location.slice(location.indexOf("[")+1, location.indexOf("]"));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   210
                        //XPath is 1-based while DOM is 0-based
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   211
                        subloc--;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   212
                        context = context.children[subloc];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   213
                        isNth = true;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   214
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   215
                    // grab attribute value @
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   216
                    else if (location.indexOf("@") > -1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   217
                        subloc = location.substr(location.indexOf("@"));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   218
                        context = subloc ? context.getAttribute(subloc.replace('@', '')) : context;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   219
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   220
                    // grab that last instance of tagName
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   221
                    else if (-1 < location.indexOf("//")) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   222
                        subloc = context.getElementsByTagName(location.substr(2));
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   223
                        context = subloc.length ? subloc[subloc.length - 1] : null;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   224
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   225
                    // find the last matching location in children
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   226
                    else if (l != i + 1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   227
                        for (m=context.childNodes.length-1; 0 <= m; m-=1) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   228
                            if (location === context.childNodes[m].tagName) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   229
                                context = context.childNodes[m];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   230
                                m = -1;
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
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   235
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   236
                if (context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   237
                    // attribute
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   238
                    if (Lang.isString(context)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   239
                        values[0] = {value: context};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   240
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   241
                    // nth child
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   242
                    else if (isNth) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   243
                        values[0] = {value: context.innerHTML};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   244
                    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   245
                    // all children
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   246
                    else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   247
                        values = Y.Array(context.childNodes, 0, true);
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
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   251
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   252
            // returning a mock-standard object for IE
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   253
            return {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   254
                index: 0,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   255
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   256
                iterateNext: function() {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   257
                    if (this.index >= this.values.length) {return undefined;}
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   258
                    var result = this.values[this.index];
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   259
                    this.index += 1;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   260
                    return result;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   261
                },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   262
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   263
                values: values
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
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   268
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   269
     * Schema-parsed result field.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   270
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   271
     * @method _parseField
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   272
     * @param field {String | Object} Required. Field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   273
     * @param result {Object} Required. Schema parsed data object.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   274
     * @param context {Object} Required. XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   275
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   276
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   277
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   278
    _parseField: function(field, result, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   279
        var key = field.key || field,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   280
            parsed;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   281
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   282
        if (field.schema) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   283
            parsed = { results: [], meta: {} };
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   284
            parsed = SchemaXML._parseResults(field.schema, context, parsed);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   285
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   286
            result[key] = parsed.results;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   287
        } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   288
            result[key] = SchemaXML._getLocationValue(field, context);
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
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   292
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   293
     * Parses results data according to schema
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   294
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   295
     * @method _parseMeta
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   296
     * @param xmldoc_in {Object} XML document parse.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   297
     * @param data_out {Object} In-progress schema-parsed data to update.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   298
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   299
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   300
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   301
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   302
    _parseMeta: function(metaFields, xmldoc_in, data_out) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   303
        if(Lang.isObject(metaFields)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   304
            var key,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   305
                xmldoc = xmldoc_in.ownerDocument || xmldoc_in;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   306
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   307
            for(key in metaFields) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   308
                if (metaFields.hasOwnProperty(key)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   309
                    data_out.meta[key] = SchemaXML._getLocationValue(metaFields[key], xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   310
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   311
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   312
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   313
        return data_out;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   314
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   315
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   316
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   317
     * Schema-parsed result to add to results list.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   318
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   319
     * @method _parseResult
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   320
     * @param fields {Array} Required. A collection of field definition.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   321
     * @param context {Object} Required. XML node or document to search within.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   322
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   323
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   324
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   325
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   326
    _parseResult: function(fields, context) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   327
        var result = {}, j;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   328
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   329
        // Find each field value
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   330
        for (j=fields.length-1; 0 <= j; j--) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   331
            SchemaXML._parseField(fields[j], result, context);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   332
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   333
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   334
        return result;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   335
    },
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   336
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   337
    /**
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   338
     * Schema-parsed list of results from full data
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   339
     *
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   340
     * @method _parseResults
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   341
     * @param schema {Object} Schema to parse against.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   342
     * @param context {Object} XML node or document to parse.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   343
     * @param data_out {Object} In-progress schema-parsed data to update.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   344
     * @return {Object} Schema-parsed data.
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   345
     * @static
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   346
     * @protected
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   347
     */
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   348
    _parseResults: function(schema, context, data_out) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   349
        if (schema.resultListLocator && Lang.isArray(schema.resultFields)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   350
            var xmldoc = context.ownerDocument || context,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   351
                fields = schema.resultFields,
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   352
                results = [],
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   353
                node, nodeList, i=0;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   354
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   355
            if (schema.resultListLocator.match(/^[:\-\w]+$/)) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   356
                nodeList = context.getElementsByTagName(schema.resultListLocator);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   357
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   358
                // loop through each result node
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   359
                for (i = nodeList.length - 1; i >= 0; --i) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   360
                    results[i] = SchemaXML._parseResult(fields, nodeList[i]);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   361
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   362
            } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   363
                nodeList = SchemaXML._getXPathResult(schema.resultListLocator, context, xmldoc);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   364
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   365
                // loop through the nodelist
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   366
                while ((node = nodeList.iterateNext())) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   367
                    results[i] = SchemaXML._parseResult(fields, node);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   368
                    i += 1;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   369
                }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   370
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   371
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   372
            if (results.length) {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   373
                data_out.results = results;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   374
            } else {
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   375
                data_out.error = new Error("XML schema result nodes retrieval failure");
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   376
            }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   377
        }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   378
        return data_out;
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   379
    }
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   380
};
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   381
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   382
Y.DataSchema.XML = Y.mix(SchemaXML, Y.DataSchema.Base);
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   383
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   384
e16a97fb364a Use YUI 3.15
gibus
parents:
diff changeset
   385
}, '@VERSION@', {"requires": ["dataschema-base"]});