#region ================== Copyright (c) 2022 Boris Iwanski /* * 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 3 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, see. */ #endregion #region ================== Namespaces using System; using System.Collections.Generic; using System.Dynamic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; using Jint; using Jint.Runtime.Interop; #endregion namespace CodeImp.DoomBuilder.UDBScript.Wrapper { internal class UDBWrapper { #region ================== Variables private GameConfigurationWrapper gameconfiguration; private TypeReference queryoptions; private ExpandoObject scriptoptions; private TypeReference angle2d; private TypeReference data; private TypeReference line2d; private MapWrapper map; private TypeReference univalue; private TypeReference vector2d; private TypeReference vector3d; private TypeReference linedef; private TypeReference sector; private TypeReference sidedef; private TypeReference thing; private TypeReference vertex; private IProgress progress; private IProgress status; private IProgress logger; #endregion #region ================== Properties /// /// Class containing methods related to the game configuration. See [GameConfiguration](GameConfiguration.md) for more information. /// public GameConfigurationWrapper GameConfiguration { get { return gameconfiguration; } } /// /// Class containing methods and properties related to querying options from the user at runtime. See [QueryOptions](QueryOptions.md) for more information. /// public TypeReference QueryOptions { get { return queryoptions; } } /// /// Object containing the script options. See [Setting script options](gettingstarted.md#setting-script-options). /// public ExpandoObject ScriptOptions { get { return scriptoptions; } } /// /// Class containing methods related to angles. See [Angle2D](Angle2D.md) for more information. /// ```js /// let rad = UDB.Angle2D.degToRad(46); /// ``` /// public TypeReference Angle2D { get { return angle2d; } } /// /// Class containing methods related to the game data. See [Data](Data.md) for more information. /// ```js /// let hasfireblu = UDB.Data.textureExists('FIREBLU1'); /// ``` /// public TypeReference Data { get { return data; } } /// /// Instantiable class that contains methods related to two-dimensional lines. See [Line2D](Line2D.md) for more information. /// ```js /// let line = new UDB.Line2D([ 32, 64 ], [ 96, 128 ]); /// ``` /// public TypeReference Line2D { get { return line2d; } } /// /// Object containing methods related to the map. See [Map](Map.md) for more information. /// ```js /// let sectors = UDB.Map.getSelectedOrHighlightedSectors(); /// ``` /// public MapWrapper Map { get { return map; } } /// /// The `UniValue` class. Is only needed when trying to assign integer values to UDMF fields. /// ```js /// s.fields.user_myintfield = new UDB.UniValue(0, 25); /// ``` /// public TypeReference UniValue { get { return univalue; } } /// /// Instantiable class that contains methods related to two-dimensional vectors. See [Vector2D](Vector2D.md) for more information. /// ```js /// let v = new UDB.Vector2D(32, 64); /// ``` /// public TypeReference Vector2D { get { return vector2d; } } /// /// Instantiable class that contains methods related to three-dimensional vectors. See [Vector3D](Vector3D.md) for more information. /// /// ```js /// let v = new UDB.Vector3D(32, 64, 128); /// ``` public TypeReference Vector3D { get { return vector3d; } } public TypeReference Linedef { get { return linedef; } } public TypeReference Sector { get { return sector; } } public TypeReference Sidedef { get { return sidedef; } } public TypeReference Thing { get { return thing; } } public TypeReference Vertex { get { return vertex; } } #endregion #region ================== Constructors internal UDBWrapper(Engine engine, ScriptInfo scriptinfo, IProgress progress, IProgress status, IProgress logger) { gameconfiguration = new GameConfigurationWrapper(); queryoptions = TypeReference.CreateTypeReference(engine, typeof(QueryOptions)); scriptoptions = scriptinfo.GetScriptOptionsObject(); angle2d = TypeReference.CreateTypeReference(engine, typeof(Angle2DWrapper)); data = TypeReference.CreateTypeReference(engine, typeof(DataWrapper)); line2d = TypeReference.CreateTypeReference(engine, typeof(Line2DWrapper)); map = new MapWrapper(); univalue = TypeReference.CreateTypeReference(engine, typeof(CodeImp.DoomBuilder.Map.UniValue)); vector2d = TypeReference.CreateTypeReference(engine, typeof(Vector2DWrapper)); vector3d = TypeReference.CreateTypeReference(engine, typeof(Vector3DWrapper)); // These can not be directly instanciated and don't have static method, but it's required to // for example use "instanceof" in scripts linedef = TypeReference.CreateTypeReference(engine, typeof(LinedefWrapper)); sector = TypeReference.CreateTypeReference(engine, typeof(SectorWrapper)); sidedef = TypeReference.CreateTypeReference(engine, typeof(SidedefWrapper)); thing = TypeReference.CreateTypeReference(engine, typeof(ThingWrapper)); vertex = TypeReference.CreateTypeReference(engine, typeof(VertexWrapper)); this.progress = progress; this.status = status; this.logger = logger; } #endregion #region ================== Methods /// /// Set the progress of the script in percent. Value can be between 0 and 100. Also shows the script running dialog. /// /// Number between 0 and 100 public void setProgress(int value) { progress.Report(value); } /* public void setStatus(string text) { status.Report(text); } */ /// /// Adds a line to the script log. Also shows the script running dialog. /// /// Line to add to the script log public void log(object text) { if (text == null) return; logger.Report(text.ToString()); } /// /// Shows a message box with an "OK" button. /// /// Message to show public void showMessage(object message) { BuilderPlug.Me.ScriptRunnerForm.InvokePaused(new Action(() => { if (message == null) message = string.Empty; MessageForm mf = new MessageForm("OK", null, message.ToString()); DialogResult result = mf.ShowDialog(); if (result == DialogResult.Abort) throw new UserScriptAbortException(); })); } /// /// Shows a message box with an "Yes" and "No" button. /// /// Message to show /// true if "Yes" was clicked, false if "No" was clicked public bool showMessageYesNo(object message) { return (bool)BuilderPlug.Me.ScriptRunnerForm.InvokePaused(new Func(() => { if (message == null) message = string.Empty; MessageForm mf = new MessageForm("Yes", "No", message.ToString()); DialogResult result = mf.ShowDialog(); if (result == DialogResult.Abort) throw new UserScriptAbortException(); return result == DialogResult.OK; })); } /// /// Exist the script prematurely without undoing its changes. /// /// Text to show in the status bar (optional) public void exit(string s=null) { if (string.IsNullOrEmpty(s)) throw new ExitScriptException(); throw new ExitScriptException(s); } /// /// Exist the script prematurely with undoing its changes. /// /// Text to show in the status bar (optional) public void die(string s=null) { if (string.IsNullOrEmpty(s)) throw new DieScriptException(); throw new DieScriptException(s); } #endregion } }