/* praat_objectMenus.cpp
 *
 * Copyright (C) 1992-2012,2013,2014,2015 Paul Boersma
 *
 * This program 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 of the License, or (at
 * your option) any later version.
 *
 * This program 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 this program; if not, write to the Free Software
 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 */

#include <ctype.h>
#include "praatP.h"
#include "praat_script.h"
#include "ScriptEditor.h"
#include "ButtonEditor.h"
#include "DataEditor.h"
#include "site.h"
#include "GraphicsP.h"
//#include <string>

#undef iam
#define iam iam_LOOP

#define EDITOR  theCurrentPraatObjects -> list [IOBJECT]. editors

/********** Callbacks of the fixed buttons. **********/

DIRECT2 (Remove) {
	WHERE_DOWN (SELECTED)
		praat_removeObject (IOBJECT);
	praat_show ();
END2 }

FORM (Rename, U"Rename object", U"Rename...") {
	LABEL (U"rename object", U"New name:")
	TEXTFIELD (U"newName", U"")
	OK2
{ int IOBJECT; WHERE (SELECTED) SET_STRING (U"newName", NAME) }
DO
	char32 *string = GET_STRING (U"newName");
	if (theCurrentPraatObjects -> totalSelection == 0)
		Melder_throw (U"Selection changed!\nNo object selected. Cannot rename.");
	if (theCurrentPraatObjects -> totalSelection > 1)
		Melder_throw (U"Selection changed!\nCannot rename more than one object at a time.");
	WHERE (SELECTED) break;
	praat_cleanUpName (string);   // this is allowed because "string" is local and dispensible
	#if 0
	std::u32string newFullName = std::u32string (Thing_className ((Thing) OBJECT)) + U" " + string;
	if (newFullName != std::u32string (FULL_NAME)) {
		Melder_free (FULL_NAME), FULL_NAME = Melder_dup_f (newFullName.c_str());
		praat_list_renameAndSelect (IOBJECT, (std::u32string (Melder_integer (ID)) + U". " + newFullName). c_str());
		for (int ieditor = 0; ieditor < praat_MAXNUM_EDITORS; ieditor ++)
			if (EDITOR [ieditor]) Thing_setName ((Thing) EDITOR [ieditor], newFullName.c_str());
		Thing_setName ((Thing) OBJECT, string);
	}
	#else
	static MelderString fullName { 0 };
	MelderString_copy (& fullName, Thing_className ((Thing) OBJECT), U" ", string);
	if (! str32equ (fullName.string, FULL_NAME)) {
		Melder_free (FULL_NAME), FULL_NAME = Melder_dup_f (fullName.string);
		autoMelderString listName;
		MelderString_append (& listName, ID, U". ", fullName.string);
		praat_list_renameAndSelect (IOBJECT, listName.string);
		for (int ieditor = 0; ieditor < praat_MAXNUM_EDITORS; ieditor ++)
			if (EDITOR [ieditor]) Thing_setName ((Thing) EDITOR [ieditor], fullName.string);
		Thing_setName ((Thing) OBJECT, string);
	}
	#endif
END2 }

FORM (Copy, U"Copy object", U"Copy...") {
	LABEL (U"copy object", U"Name of new object:")
	TEXTFIELD (U"newName", U"")
	OK2
{ int IOBJECT; WHERE (SELECTED) SET_STRING (U"newName", NAME) }
DO
	if (theCurrentPraatObjects -> totalSelection == 0)
		Melder_throw (U"Selection changed!\nNo object selected. Cannot copy.");
	if (theCurrentPraatObjects -> totalSelection > 1)
		Melder_throw (U"Selection changed!\nCannot copy more than one object at a time.");
	WHERE (SELECTED) {
		char32 *name = GET_STRING (U"newName");
		praat_new (Data_copy ((Data) OBJECT), name);
	}
END2 }

DIRECT2 (Info) {
	if (theCurrentPraatObjects -> totalSelection == 0)
		Melder_throw (U"Selection changed!\nNo object selected. Cannot query.");
	if (theCurrentPraatObjects -> totalSelection > 1)
		Melder_throw (U"Selection changed!\nCannot query more than one object at a time.");
	WHERE (SELECTED) Thing_infoWithIdAndFile ((Thing) OBJECT, ID, & theCurrentPraatObjects -> list [IOBJECT]. file);
END2 }

DIRECT2 (Inspect) {
	if (theCurrentPraatObjects -> totalSelection == 0)
		Melder_throw (U"Selection changed!\nNo object selected. Cannot inspect.");
	if (theCurrentPraatApplication -> batch) {
		Melder_throw (U"Cannot inspect data from batch.");
	} else {
		WHERE (SELECTED) {
			praat_installEditor (DataEditor_create (ID_AND_FULL_NAME, OBJECT), IOBJECT);
		}
	}
END2 }

/********** The fixed menus. **********/

static GuiMenu praatMenu, editMenu, windowMenu, newMenu, readMenu, goodiesMenu, preferencesMenu, technicalMenu, applicationHelpMenu, helpMenu;

GuiMenu praat_objects_resolveMenu (const char32 *menu) {
	return
		str32equ (menu, U"Praat") || str32equ (menu, U"Control") ? praatMenu :
		#if cocoa
			str32equ (menu, U"Edit") ? editMenu :
			str32equ (menu, U"Window") ? windowMenu :
		#endif
		str32equ (menu, U"New") || str32equ (menu, U"Create") ? newMenu :
		str32equ (menu, U"Open") || str32equ (menu, U"Read") ? readMenu :
		str32equ (menu, U"Help") ? helpMenu :
		str32equ (menu, U"Goodies") ? goodiesMenu :
		str32equ (menu, U"Preferences") ? preferencesMenu :
		str32equ (menu, U"Technical") ? technicalMenu :
		#ifdef macintosh
			str32equ (menu, U"ApplicationHelp") ? applicationHelpMenu :
		#else
			str32equ (menu, U"ApplicationHelp") ? helpMenu :
		#endif
		newMenu;   /* Default. */
}

/********** Callbacks of the Praat menu. **********/

DIRECT2 (About) {
	praat_showLogo (FALSE);
END2 }

DIRECT2 (praat_newScript) {
	autoScriptEditor editor = ScriptEditor_createFromText (NULL, NULL);
	editor.transfer();   // the user becomes the owner
END2 }

DIRECT2 (praat_openScript) {
	autoScriptEditor editor = ScriptEditor_createFromText (NULL, NULL);
	TextEditor_showOpen (editor.peek());
	editor.transfer();   // the user becomes the owner
END2 }

static ButtonEditor theButtonEditor;

static void cb_ButtonEditor_destruction (Editor editor, void *closure) {
	(void) editor;
	(void) closure;
	theButtonEditor = NULL;
}

DIRECT2 (praat_editButtons) {
	if (theButtonEditor) {
		Editor_raise (theButtonEditor);
	} else {
		theButtonEditor = ButtonEditor_create ();
		Editor_setDestructionCallback (theButtonEditor, cb_ButtonEditor_destruction, NULL);
	}
END2 }

FORM (praat_addMenuCommand, U"Add menu command", U"Add menu command...") {
	WORD (U"Window", U"Objects")
	WORD (U"Menu", U"New")
	SENTENCE (U"Command", U"Hallo...")
	SENTENCE (U"After command", U"")
	INTEGER (U"Depth", U"0")
	LABEL (U"", U"Script file:")
	TEXTFIELD (U"Script", U"/u/miep/hallo.praat")
	OK2
DO
	praat_addMenuCommandScript (GET_STRING (U"Window"), GET_STRING (U"Menu"),
		GET_STRING (U"Command"), GET_STRING (U"After command"),
		GET_INTEGER (U"Depth"), GET_STRING (U"Script"));
END2 }

FORM (praat_hideMenuCommand, U"Hide menu command", U"Hide menu command...") {
	WORD (U"Window", U"Objects")
	WORD (U"Menu", U"New")
	SENTENCE (U"Command", U"Hallo...")
	OK2
DO
	praat_hideMenuCommand (GET_STRING (U"Window"), GET_STRING (U"Menu"), GET_STRING (U"Command"));
END2 }

FORM (praat_showMenuCommand, U"Show menu command", U"Show menu command...") {
	WORD (U"Window", U"Objects")
	WORD (U"Menu", U"New")
	SENTENCE (U"Command", U"Hallo...")
	OK2
DO
	praat_showMenuCommand (GET_STRING (U"Window"), GET_STRING (U"Menu"), GET_STRING (U"Command"));
END2 }

FORM (praat_addAction, U"Add action command", U"Add action command...") {
	WORD (U"Class 1", U"Sound")
	INTEGER (U"Number 1", U"0")
	WORD (U"Class 2", U"")
	INTEGER (U"Number 2", U"0")
	WORD (U"Class 3", U"")
	INTEGER (U"Number 3", U"0")
	SENTENCE (U"Command", U"Play reverse")
	SENTENCE (U"After command", U"Play")
	INTEGER (U"Depth", U"0")
	LABEL (U"", U"Script file:")
	TEXTFIELD (U"Script", U"/u/miep/playReverse.praat")
	OK2
DO
	praat_addActionScript (GET_STRING (U"Class 1"), GET_INTEGER (U"Number 1"),
		GET_STRING (U"Class 2"), GET_INTEGER (U"Number 2"), GET_STRING (U"Class 3"),
		GET_INTEGER (U"Number 3"), GET_STRING (U"Command"), GET_STRING (U"After command"),
		GET_INTEGER (U"Depth"), GET_STRING (U"Script"));
END2 }

FORM (praat_hideAction, U"Hide action command", U"Hide action command...") {
	WORD (U"Class 1", U"Sound")
	WORD (U"Class 2", U"")
	WORD (U"Class 3", U"")
	SENTENCE (U"Command", U"Play")
	OK2
DO
	praat_hideAction_classNames (GET_STRING (U"Class 1"), GET_STRING (U"Class 2"), GET_STRING (U"Class 3"), GET_STRING (U"Command"));
END2 }

FORM (praat_showAction, U"Show action command", U"Show action command...") {
	WORD (U"Class 1", U"Sound")
	WORD (U"Class 2", U"")
	WORD (U"Class 3", U"")
	SENTENCE (U"Command", U"Play")
	OK2
DO
	praat_showAction_classNames (GET_STRING (U"Class 1"), GET_STRING (U"Class 2"), GET_STRING (U"Class 3"), GET_STRING (U"Command"));
END2 }

/********** Callbacks of the Preferences menu. **********/

FORM (TextInputEncodingSettings, U"Text reading preferences", U"Unicode") {
	RADIO_ENUM (U"Encoding of 8-bit text files", kMelder_textInputEncoding, DEFAULT)
	OK2
SET_ENUM (U"Encoding of 8-bit text files", kMelder_textInputEncoding, Melder_getInputEncoding ())
DO
	Melder_setInputEncoding (GET_ENUM (kMelder_textInputEncoding, U"Encoding of 8-bit text files"));
END2 }

FORM (TextOutputEncodingSettings, U"Text writing preferences", U"Unicode") {
	RADIO_ENUM (U"Output encoding", kMelder_textOutputEncoding, DEFAULT)
	OK2
SET_ENUM (U"Output encoding", kMelder_textOutputEncoding, Melder_getOutputEncoding ())
DO
	Melder_setOutputEncoding (GET_ENUM (kMelder_textOutputEncoding, U"Output encoding"));
END2 }

FORM (GraphicsCjkFontStyleSettings, U"CJK font style preferences", 0) {
	OPTIONMENU_ENUM (U"CJK font style", kGraphics_cjkFontStyle, DEFAULT)
	OK2
SET_ENUM (U"CJK font style", kGraphics_cjkFontStyle, theGraphicsCjkFontStyle)
DO
	theGraphicsCjkFontStyle = GET_ENUM (kGraphics_cjkFontStyle, U"CJK font style");
END2 }


/********** Callbacks of the Goodies menu. **********/

FORM (praat_calculator, U"Calculator", U"Calculator") {
	LABEL (U"", U"Type any numeric formula or string formula:")
	TEXTFIELD (U"expression", U"5*5")
	LABEL (U"", U"Note that you can include many special functions in your formula,")
	LABEL (U"", U"including statistical functions and acoustics-auditory conversions.")
	LABEL (U"", U"For details, click Help.")
	OK2
DO
	struct Formula_Result result;
	if (interpreter == NULL) {
		interpreter = Interpreter_create (NULL, NULL);
		try {
			Interpreter_anyExpression (interpreter, GET_STRING (U"expression"), & result);
			forget (interpreter);
		} catch (MelderError) {
			forget (interpreter);
			throw;
		}
	} else {
		Interpreter_anyExpression (interpreter, GET_STRING (U"expression"), & result);
	}
	switch (result. expressionType) {
		case kFormula_EXPRESSION_TYPE_NUMERIC: {
			Melder_information (result. result.numericResult);
		} break;
		case kFormula_EXPRESSION_TYPE_STRING: {
			Melder_information (result. result.stringResult);
			Melder_free (result. result.stringResult);
		} break;
		case kFormula_EXPRESSION_TYPE_NUMERIC_ARRAY: {
		}
	}
END2 }

FORM (praat_reportDifferenceOfTwoProportions, U"Report difference of two proportions", U"Difference of two proportions") {
	INTEGER (U"left Row 1", U"71")
	INTEGER (U"right Row 1", U"39")
	INTEGER (U"left Row 2", U"93")
	INTEGER (U"right Row 2", U"27")
	OK2
DO
	double a = GET_INTEGER (U"left Row 1"), b = GET_INTEGER (U"right Row 1");
	double c = GET_INTEGER (U"left Row 2"), d = GET_INTEGER (U"right Row 2");
	double n = a + b + c + d;
	double aexp, bexp, cexp, dexp, crossDifference, x2;
	REQUIRE (a >= 0 && b >= 0 && c >= 0 && d >= 0, U"Numbers must not be negative.")
	REQUIRE ((a > 0 || b > 0) && (c > 0 || d > 0), U"Row totals must be positive.")
	REQUIRE ((a > 0 || c > 0) && (b > 0 || d > 0), U"Column totals must be positive.")
	MelderInfo_open ();
	MelderInfo_writeLine (U"Observed row 1 =    ", lround (a), U"    ", lround (b));
	MelderInfo_writeLine (U"Observed row 2 =    ", lround (c), U"    ", lround (d));
	aexp = (a + b) * (a + c) / n;
	bexp = (a + b) * (b + d) / n;
	cexp = (a + c) * (c + d) / n;
	dexp = (b + d) * (c + d) / n;
	MelderInfo_writeLine (U"");
	MelderInfo_writeLine (U"Expected row 1 =    ", aexp, U"    ", bexp);
	MelderInfo_writeLine (U"Expected row 2 =    ", cexp, U"    ", dexp);
	/*
	 * Continuity correction:
	 * bring the observed numbers closer to the expected numbers by 0.5 (if possible).
	 */
	if (a < aexp) { a += 0.5; if (a > aexp) a = aexp; }
	else if (a > aexp) { a -= 0.5; if (a < aexp) a = aexp; }
	if (b < bexp) { b += 0.5; if (b > bexp) b = bexp; }
	else if (b > bexp) { b -= 0.5; if (b < bexp) b = bexp; }
	if (c < cexp) { c += 0.5; if (c > cexp) c = cexp; }
	else if (c > cexp) { c -= 0.5; if (c < cexp) c = cexp; }
	if (d < dexp) { d += 0.5; if (d > dexp) d = dexp; }
	else if (d > dexp) { d -= 0.5; if (d < dexp) d = dexp; }
	MelderInfo_writeLine (U"");
	MelderInfo_writeLine (U"Corrected observed row 1 =    ", a, U"    ", b);
	MelderInfo_writeLine (U"Corrected observed row 2 =    ", c, U"    ", d);
	
	n = a + b + c + d;
	crossDifference = a * d - b * c;
	x2 = n * crossDifference * crossDifference / (a + b) / (c + d) / (a + c) / (b + d);
	MelderInfo_writeLine (U"");
	MelderInfo_writeLine (U"Chi-square =    ", x2);
	MelderInfo_writeLine (U"Two-tailed p =    ", NUMchiSquareQ (x2, 1));
	MelderInfo_close ();
END2 }

/********** Callbacks of the Technical menu. **********/

FORM (praat_debug, U"Set debugging options", 0) {
	LABEL (U"", U"If you switch Tracing on, Praat will write lots of detailed ")
	LABEL (U"", U"information about what goes on in Praat")
	structMelderDir dir;
	Melder_getPrefDir (& dir);
	structMelderFile file;
	#ifdef UNIX
		MelderDir_getFile (& dir, U"tracing", & file);
	#else
		MelderDir_getFile (& dir, U"Tracing.txt", & file);
	#endif
	LABEL (U"", Melder_cat (U"to ", Melder_fileToPath (& file), U"."))
	BOOLEAN (U"Tracing", 0)
	LABEL (U"", U"Setting the following to anything other than zero")
	LABEL (U"", U"will alter the behaviour of Praat")
	LABEL (U"", U"in unpredictable ways.")
	INTEGER (U"Debug option", U"0")
	OK2
SET_INTEGER (U"Tracing", Melder_isTracing)
SET_INTEGER (U"Debug option", Melder_debug)
DO
	Melder_setTracing (GET_INTEGER (U"Tracing"));
	Melder_debug = GET_INTEGER (U"Debug option");
END2 }

DIRECT2 (praat_listReadableTypesOfObjects) {
	Thing_listReadableClasses ();
END2 }

DIRECT2 (praat_reportGraphicalProperties) {
	praat_reportGraphicalProperties ();
END2 }

DIRECT2 (praat_reportIntegerProperties) {
	praat_reportIntegerProperties ();
END2 }

DIRECT2 (praat_reportMemoryUse) {
	praat_reportMemoryUse ();
END2 }

DIRECT2 (praat_reportTextProperties) {
	praat_reportTextProperties ();
END2 }

/********** Callbacks of the Open menu. **********/

static void readFromFile (MelderFile file) {
	autoData object = (Data) Data_readFromFile (file);
	if (object.peek() == NULL) return;
	if (Thing_member (object.peek(), classManPages) && ! Melder_batch) {
		ManPages pages = (ManPages) object.peek();
		ManPage firstPage = static_cast<ManPage> (pages -> pages -> item [1]);
		Manual_create (firstPage -> title, object.transfer(), true);
		if (pages -> executable)
			Melder_warning (U"These manual pages contain links to executable scripts.\n"
				"Only navigate these pages if you trust their author!");
		return;
	}
	if (Thing_member (object.peek(), classScript) && ! Melder_batch) {
		ScriptEditor_createFromScript (NULL, (Script) object.peek());
		return;
	}
	praat_newWithFile (object.transfer(), file, MelderFile_name (file));
	praat_updateSelection ();
}

FORM_READ2 (Data_readFromFile, U"Read Object(s) from file", 0, true) {
	readFromFile (file);
END2 }

/********** Callbacks of the Save menu. **********/

FORM_WRITE2 (Data_writeToTextFile, U"Save Object(s) as one text file", 0, 0) {
	if (theCurrentPraatObjects -> totalSelection == 1) {
		LOOP {
			iam (Data);
			Data_writeToTextFile (me, file);
		}
	} else {
		autoCollection set = praat_getSelectedObjects ();
		Data_writeToTextFile (set.peek(), file);
	}
END2 }

FORM_WRITE2 (Data_writeToShortTextFile, U"Save Object(s) as one short text file", 0, 0) {
	if (theCurrentPraatObjects -> totalSelection == 1) {
		LOOP {
			iam (Data);
			Data_writeToShortTextFile (me, file);
		}
	} else {
		autoCollection set = praat_getSelectedObjects ();
		Data_writeToShortTextFile (set.peek(), file);
	}
END2 }

FORM_WRITE2 (Data_writeToBinaryFile, U"Save Object(s) as one binary file", 0, 0) {
	if (theCurrentPraatObjects -> totalSelection == 1) {
		LOOP {
			iam (Data);
			Data_writeToBinaryFile (me, file);
		}
	} else {
		autoCollection set = praat_getSelectedObjects ();
		Data_writeToBinaryFile (set.peek(), file);
	}
END2 }

FORM (ManPages_saveToHtmlDirectory, U"Save all pages as HTML files", 0) {
	LABEL (U"", U"Type a directory name:")
	TEXTFIELD (U"directory", U"")
	OK2
structMelderDir currentDirectory = { { 0 } };
Melder_getDefaultDir (& currentDirectory);
SET_STRING (U"directory", Melder_dirToPath (& currentDirectory))
DO
	char32 *directory = GET_STRING (U"directory");
	LOOP {
		iam (ManPages);
		ManPages_writeAllToHtmlDir (me, directory);
	}
END2 }

DIRECT2 (ManPages_view) {
	LOOP {
		iam (ManPages);
		ManPage firstPage = static_cast<ManPage> (my pages -> item [1]);
		autoManual manual = Manual_create (firstPage -> title, me, false);
		if (my executable)
			Melder_warning (U"These manual pages contain links to executable scripts.\n"
				"Only navigate these pages if you trust their author!");
		praat_installEditor (manual.transfer(), IOBJECT);
	}
END2 }

/********** Callbacks of the Help menu. **********/

FORM (SearchManual, U"Search manual", U"Manual") {
	LABEL (U"", U"Search for strings (separate with spaces):")
	TEXTFIELD (U"query", U"")
	OK2
DO
	if (theCurrentPraatApplication -> batch)
		Melder_throw (U"Cannot view a manual from batch.");
	Manual manPage = Manual_create (U"Intro", theCurrentPraatApplication -> manPages, false);
	Manual_search (manPage, GET_STRING (U"query"));
END2 }

FORM (GoToManualPage, U"Go to manual page", 0) {
	{long numberOfPages;
	const char32 **pages = ManPages_getTitles (theCurrentPraatApplication -> manPages, & numberOfPages);
	LIST (U"Page", numberOfPages, pages, 1)}
	OK2
DO
	if (theCurrentPraatApplication -> batch)
		Melder_throw (U"Cannot view a manual from batch.");
	Manual manPage = Manual_create (U"Intro", theCurrentPraatApplication -> manPages, false);
	HyperPage_goToPage_i (manPage, GET_INTEGER (U"Page"));
END2 }

FORM (WriteManualToHtmlDirectory, U"Save all pages as HTML files", 0) {
	LABEL (U"", U"Type a directory name:")
	TEXTFIELD (U"directory", U"")
	OK2
structMelderDir currentDirectory = { { 0 } };
Melder_getDefaultDir (& currentDirectory);
SET_STRING (U"directory", Melder_dirToPath (& currentDirectory))
DO
	char32 *directory = GET_STRING (U"directory");
	ManPages_writeAllToHtmlDir (theCurrentPraatApplication -> manPages, directory);
END2 }

/********** Menu descriptions. **********/

void praat_show (void) {
	/*
	 * (De)sensitivize the fixed buttons as appropriate for the current selection.
	 */
	praat_sensitivizeFixedButtonCommand (U"Remove", theCurrentPraatObjects -> totalSelection != 0);
	praat_sensitivizeFixedButtonCommand (U"Rename...", theCurrentPraatObjects -> totalSelection == 1);
	praat_sensitivizeFixedButtonCommand (U"Copy...", theCurrentPraatObjects -> totalSelection == 1);
	praat_sensitivizeFixedButtonCommand (U"Info", theCurrentPraatObjects -> totalSelection == 1);
	praat_sensitivizeFixedButtonCommand (U"Inspect", theCurrentPraatObjects -> totalSelection != 0);
	praat_actions_show ();
	if (theCurrentPraatApplication == & theForegroundPraatApplication && theButtonEditor)
		Editor_dataChanged (theButtonEditor);
}

/********** Menu descriptions. **********/

void praat_addFixedButtons (GuiWindow window) {
	praat_addFixedButtonCommand (window, U"Rename...", DO_Rename, 8, 70);
	praat_addFixedButtonCommand (window, U"Copy...", DO_Copy, 98, 70);
	praat_addFixedButtonCommand (window, U"Inspect", DO_Inspect, 8, 40);
	praat_addFixedButtonCommand (window, U"Info", DO_Info, 98, 40);
	praat_addFixedButtonCommand (window, U"Remove", DO_Remove, 8, 10);
}

static void searchProc (void) {
	DO_SearchManual (NULL, 0, NULL, NULL, NULL, NULL, false, NULL);
}

static MelderString itemTitle_about { 0 };

static Any scriptRecognizer (int nread, const char *header, MelderFile file) {
	const char32 *name = MelderFile_name (file);
	if (nread < 2) return NULL;
	if ((header [0] == '#' && header [1] == '!') || str32str (name, U".praat") == name + str32len (name) - 6
	    || str32str (name, U".html") == name + str32len (name) - 5)
	{
		return Script_createFromFile (file);
	}
	return NULL;
}

static void cb_openDocument (MelderFile file) {
	try {
		readFromFile (file);
	} catch (MelderError) {
		Melder_flushError ();
	}
}

#if cocoa
DIRECT2 (praat_cut) {
	[[[NSApp keyWindow] fieldEditor: YES forObject: nil] cut: nil];
END2 }
DIRECT2 (praat_copy) {
	[[[NSApp keyWindow] fieldEditor: YES forObject: nil] copy: nil];
END2 }
DIRECT2 (praat_paste) {
	[[[NSApp keyWindow] fieldEditor: YES forObject: nil] pasteAsPlainText: nil];
END2 }
DIRECT2 (praat_minimize) {
	[[NSApp keyWindow] performMiniaturize: nil];
END2 }
DIRECT2 (praat_zoom) {
	[[NSApp keyWindow] performZoom: nil];
END2 }
DIRECT2 (praat_close) {
	[[NSApp keyWindow] performClose: nil];
END2 }
#endif

void praat_addMenus (GuiWindow window) {
	Melder_setSearchProc (searchProc);

	Data_recognizeFileType (scriptRecognizer);

	/*
	 * Create the menu titles in the bar.
	 */
	if (! theCurrentPraatApplication -> batch) {
		#ifdef macintosh
			praatMenu = GuiMenu_createInWindow (NULL, U"\024", 0);
			#if cocoa
				editMenu = GuiMenu_createInWindow (NULL, U"Edit", 0);
				windowMenu = GuiMenu_createInWindow (NULL, U"Window", 0);
			#endif
		#else
			praatMenu = GuiMenu_createInWindow (window, U"Praat", 0);
		#endif
		newMenu = GuiMenu_createInWindow (window, U"New", 0);
		readMenu = GuiMenu_createInWindow (window, U"Open", 0);
		praat_actions_createWriteMenu (window);
		#ifdef macintosh
			applicationHelpMenu = GuiMenu_createInWindow (NULL, U"Help", 0);
		#endif
		helpMenu = GuiMenu_createInWindow (window, U"Help", 0);
	}
	
	MelderString_append (& itemTitle_about, U"About ", praatP.title, U"...");
	#ifdef macintosh
		praat_addMenuCommand (U"Objects", U"Praat", itemTitle_about.string, 0, praat_UNHIDABLE, DO_About);
		#if cocoa
			praat_addMenuCommand (U"Objects", U"Edit", U"Cut", 0, praat_UNHIDABLE + 'X', DO_praat_cut);
			praat_addMenuCommand (U"Objects", U"Edit", U"Copy", 0, praat_UNHIDABLE + 'C', DO_praat_copy);
			praat_addMenuCommand (U"Objects", U"Edit", U"Paste", 0, praat_UNHIDABLE + 'V', DO_praat_paste);
			praat_addMenuCommand (U"Objects", U"Window", U"Minimize", 0, praat_UNHIDABLE, DO_praat_minimize);
			praat_addMenuCommand (U"Objects", U"Window", U"Zoom", 0, praat_UNHIDABLE, DO_praat_zoom);
			praat_addMenuCommand (U"Objects", U"Window", U"Close", 0, 'W', DO_praat_close);
		#endif
	#endif
	#ifdef UNIX
		praat_addMenuCommand (U"Objects", U"Praat", itemTitle_about.string, 0, praat_UNHIDABLE, DO_About);
	#endif
	praat_addMenuCommand (U"Objects", U"Praat", U"-- script --", 0, 0, 0);
	praat_addMenuCommand (U"Objects", U"Praat", U"New Praat script", 0, 0, DO_praat_newScript);
	praat_addMenuCommand (U"Objects", U"Praat", U"Open Praat script...", 0, 0, DO_praat_openScript);
	praat_addMenuCommand (U"Objects", U"Praat", U"-- buttons --", 0, 0, 0);
	praat_addMenuCommand (U"Objects", U"Praat", U"Add menu command...", 0, praat_HIDDEN, DO_praat_addMenuCommand);
	praat_addMenuCommand (U"Objects", U"Praat", U"Hide menu command...", 0, praat_HIDDEN, DO_praat_hideMenuCommand);
	praat_addMenuCommand (U"Objects", U"Praat", U"Show menu command...", 0, praat_HIDDEN, DO_praat_showMenuCommand);
	praat_addMenuCommand (U"Objects", U"Praat", U"Add action command...", 0, praat_HIDDEN, DO_praat_addAction);
	praat_addMenuCommand (U"Objects", U"Praat", U"Hide action command...", 0, praat_HIDDEN, DO_praat_hideAction);
	praat_addMenuCommand (U"Objects", U"Praat", U"Show action command...", 0, praat_HIDDEN, DO_praat_showAction);

	GuiMenuItem menuItem = praat_addMenuCommand (U"Objects", U"Praat", U"Goodies", 0, praat_UNHIDABLE, 0);
	goodiesMenu = menuItem ? menuItem -> d_menu : NULL;
	praat_addMenuCommand (U"Objects", U"Goodies", U"Calculator...", 0, 'U', DO_praat_calculator);
	praat_addMenuCommand (U"Objects", U"Goodies", U"Report difference of two proportions...", 0, 0, DO_praat_reportDifferenceOfTwoProportions);

	menuItem = praat_addMenuCommand (U"Objects", U"Praat", U"Preferences", 0, praat_UNHIDABLE, 0);
	preferencesMenu = menuItem ? menuItem -> d_menu : NULL;
	praat_addMenuCommand (U"Objects", U"Preferences", U"Buttons...", 0, praat_UNHIDABLE, DO_praat_editButtons);   /* Cannot be hidden. */
	praat_addMenuCommand (U"Objects", U"Preferences", U"-- encoding prefs --", 0, 0, 0);
	praat_addMenuCommand (U"Objects", U"Preferences", U"Text reading preferences...", 0, 0, DO_TextInputEncodingSettings);
	praat_addMenuCommand (U"Objects", U"Preferences", U"Text writing preferences...", 0, 0, DO_TextOutputEncodingSettings);
	praat_addMenuCommand (U"Objects", U"Preferences", U"CJK font style preferences...", 0, 0, DO_GraphicsCjkFontStyleSettings);

	menuItem = praat_addMenuCommand (U"Objects", U"Praat", U"Technical", 0, praat_UNHIDABLE, 0);
	technicalMenu = menuItem ? menuItem -> d_menu : NULL;
	praat_addMenuCommand (U"Objects", U"Technical", U"List readable types of objects", 0, 0, DO_praat_listReadableTypesOfObjects);
	praat_addMenuCommand (U"Objects", U"Technical", U"Report memory use", 0, 0, DO_praat_reportMemoryUse);
	praat_addMenuCommand (U"Objects", U"Technical", U"Report integer properties", 0, 0, DO_praat_reportIntegerProperties);
	praat_addMenuCommand (U"Objects", U"Technical", U"Report text properties", 0, 0, DO_praat_reportTextProperties);
	praat_addMenuCommand (U"Objects", U"Technical", U"Report graphical properties", 0, 0, DO_praat_reportGraphicalProperties);
	praat_addMenuCommand (U"Objects", U"Technical", U"Debug...", 0, 0, DO_praat_debug);

	praat_addMenuCommand (U"Objects", U"Open", U"Read from file...", 0, praat_ATTRACTIVE + 'O', DO_Data_readFromFile);

	praat_addAction1 (classData, 0, U"Save as text file...", 0, 0, DO_Data_writeToTextFile);
	praat_addAction1 (classData, 0, U"Write to text file...", 0, praat_HIDDEN, DO_Data_writeToTextFile);
	praat_addAction1 (classData, 0, U"Save as short text file...", 0, 0, DO_Data_writeToShortTextFile);
	praat_addAction1 (classData, 0, U"Write to short text file...", 0, praat_HIDDEN, DO_Data_writeToShortTextFile);
	praat_addAction1 (classData, 0, U"Save as binary file...", 0, 0, DO_Data_writeToBinaryFile);
	praat_addAction1 (classData, 0, U"Write to binary file...", 0, praat_HIDDEN, DO_Data_writeToBinaryFile);

	praat_addAction1 (classManPages, 1, U"Save to HTML directory...", 0, 0, DO_ManPages_saveToHtmlDirectory);
	praat_addAction1 (classManPages, 1, U"View", 0, 0, DO_ManPages_view);
}

void praat_addMenus2 (void) {
	praat_addMenuCommand (U"Objects", U"ApplicationHelp", U"-- manual --", 0, 0, 0);
	praat_addMenuCommand (U"Objects", U"ApplicationHelp", U"Go to manual page...", 0, 0, DO_GoToManualPage);
	praat_addMenuCommand (U"Objects", U"ApplicationHelp", U"Write manual to HTML directory...", 0, praat_HIDDEN, DO_WriteManualToHtmlDirectory);
	praat_addMenuCommand (U"Objects", U"ApplicationHelp",
		Melder_cat (U"Search ", praatP.title, U" manual..."),
		0, 'M', DO_SearchManual);
	#ifdef _WIN32
		praat_addMenuCommand (U"Objects", U"Help", U"-- about --", 0, 0, 0);
		praat_addMenuCommand (U"Objects", U"Help", itemTitle_about.string, 0, praat_UNHIDABLE, DO_About);
	#endif

	#if defined (macintosh) || defined (_WIN32)
		Gui_setOpenDocumentCallback (cb_openDocument);
	#endif
}

/* End of file praat_objectMenus.cpp */
