/* GnomeCharacterData.java -
   Copyright (C) 2004 Free Software Foundation, Inc.

This file is part of GNU Classpath.

GNU Classpath is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.

GNU Classpath is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
General Public License for more details.

You should have received a copy of the GNU General Public License
along with GNU Classpath; see the file COPYING.  If not, write to the
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301 USA.

Linking this library statically or dynamically with other modules is
making a combined work based on this library.  Thus, the terms and
conditions of the GNU General Public License cover the whole
combination.

As a special exception, the copyright holders of this library give you
permission to link this library with independent modules to produce an
executable, regardless of the license terms of these independent
modules, and to copy and distribute the resulting executable under
terms of your choice, provided that you also meet, for each linked
independent module, the terms and conditions of the license of that
module.  An independent module is a module which is not derived from
or based on this library.  If you modify this library, you may extend
this exception to your version of the library, but you are not
obligated to do so.  If you do not wish to do so, delete this
exception statement from your version. */

package gnu.xml.libxmlj.dom;

import gnu.java.lang.CPStringBuilder;

import org.w3c.dom.CharacterData;
import org.w3c.dom.DOMException;

/**
 * A DOM character data node implemented in libxml2.
 *
 * @author <a href='mailto:dog@gnu.org'>Chris Burdess</a>
 */
abstract class GnomeCharacterData
  extends GnomeNode
  implements CharacterData
{

  GnomeCharacterData(Object id)
  {
    super(id);
  }

  public String getData()
    throws DOMException
  {
    return getNodeValue();
  }

  public void setData(String data)
    throws DOMException
  {
    setNodeValue(data);
  }

  public int getLength()
  {
    return getData().length();
  }

  public String substringData(int offset, int count)
    throws DOMException
  {
    return getData().substring(offset, offset + count);
  }

  public void appendData(String arg)
    throws DOMException
  {
    setData(getData() + arg);
  }

  public void insertData(int offset, String arg)
    throws DOMException
  {
    String data = getData();
    setData(data.substring(0, offset) + arg + data.substring(offset));
  }

  public void deleteData(int offset, int count)
    throws DOMException
  {
    String data = getData();
    setData(data.substring(0, offset) + data.substring(offset + count));
  }

  public void replaceData(int offset, int count, String arg)
  {
    String data = getData();
    setData(data.substring(0, offset) + arg +
            data.substring(offset + count));
  }

  public String toString()
  {
    CPStringBuilder buffer = new CPStringBuilder(getClass().getName());
    buffer.append("[data=");
    buffer.append(getData());
    buffer.append("]");
    return buffer.toString();
  }

}
