/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * 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 Mozilla Communicator client code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * David Hyatt (Original Author) * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsIAtom.h" #include "nsString.h" #include "jsapi.h" #include "nsIContent.h" #include "nsString.h" #include "nsUnicharUtils.h" #include "nsReadableUtils.h" #include "mozilla/FunctionTimer.h" #include "nsXBLProtoImplField.h" #include "nsIScriptContext.h" #include "nsContentUtils.h" #include "nsIURI.h" #include "nsXBLSerialize.h" #include "nsXBLPrototypeBinding.h" nsXBLProtoImplField::nsXBLProtoImplField(const PRUnichar* aName, const PRUnichar* aReadOnly) : mNext(nsnull), mFieldText(nsnull), mFieldTextLength(0), mLineNumber(0) { MOZ_COUNT_CTOR(nsXBLProtoImplField); mName = NS_strdup(aName); // XXXbz make more sense to use a stringbuffer? mJSAttributes = JSPROP_ENUMERATE; if (aReadOnly) { nsAutoString readOnly; readOnly.Assign(aReadOnly); if (readOnly.LowerCaseEqualsLiteral("true")) mJSAttributes |= JSPROP_READONLY; } } nsXBLProtoImplField::nsXBLProtoImplField(const bool aIsReadOnly) : mNext(nsnull), mFieldText(nsnull), mFieldTextLength(0), mLineNumber(0) { MOZ_COUNT_CTOR(nsXBLProtoImplField); mJSAttributes = JSPROP_ENUMERATE; if (aIsReadOnly) mJSAttributes |= JSPROP_READONLY; } nsXBLProtoImplField::~nsXBLProtoImplField() { MOZ_COUNT_DTOR(nsXBLProtoImplField); if (mFieldText) nsMemory::Free(mFieldText); NS_Free(mName); NS_CONTENT_DELETE_LIST_MEMBER(nsXBLProtoImplField, this, mNext); } void nsXBLProtoImplField::AppendFieldText(const nsAString& aText) { if (mFieldText) { nsDependentString fieldTextStr(mFieldText, mFieldTextLength); nsAutoString newFieldText = fieldTextStr + aText; PRUnichar* temp = mFieldText; mFieldText = ToNewUnicode(newFieldText); mFieldTextLength = newFieldText.Length(); nsMemory::Free(temp); } else { mFieldText = ToNewUnicode(aText); mFieldTextLength = aText.Length(); } } nsresult nsXBLProtoImplField::InstallField(nsIScriptContext* aContext, JSObject* aBoundNode, nsIPrincipal* aPrincipal, nsIURI* aBindingDocURI, bool* aDidInstall) const { NS_TIME_FUNCTION_MIN(5); NS_PRECONDITION(aBoundNode, "uh-oh, bound node should NOT be null or bad things will " "happen"); *aDidInstall = false; if (mFieldTextLength == 0) { return NS_OK; } // EvaluateStringWithValue and JS_DefineUCProperty can both trigger GC, so // protect |result| here. nsresult rv; nsCAutoString uriSpec; aBindingDocURI->GetSpec(uriSpec); JSContext* cx = aContext->GetNativeContext(); NS_ASSERTION(!::JS_IsExceptionPending(cx), "Shouldn't get here when an exception is pending!"); // compile the literal string bool undefined; nsCOMPtr context = aContext; JSAutoRequest ar(cx); jsval result = JSVAL_NULL; rv = context->EvaluateStringWithValue(nsDependentString(mFieldText, mFieldTextLength), aBoundNode, aPrincipal, uriSpec.get(), mLineNumber, JSVERSION_LATEST, &result, &undefined); if (NS_FAILED(rv)) { return rv; } if (undefined) { result = JSVAL_VOID; } // Define the evaluated result as a JS property nsDependentString name(mName); if (!::JS_DefineUCProperty(cx, aBoundNode, reinterpret_cast(mName), name.Length(), result, nsnull, nsnull, mJSAttributes)) { return NS_ERROR_OUT_OF_MEMORY; } *aDidInstall = true; return NS_OK; } nsresult nsXBLProtoImplField::Read(nsIScriptContext* aContext, nsIObjectInputStream* aStream) { nsAutoString name; nsresult rv = aStream->ReadString(name); NS_ENSURE_SUCCESS(rv, rv); mName = ToNewUnicode(name); rv = aStream->Read32(&mLineNumber); NS_ENSURE_SUCCESS(rv, rv); nsAutoString fieldText; rv = aStream->ReadString(fieldText); NS_ENSURE_SUCCESS(rv, rv); mFieldTextLength = fieldText.Length(); if (mFieldTextLength) mFieldText = ToNewUnicode(fieldText); return NS_OK; } nsresult nsXBLProtoImplField::Write(nsIScriptContext* aContext, nsIObjectOutputStream* aStream) { XBLBindingSerializeDetails type = XBLBinding_Serialize_Field; if (mJSAttributes & JSPROP_READONLY) { type |= XBLBinding_Serialize_ReadOnly; } nsresult rv = aStream->Write8(type); NS_ENSURE_SUCCESS(rv, rv); rv = aStream->WriteWStringZ(mName); NS_ENSURE_SUCCESS(rv, rv); rv = aStream->Write32(mLineNumber); NS_ENSURE_SUCCESS(rv, rv); return aStream->WriteWStringZ(mFieldText ? mFieldText : EmptyString().get()); }