blob: 4969068a8c6a3c292e293885fc0c96c34cf9e123 [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.openide.text;
import java.beans.PropertyChangeListener;
import java.beans.VetoableChangeListener;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import javax.swing.text.BadLocationException;
import javax.swing.text.EditorKit;
import javax.swing.text.StyledDocument;
import org.netbeans.junit.NbTestCase;
import org.openide.awt.UndoRedo;
import org.openide.util.Exceptions;
import org.openide.util.Lookup;
import org.openide.util.Task;
import org.openide.windows.CloneableOpenSupport;
import org.openide.windows.CloneableTopComponent;
/**
* Test that UndoRedo methods whcih return string implemented in CloneableEditorSupport.CESUndoRedoManager
* are not blocked by loading document. If document is not ready these methods just return empty string.
*
* Issue #143143
*
* @author Marek Slama
*/
public class CloneableEditor143143Test extends NbTestCase
implements CloneableEditorSupport.Env {
static {
System.setProperty("org.openide.windows.DummyWindowManager.VISIBLE", "false");
}
/** the support to work with */
private transient CES support;
// Env variables
private transient String content = "";
private transient boolean valid = true;
private transient boolean modified = false;
/** if not null contains message why this document cannot be modified */
private transient String cannotBeModified;
private transient Date date = new Date ();
private transient List/*<PropertyChangeListener>*/ propL = new ArrayList ();
private transient VetoableChangeListener vetoL;
private static CloneableEditor143143Test RUNNING;
private static boolean isWaiting = false;
private static final Object LOCK = new Object();
public CloneableEditor143143Test(String s) {
super(s);
}
protected void setUp () {
support = new CES (this, Lookup.EMPTY);
RUNNING = this;
}
protected boolean runInEQ() {
return true;
}
@Override
protected int timeOut() {
return 15000;
}
private Object writeReplace () {
return new Replace ();
}
public void testBlockGetDocument () throws Exception {
//Start asynchronous loading, it contains wait so document
//loading is blocked till notifyAll below.
synchronized (LOCK) {
isWaiting = false;
}
Task prepare = support.prepareDocument();
while (true) {
Thread.sleep(200);
synchronized (LOCK) {
if (isWaiting) {
break;
}
}
}
UndoRedo u = support.getUndoRedo();
//Check that following methods will return even if document is being loaded
u.getUndoPresentationName();
u.getRedoPresentationName();
//Finish document loading cleanly
synchronized (LOCK) {
LOCK.notifyAll();
}
prepare.waitFinished();
}
//
// Implementation of the CloneableEditorSupport.Env
//
public synchronized void addPropertyChangeListener(PropertyChangeListener l) {
propL.add (l);
}
public synchronized void removePropertyChangeListener(PropertyChangeListener l) {
propL.remove (l);
}
public synchronized void addVetoableChangeListener(VetoableChangeListener l) {
assertNull ("This is the first veto listener", vetoL);
vetoL = l;
}
public void removeVetoableChangeListener(VetoableChangeListener l) {
assertEquals ("Removing the right veto one", vetoL, l);
vetoL = null;
}
public CloneableOpenSupport findCloneableOpenSupport() {
return RUNNING.support;
}
public String getMimeType() {
return "text/plain";
}
public Date getTime() {
return date;
}
public InputStream inputStream() throws IOException {
return new ByteArrayInputStream (content.getBytes ());
}
public OutputStream outputStream() throws IOException {
class ContentStream extends ByteArrayOutputStream {
public void close () throws IOException {
super.close ();
content = new String (toByteArray ());
}
}
return new ContentStream ();
}
public boolean isValid() {
return valid;
}
public boolean isModified() {
return modified;
}
public void markModified() throws IOException {
if (cannotBeModified != null) {
final String notify = cannotBeModified;
IOException e = new IOException () {
public String getLocalizedMessage () {
return notify;
}
};
Exceptions.attachLocalizedMessage(e, cannotBeModified);
throw e;
}
modified = true;
}
public void unmarkModified() {
modified = false;
}
/** Implementation of the CES */
private static final class CES extends CloneableEditorSupport {
public CES (Env env, Lookup l) {
super (env, l);
}
public CloneableTopComponent.Ref getRef () {
return allEditors;
}
protected String messageName() {
return "Name";
}
protected String messageOpened() {
return "Opened";
}
protected String messageOpening() {
return "Opening";
}
protected String messageSave() {
return "Save";
}
protected String messageToolTip() {
return "ToolTip";
}
protected void loadFromStreamToKit(StyledDocument doc, InputStream stream, EditorKit kit)
throws IOException, BadLocationException {
synchronized (LOCK) {
isWaiting = true;
try {
LOCK.wait();
} catch (InterruptedException ex) {
}
}
super.loadFromStreamToKit(doc, stream, kit);
}
}
private static final class Replace implements Serializable {
public Object readResolve () {
return RUNNING;
}
}
}