/* -*- Mode: java; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
*
* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is Rhino code, released
* May 6, 1999.
*
* The Initial Developer of the Original Code is
* Netscape Communications Corporation.
* Portions created by the Initial Developer are Copyright (C) 1997-2000
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Ethan Hugg
* Terry Lucas
* Milen Nankov
*
* Alternatively, the contents of this file may be used under the terms of
* the GNU General Public License Version 2 or later (the "GPL"), in which
* case the provisions of the GPL are applicable instead of those above. If
* you wish to allow use of your version of this file only under the terms of
* the GPL and not to allow others to use your version of this file under the
* MPL, indicate your decision by deleting the provisions above and replacing
* them with the notice and other provisions required by the GPL. If you do
* not delete the provisions above, a recipient may use your version of this
* file under either the MPL or the GPL.
*
* ***** END LICENSE BLOCK ***** */
package org.mozilla.javascript.xmlimpl;
import org.mozilla.javascript.*;
/**
* Class QName
*
*/
final class QName extends IdScriptableObject
{
static final long serialVersionUID = 416745167693026750L;
private static final Object QNAME_TAG = new Object();
private XMLLibImpl lib;
private QName prototype;
private XmlNode.QName delegate;
private QName() {
}
static QName create(XMLLibImpl lib, Scriptable scope, QName prototype, XmlNode.QName delegate) {
QName rv = new QName();
rv.lib = lib;
rv.setParentScope(scope);
rv.prototype = prototype;
rv.setPrototype(prototype);
rv.delegate = delegate;
return rv;
}
// /** @deprecated */
// static QName create(XMLLibImpl lib, XmlNode.QName nodeQname) {
// return create(lib, lib.globalScope(), lib.qnamePrototype(), nodeQname);
// }
void exportAsJSClass(boolean sealed) {
exportAsJSClass(MAX_PROTOTYPE_ID, getParentScope(), sealed);
}
public String toString() {
// ECMA357 13.3.4.2
if (delegate.getNamespace() == null) {
return "*::" + localName();
} else if (delegate.getNamespace().isGlobal()) {
// leave as empty
return localName();
} else {
return uri() + "::" + localName();
}
}
public String localName() {
if (delegate.getLocalName() == null) return "*";
return delegate.getLocalName();
}
/**
@deprecated
This property is supposed to be invisible and I think we can make it private at some point, though Namespace
might need it
*/
String prefix() {
if (delegate.getNamespace() == null) return null;
return delegate.getNamespace().getPrefix();
}
String uri() {
if (delegate.getNamespace() == null) return null;
return delegate.getNamespace().getUri();
}
/** @deprecated */
final XmlNode.QName toNodeQname() {
return delegate;
}
final XmlNode.QName getDelegate() {
return delegate;
}
public boolean equals(Object obj) {
if(!(obj instanceof QName)) return false;
return equals((QName)obj);
}
protected Object equivalentValues(Object value)
{
if(!(value instanceof QName)) return Scriptable.NOT_FOUND;
boolean result = equals((QName)value);
return result ? Boolean.TRUE : Boolean.FALSE;
}
private boolean equals(QName q) {
return this.delegate.isEqualTo(q.delegate);
}
public String getClassName() {
return "QName";
}
public Object getDefaultValue(Class hint) {
return toString();
}
// #string_id_map#
private static final int
Id_localName = 1,
Id_uri = 2,
MAX_INSTANCE_ID = 2;
protected int getMaxInstanceId()
{
return super.getMaxInstanceId() + MAX_INSTANCE_ID;
}
protected int findInstanceIdInfo(String s)
{
int id;
// #generated# Last update: 2007-08-20 08:21:41 EDT
L0: { id = 0; String X = null;
int s_length = s.length();
if (s_length==3) { X="uri";id=Id_uri; }
else if (s_length==9) { X="localName";id=Id_localName; }
if (X!=null && X!=s && !X.equals(s)) id = 0;
break L0;
}
// #/generated#
if (id == 0) return super.findInstanceIdInfo(s);
int attr;
switch (id) {
case Id_localName:
case Id_uri:
attr = PERMANENT | READONLY;
break;
default: throw new IllegalStateException();
}
return instanceIdInfo(attr, super.getMaxInstanceId() + id);
}
// #/string_id_map#
protected String getInstanceIdName(int id)
{
switch (id - super.getMaxInstanceId()) {
case Id_localName: return "localName";
case Id_uri: return "uri";
}
return super.getInstanceIdName(id);
}
protected Object getInstanceIdValue(int id)
{
switch (id - super.getMaxInstanceId()) {
case Id_localName: return localName();
case Id_uri: return uri();
}
return super.getInstanceIdValue(id);
}
// #string_id_map#
private static final int
Id_constructor = 1,
Id_toString = 2,
Id_toSource = 3,
MAX_PROTOTYPE_ID = 3;
protected int findPrototypeId(String s)
{
int id;
// #generated# Last update: 2007-08-20 08:21:41 EDT
L0: { id = 0; String X = null; int c;
int s_length = s.length();
if (s_length==8) {
c=s.charAt(3);
if (c=='o') { X="toSource";id=Id_toSource; }
else if (c=='t') { X="toString";id=Id_toString; }
}
else if (s_length==11) { X="constructor";id=Id_constructor; }
if (X!=null && X!=s && !X.equals(s)) id = 0;
break L0;
}
// #/generated#
return id;
}
// #/string_id_map#
protected void initPrototypeId(int id)
{
String s;
int arity;
switch (id) {
case Id_constructor: arity=2; s="constructor"; break;
case Id_toString: arity=0; s="toString"; break;
case Id_toSource: arity=0; s="toSource"; break;
default: throw new IllegalArgumentException(String.valueOf(id));
}
initPrototypeMethod(QNAME_TAG, id, s, arity);
}
public Object execIdCall(IdFunctionObject f,
Context cx,
Scriptable scope,
Scriptable thisObj,
Object[] args)
{
if (!f.hasTag(QNAME_TAG)) {
return super.execIdCall(f, cx, scope, thisObj, args);
}
int id = f.methodId();
switch (id) {
case Id_constructor:
return jsConstructor(cx, (thisObj == null), args);
case Id_toString:
return realThis(thisObj, f).toString();
case Id_toSource:
return realThis(thisObj, f).js_toSource();
}
throw new IllegalArgumentException(String.valueOf(id));
}
private QName realThis(Scriptable thisObj, IdFunctionObject f)
{
if(!(thisObj instanceof QName))
throw incompatibleCallError(f);
return (QName)thisObj;
}
QName newQName(XMLLibImpl lib, String q_uri, String q_localName, String q_prefix) {
QName prototype = this.prototype;
if (prototype == null) {
prototype = this;
}
XmlNode.Namespace ns = null;
if (q_prefix != null) {
ns = XmlNode.Namespace.create(q_uri, q_prefix);
} else if (q_uri != null) {
ns = XmlNode.Namespace.create(q_uri);
} else {
ns = null;
}
if (q_localName != null && q_localName.equals("*")) q_localName = null;
return create(lib, this.getParentScope(), prototype, XmlNode.QName.create(ns, q_localName));
}
// See ECMA357 13.3.2
QName constructQName(XMLLibImpl lib, Context cx, Object namespace, Object name) {
String nameString = null;
if (name instanceof QName) {
if (namespace == Undefined.instance) {
return (QName)name;
} else {
nameString = ((QName)name).localName();
}
}
if (name == Undefined.instance) {
nameString = "";
} else {
nameString = ScriptRuntime.toString(name);
}
if (namespace == Undefined.instance) {
if ("*".equals(nameString)) {
namespace = null;
} else {
namespace = lib.getDefaultNamespace(cx);
}
}
Namespace namespaceNamespace = null;
if (namespace == null) {
// leave as null
} else if (namespace instanceof Namespace) {
namespaceNamespace = (Namespace)namespace;
} else {
namespaceNamespace = lib.newNamespace(ScriptRuntime.toString(namespace));
}
String q_localName = nameString;
String q_uri;
String q_prefix;
if (namespace == null) {
q_uri = null;
q_prefix = null; // corresponds to undefined; see QName class
} else {
q_uri = namespaceNamespace.uri();
q_prefix = namespaceNamespace.prefix();
}
return newQName(lib, q_uri, q_localName, q_prefix);
}
QName constructQName(XMLLibImpl lib, Context cx, Object nameValue) {
return constructQName(lib, cx, Undefined.instance, nameValue);
}
QName castToQName(XMLLibImpl lib, Context cx, Object qnameValue) {
if (qnameValue instanceof QName) {
return (QName)qnameValue;
}
return constructQName(lib, cx, qnameValue);
}
private Object jsConstructor(Context cx, boolean inNewExpr, Object[] args) {
// See ECMA357 13.3.2
if (!inNewExpr && args.length == 1) {
return castToQName(lib, cx, args[0]);
}
if (args.length == 0) {
return constructQName(lib, cx, Undefined.instance);
} else if (args.length == 1) {
return constructQName(lib, cx, args[0]);
} else {
return constructQName(lib, cx, args[0], args[1]);
}
}
private String js_toSource() {
StringBuffer sb = new StringBuffer();
sb.append('(');
toSourceImpl(uri(), localName(), prefix(), sb);
sb.append(')');
return sb.toString();
}
private static void toSourceImpl(String uri, String localName, String prefix, StringBuffer sb) {
sb.append("new QName(");
if (uri == null && prefix == null) {
if (!"*".equals(localName)) {
sb.append("null, ");
}
} else {
Namespace.toSourceImpl(prefix, uri, sb);
sb.append(", ");
}
sb.append('\'');
sb.append(ScriptRuntime.escapeString(localName, '\''));
sb.append("')");
}
}