2 // Radegast Metaverse Client
3 // Copyright (c) 2009-2012, Radegast Development Team
4 // All rights reserved.
6 // Redistribution and use in source and binary forms, with or without
7 // modification, are permitted provided that the following conditions are met:
9 // * Redistributions of source code must retain the above copyright notice,
10 // this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above copyright
12 // notice, this list of conditions and the following disclaimer in the
13 // documentation and/or other materials provided with the distribution.
14 // * Neither the name of the application "Radegast", nor the names of its
15 // contributors may be used to endorse or promote products derived from
16 // this software without specific prior written permission.
18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19 // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20 // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
22 // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
24 // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
25 // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
26 // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 using System.Collections.Generic;
35 using System.Text.RegularExpressions;
37 using System.Threading;
38 using System.Windows.Forms;
39 using System.Resources;
42 using Radegast.Netcom;
44 using OpenMetaverse.StructuredData;
45 using OpenMetaverse.Assets;
49 public partial class frmMain : RadegastForm
51 #region Public members
52 public static ImageList ResourceImages = new ImageList();
53 public static List<string> ImageNames = new List<string>();
54 public bool PreventParcelUpdate = false;
55 public delegate void ProfileHandlerDelegate(string agentName, UUID agentID);
56 public ProfileHandlerDelegate ShowAgentProfile;
58 public TabsConsole TabConsole
60 get { return tabsConsole; }
63 public MapConsole WorldMap
69 return (MapConsole)MapTab.Control;
75 public RadegastTab MapTab
79 if (tabsConsole.TabExists("map"))
81 return tabsConsole.Tabs["map"];
90 public MediaConsole MediaConsole { get { return mediaConsole; } }
93 /// Drop down that contains the tools menu
95 public ToolStripDropDownButton ToolsMenu
97 get { return tbnTools; }
101 /// Dropdown that contains the heelp menu
103 public ToolStripDropDownButton HelpMenu
105 get { return tbtnHelp; }
109 /// Drop down that contants the plugins menu. Make sure to set it Visible if
110 /// you add items to this menu, it's hidden by default
112 public ToolStripDropDownButton PluginsMenu
114 get { return tbnPlugins; }
119 #region Private members
120 private RadegastInstance instance;
121 private GridClient client { get { return instance.Client; } }
122 private RadegastNetcom netcom { get { return instance.Netcom; } }
123 private TabsConsole tabsConsole;
124 private System.Timers.Timer statusTimer;
125 private AutoPilot ap;
126 private bool AutoPilotActive = false;
127 private TransparentButton btnDialogNextControl;
128 private MediaConsole mediaConsole;
131 #region Constructor and disposal
132 public frmMain(RadegastInstance instance)
135 InitializeComponent();
136 Disposed += new EventHandler(frmMain_Disposed);
138 this.instance = instance;
139 this.instance.ClientChanged += new EventHandler<ClientChangedEventArgs>(instance_ClientChanged);
140 netcom.NetcomSync = this;
141 ShowAgentProfile = ShowAgentProfileInternal;
143 pnlDialog.Visible = false;
144 btnDialogNextControl = new TransparentButton();
145 pnlDialog.Controls.Add(btnDialogNextControl);
148 btnDialogNextControl.Size = new Size(35, 20);
149 btnDialogNextControl.BackColor = Color.Transparent;
150 btnDialogNextControl.ForeColor = Color.Gold;
151 btnDialogNextControl.FlatAppearance.BorderSize = 0;
152 btnDialogNextControl.FlatStyle = FlatStyle.Flat;
153 btnDialogNextControl.Anchor = AnchorStyles.Bottom | AnchorStyles.Right;
154 btnDialogNextControl.Text = ">>";
155 btnDialogNextControl.Font = new Font(btnDialogNextControl.Font, FontStyle.Bold);
156 btnDialogNextControl.Margin = new Padding(0);
157 btnDialogNextControl.Padding = new Padding(0);
158 btnDialogNextControl.UseVisualStyleBackColor = false;
159 btnDialogNextControl.Top = btnDialogNextControl.Parent.ClientSize.Height - btnDialogNextControl.Size.Height;
160 btnDialogNextControl.Left = btnDialogNextControl.Parent.ClientSize.Width - btnDialogNextControl.Size.Width;
161 btnDialogNextControl.Click += new EventHandler(btnDialogNextControl_Click);
163 if (instance.MonoRuntime)
165 statusStrip1.LayoutStyle = ToolStripLayoutStyle.Table;
169 netcom.ClientLoginStatus += new EventHandler<LoginProgressEventArgs>(netcom_ClientLoginStatus);
170 netcom.ClientLoggedOut += new EventHandler(netcom_ClientLoggedOut);
171 netcom.ClientDisconnected += new EventHandler<DisconnectedEventArgs>(netcom_ClientDisconnected);
172 instance.Names.NameUpdated += new EventHandler<UUIDNameReplyEventArgs>(Names_NameUpdated);
173 RegisterClientEvents(client);
175 InitializeStatusTimer();
176 RefreshWindowTitle();
179 private void RegisterClientEvents(GridClient client)
181 client.Parcels.ParcelProperties += new EventHandler<ParcelPropertiesEventArgs>(Parcels_ParcelProperties);
182 client.Self.MoneyBalanceReply += new EventHandler<MoneyBalanceReplyEventArgs>(Self_MoneyBalanceReply);
183 client.Self.MoneyBalance += new EventHandler<BalanceEventArgs>(Self_MoneyBalance);
186 private void UnregisterClientEvents(GridClient client)
188 client.Parcels.ParcelProperties -= new EventHandler<ParcelPropertiesEventArgs>(Parcels_ParcelProperties);
189 client.Self.MoneyBalanceReply -= new EventHandler<MoneyBalanceReplyEventArgs>(Self_MoneyBalanceReply);
190 client.Self.MoneyBalance -= new EventHandler<BalanceEventArgs>(Self_MoneyBalance);
193 void instance_ClientChanged(object sender, ClientChangedEventArgs e)
195 UnregisterClientEvents(e.OldClient);
196 RegisterClientEvents(client);
199 void frmMain_Disposed(object sender, EventArgs e)
203 netcom.NetcomSync = null;
204 netcom.ClientLoginStatus -= new EventHandler<LoginProgressEventArgs>(netcom_ClientLoginStatus);
205 netcom.ClientLoggedOut -= new EventHandler(netcom_ClientLoggedOut);
206 netcom.ClientDisconnected -= new EventHandler<DisconnectedEventArgs>(netcom_ClientDisconnected);
211 UnregisterClientEvents(client);
214 if (instance != null && instance.Names != null)
216 instance.Names.NameUpdated -= new EventHandler<UUIDNameReplyEventArgs>(Names_NameUpdated);
219 this.instance.CleanUp();
223 #region Event handlers
224 bool firstMoneyNotification = true;
225 void Self_MoneyBalance(object sender, BalanceEventArgs e)
228 int.TryParse(tlblMoneyBalance.Text, out oldBalance);
229 int delta = Math.Abs(oldBalance - e.Balance);
231 if (firstMoneyNotification)
233 firstMoneyNotification = false;
239 if (oldBalance > e.Balance)
241 instance.MediaManager.PlayUISound(UISounds.MoneyIn);
245 instance.MediaManager.PlayUISound(UISounds.MoneyOut);
251 void Names_NameUpdated(object sender, UUIDNameReplyEventArgs e)
253 if (!e.Names.ContainsKey(client.Self.AgentID)) return;
257 if (IsHandleCreated || !instance.MonoRuntime)
259 BeginInvoke(new MethodInvoker(() => Names_NameUpdated(sender, e)));
264 RefreshWindowTitle();
268 void Self_MoneyBalanceReply(object sender, MoneyBalanceReplyEventArgs e)
270 if (!String.IsNullOrEmpty(e.Description))
272 if (instance.GlobalSettings["transaction_notification_dialog"].AsBoolean())
273 AddNotification(new ntfGeneric(instance, e.Description));
274 if (instance.GlobalSettings["transaction_notification_chat"].AsBoolean())
275 TabConsole.DisplayNotificationInChat(e.Description);
279 public void InitializeControls()
281 InitializeTabsConsole();
283 if (instance.MediaManager.SoundSystemAvailable)
285 mediaConsole = new MediaConsole(instance);
286 tbtnMedia.Visible = true;
290 public bool InAutoReconnect { get; set; }
292 private void DisplayAutoReconnectForm()
294 if (IsDisposed) return;
298 BeginInvoke(new MethodInvoker(DisplayAutoReconnectForm));
302 InAutoReconnect = true;
303 frmReconnect dialog = new frmReconnect(instance, instance.GlobalSettings["reconnect_time"]);
304 dialog.ShowDialog(this);
309 public void BeginAutoReconnect()
311 // Sleep for 3 seconds on a separate thread while things unwind on
312 // disconnect, since ShowDialog() blocks GUI thread
313 (new Thread(new ThreadStart(() =>
315 System.Threading.Thread.Sleep(3000);
316 DisplayAutoReconnectForm();
320 Name = "Reconnect Delay Thread",
326 private void netcom_ClientLoginStatus(object sender, LoginProgressEventArgs e)
328 if (e.Status == LoginStatus.Failed)
332 if (instance.GlobalSettings["auto_reconnect"].AsBoolean())
333 BeginAutoReconnect();
335 InAutoReconnect = false;
338 else if (e.Status == LoginStatus.Success)
340 InAutoReconnect = false;
341 reconnectToolStripMenuItem.Enabled = false;
342 loginToolStripMenuItem.Enabled = false;
343 tsb3D.Enabled = tbtnVoice.Enabled = disconnectToolStripMenuItem.Enabled =
344 tbtnGroups.Enabled = tbnObjects.Enabled = tbtnWorld.Enabled = tbnTools.Enabled = tmnuImport.Enabled =
345 tbtnFriends.Enabled = tbtnInventory.Enabled = tbtnSearch.Enabled = tbtnMap.Enabled = true;
348 RefreshWindowTitle();
352 private void netcom_ClientLoggedOut(object sender, EventArgs e)
354 tsb3D.Enabled = tbtnVoice.Enabled = disconnectToolStripMenuItem.Enabled =
355 tbtnGroups.Enabled = tbnObjects.Enabled = tbtnWorld.Enabled = tbnTools.Enabled = tmnuImport.Enabled =
356 tbtnFriends.Enabled = tbtnInventory.Enabled = tbtnSearch.Enabled = tbtnMap.Enabled = false;
358 reconnectToolStripMenuItem.Enabled = true;
359 loginToolStripMenuItem.Enabled = true;
360 InAutoReconnect = false;
362 if (statusTimer != null)
366 RefreshWindowTitle();
369 private void netcom_ClientDisconnected(object sender, DisconnectedEventArgs e)
371 firstMoneyNotification = true;
373 if (e.Reason == NetworkManager.DisconnectType.ClientInitiated) return;
374 netcom_ClientLoggedOut(sender, EventArgs.Empty);
376 if (instance.GlobalSettings["auto_reconnect"].AsBoolean())
378 BeginAutoReconnect();
382 private void frmMain_FormClosing(object sender, FormClosingEventArgs e)
384 if (statusTimer != null)
387 statusTimer.Dispose();
391 if (mediaConsole != null)
393 if (tabsConsole.TabExists("media"))
395 tabsConsole.Tabs["media"].AllowClose = true;
396 tabsConsole.Tabs["media"].Close();
400 mediaConsole.Dispose();
405 if (netcom.IsLoggedIn)
407 Thread saveInvToDisk = new Thread(new ThreadStart(
410 client.Inventory.Store.SaveToDisk(instance.InventoryCacheFileName);
412 saveInvToDisk.Name = "Save inventory to disk";
413 saveInvToDisk.Start();
420 # region Update status
422 void Parcels_ParcelProperties(object sender, ParcelPropertiesEventArgs e)
424 if (PreventParcelUpdate || e.Result != ParcelResult.Single) return;
427 BeginInvoke(new MethodInvoker(() => Parcels_ParcelProperties(sender, e)));
431 Parcel parcel = instance.State.Parcel = e.Parcel;
433 tlblParcel.Text = parcel.Name;
434 tlblParcel.ToolTipText = parcel.Desc;
436 if ((parcel.Flags & ParcelFlags.AllowFly) != ParcelFlags.AllowFly)
437 icoNoFly.Visible = true;
439 icoNoFly.Visible = false;
441 if ((parcel.Flags & ParcelFlags.CreateObjects) != ParcelFlags.CreateObjects)
442 icoNoBuild.Visible = true;
444 icoNoBuild.Visible = false;
446 if ((parcel.Flags & ParcelFlags.AllowOtherScripts) != ParcelFlags.AllowOtherScripts)
447 icoNoScript.Visible = true;
449 icoNoScript.Visible = false;
451 if ((parcel.Flags & ParcelFlags.RestrictPushObject) == ParcelFlags.RestrictPushObject)
452 icoNoPush.Visible = true;
454 icoNoPush.Visible = false;
456 if ((parcel.Flags & ParcelFlags.AllowDamage) == ParcelFlags.AllowDamage)
457 icoHealth.Visible = true;
459 icoHealth.Visible = false;
461 if ((parcel.Flags & ParcelFlags.AllowVoiceChat) != ParcelFlags.AllowVoiceChat)
462 icoNoVoice.Visible = true;
464 icoNoVoice.Visible = false;
467 private void RefreshStatusBar()
469 if (netcom.IsLoggedIn)
471 tlblLoginName.Text = instance.Names.Get(client.Self.AgentID, client.Self.Name);
472 tlblMoneyBalance.Text = client.Self.Balance.ToString();
473 icoHealth.Text = client.Self.Health.ToString() + "%";
475 var cs = client.Network.CurrentSim;
476 tlblRegionInfo.Text =
477 (cs == null ? "No region" : cs.Name) +
478 " (" + Math.Floor(client.Self.SimPosition.X).ToString() + ", " +
479 Math.Floor(client.Self.SimPosition.Y).ToString() + ", " +
480 Math.Floor(client.Self.SimPosition.Z).ToString() + ")";
484 tlblLoginName.Text = "Offline";
485 tlblMoneyBalance.Text = "0";
486 icoHealth.Text = "0%";
487 tlblRegionInfo.Text = "No Region";
488 tlblParcel.Text = "No Parcel";
490 icoHealth.Visible = false;
491 icoNoBuild.Visible = false;
492 icoNoFly.Visible = false;
493 icoNoPush.Visible = false;
494 icoNoScript.Visible = false;
495 icoNoVoice.Visible = false;
499 private void RefreshWindowTitle()
501 string name = instance.Names.Get(client.Self.AgentID, client.Self.Name);
502 StringBuilder sb = new StringBuilder();
503 sb.Append("Radegast - ");
505 if (netcom.IsLoggedIn)
507 sb.Append("[" + name + "]");
509 if (instance.State.IsAway)
511 sb.Append(" - Away");
512 if (instance.State.IsBusy) sb.Append(", Busy");
514 else if (instance.State.IsBusy)
516 sb.Append(" - Busy");
519 if (instance.State.IsFollowing)
521 sb.Append(" - Following ");
522 sb.Append(instance.State.FollowName);
527 sb.Append("Logged Out");
530 this.Text = sb.ToString();
532 // When minimized to tray, update tray tool tip also
533 if (WindowState == FormWindowState.Minimized && instance.GlobalSettings["minimize_to_tray"])
535 trayIcon.Text = sb.ToString();
536 ctxTrayMenuLabel.Text = sb.ToString();
542 private void InitializeStatusTimer()
544 statusTimer = new System.Timers.Timer(250);
545 statusTimer.SynchronizingObject = this;
546 statusTimer.Elapsed += new ElapsedEventHandler(statusTimer_Elapsed);
549 private void statusTimer_Elapsed(object sender, ElapsedEventArgs e)
551 // Mono sometimes fires timer after is's disposed
554 RefreshWindowTitle();
561 #region Initialization, configuration, and key shortcuts
562 private void InitializeTabsConsole()
564 tabsConsole = new TabsConsole(instance);
565 tabsConsole.Dock = DockStyle.Fill;
566 toolStripContainer1.ContentPanel.Controls.Add(tabsConsole);
569 private void frmMain_KeyDown(object sender, KeyEventArgs e)
571 // Ctrl-Alt-Shift-H Say "Hippos!" in chat
572 if (e.Modifiers == (Keys.Control | Keys.Shift | Keys.Alt) && e.KeyCode == Keys.H)
574 e.Handled = e.SuppressKeyPress = true;
575 netcom.ChatOut("Hippos!", ChatType.Normal, 0);
579 // Ctrl-Shift-1 (sim/parcel info)
580 if (e.Modifiers == (Keys.Control | Keys.Shift) && e.KeyCode == Keys.D1)
582 e.Handled = e.SuppressKeyPress = true;
583 DisplayRegionParcelConsole();
588 if (e.Modifiers == Keys.Control && e.KeyCode == Keys.W)
590 e.Handled = e.SuppressKeyPress = true;
591 RadegastTab tab = tabsConsole.SelectedTab;
597 else if (tab.AllowHide)
605 // Ctl-Shift-H: Teleport Home
606 if (e.Modifiers == (Keys.Control | Keys.Shift) && e.KeyCode == Keys.H)
608 e.Handled = e.SuppressKeyPress = true;
609 tmnuTeleportHome.PerformClick();
613 // Alt-Ctrl-D Open debug console
614 if (e.Modifiers == (Keys.Control | Keys.Alt) && e.KeyCode == Keys.D)
616 e.Handled = e.SuppressKeyPress = true;
617 debugConsoleToolStripMenuItem.PerformClick();
621 // Alt 1-8: Toggle various tabs
622 if (e.Modifiers == Keys.Alt)
627 e.Handled = e.SuppressKeyPress = true;
628 tabsConsole.Tabs["chat"].Select();
632 e.Handled = e.SuppressKeyPress = true;
633 tbtnFriends.PerformClick();
637 e.Handled = e.SuppressKeyPress = true;
638 tbtnGroups.PerformClick();
642 e.Handled = e.SuppressKeyPress = true;
643 tbtnInventory.PerformClick();
647 e.Handled = e.SuppressKeyPress = true;
648 tbtnSearch.PerformClick();
652 e.Handled = e.SuppressKeyPress = true;
653 tbtnMap.PerformClick();
657 e.Handled = e.SuppressKeyPress = true;
658 tbnObjects.PerformClick();
662 e.Handled = e.SuppressKeyPress = true;
663 tbtnMedia.PerformClick();
667 e.Handled = e.SuppressKeyPress = true;
668 tbtnVoice.PerformClick();
673 // ctrl-g, goto slurl
674 if (e.Control && e.KeyCode == Keys.G)
676 if (!ProcessLink(Clipboard.GetText(), true))
677 MapToCurrentLocation();
679 e.Handled = e.SuppressKeyPress = true;
683 // ctrl-(shift)-tab for next/previous tab
684 if (e.Control && e.KeyCode == Keys.Tab)
688 TabConsole.SelectPreviousTab();
692 TabConsole.SelectNextTab();
694 e.Handled = e.SuppressKeyPress = true;
699 bool firstLoad = true;
701 private void frmMain_Load(object sender, EventArgs e)
706 tabsConsole.SelectTab("login");
707 ResourceManager rm = Properties.Resources.ResourceManager;
708 ResourceSet set = rm.GetResourceSet(System.Globalization.CultureInfo.CurrentCulture, true, true);
709 System.Collections.IDictionaryEnumerator de = set.GetEnumerator();
710 while (de.MoveNext() == true)
712 if (de.Entry.Value is Image)
714 Bitmap bitMap = de.Entry.Value as Bitmap;
715 ResourceImages.Images.Add(bitMap);
716 ImageNames.Add(de.Entry.Key.ToString());
719 StartUpdateCheck(false);
721 if (instance.PlainColors)
723 pnlDialog.BackColor = System.Drawing.Color.FromArgb(120, 220, 255);
730 #region Public methods
732 private Dictionary<UUID, frmProfile> shownProfiles = new Dictionary<UUID, frmProfile>();
734 void ShowAgentProfileInternal(string name, UUID agentID)
738 frmProfile profile = null;
739 if (shownProfiles.TryGetValue(agentID, out profile))
741 profile.WindowState = FormWindowState.Normal;
746 profile = new frmProfile(instance, name, agentID);
748 profile.Disposed += (object sender, EventArgs e) =>
752 frmProfile agentProfile = (frmProfile)sender;
753 if (shownProfiles.ContainsKey(agentProfile.AgentID))
754 shownProfiles.Remove(agentProfile.AgentID);
760 shownProfiles.Add(agentID, profile);
765 private Dictionary<UUID, frmGroupInfo> shownGroupProfiles = new Dictionary<UUID, frmGroupInfo>();
767 public void ShowGroupProfile(UUID id)
769 ShowGroupProfile(new OpenMetaverse.Group()
775 public void ShowGroupProfile(AvatarGroup group)
777 ShowGroupProfile(new OpenMetaverse.Group()
780 InsigniaID = group.GroupInsigniaID,
781 Name = group.GroupName
786 public void ShowGroupProfile(OpenMetaverse.Group group)
790 BeginInvoke(new MethodInvoker(() => ShowGroupProfile(group)));
794 lock (shownGroupProfiles)
796 frmGroupInfo profile = null;
797 if (shownGroupProfiles.TryGetValue(group.ID, out profile))
799 profile.WindowState = FormWindowState.Normal;
804 profile = new frmGroupInfo(instance, group);
806 profile.Disposed += (object sender, EventArgs e) =>
808 lock (shownGroupProfiles)
810 frmGroupInfo groupProfile = (frmGroupInfo)sender;
811 if (shownGroupProfiles.ContainsKey(groupProfile.Group.ID))
812 shownGroupProfiles.Remove(groupProfile.Group.ID);
818 shownGroupProfiles.Add(group.ID, profile);
823 public bool ProcessSecondlifeURI(string link)
825 // First try if we have a region name, assume it's a teleport link if we have
826 Regex r = new Regex(@"^(secondlife://)(?<region>[^/$]+)(/(?<x>\d+))?((/?<y>\d+))?(/(?<z>\d+))?",
827 RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture | RegexOptions.IgnoreCase);
828 Match m = r.Match(link);
832 string region = HttpUtility.UrlDecode(m.Groups["region"].Value);
833 int x = string.IsNullOrEmpty(m.Groups["x"].Value) ? 128 : int.Parse(m.Groups["x"].Value);
834 int y = string.IsNullOrEmpty(m.Groups["y"].Value) ? 128 : int.Parse(m.Groups["y"].Value);
835 int z = string.IsNullOrEmpty(m.Groups["z"].Value) ? 0 : int.Parse(m.Groups["z"].Value);
837 WorldMap.DisplayLocation(region, x, y, z);
841 // Is it group profile link
842 r = new Regex(@"^secondlife:///app/group/(?<id>[^/]+)/about",
843 RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture | RegexOptions.IgnoreCase);
849 if (UUID.TryParse(m.Groups["id"].Value, out id))
851 ShowGroupProfile(id);
857 // Is it user profile link
858 r = new Regex(@"^secondlife:///app/agent/(?<id>[^/]+)/about",
859 RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture | RegexOptions.IgnoreCase);
865 if (UUID.TryParse(m.Groups["id"].Value, out id))
867 ShowAgentProfile(instance.Names.Get(id), id);
876 public void ProcessLink(string link)
878 ProcessLink(link, false);
881 public bool ProcessLink(string link, bool onlyMap)
883 var pos = link.IndexOf(RRichTextBox.LinkSeparator);
886 link = link.Substring(pos + 1);
889 if (link.StartsWith("secondlife://"))
891 return ProcessSecondlifeURI(link);
894 if (!link.Contains("://"))
896 link = "http://" + link;
899 Regex r = new Regex(@"^(http://(slurl\.com|maps\.secondlife\.com)/secondlife/|secondlife://)(?<region>[^/]+)/(?<x>\d+)/(?<y>\d+)(/(?<z>\d+))?",
900 RegexOptions.CultureInvariant | RegexOptions.ExplicitCapture | RegexOptions.IgnoreCase
902 Match m = r.Match(link);
906 string region = HttpUtility.UrlDecode(m.Groups["region"].Value);
907 int x = int.Parse(m.Groups["x"].Value);
908 int y = int.Parse(m.Groups["y"].Value);
911 if (!string.IsNullOrEmpty(m.Groups["z"].Value))
913 z = int.Parse(m.Groups["z"].Value);
917 WorldMap.DisplayLocation(region, x, y, z);
922 System.Diagnostics.Process.Start(link);
928 #region Notifications
929 CircularList<Control> notifications = new CircularList<Control>();
931 public Color NotificationBackground
933 get { return pnlDialog.BackColor; }
936 void ResizeNotificationByControl(Control active)
938 int Width = active.Size.Width + 6;
939 int Height = notifications.Count > 1 ? active.Size.Height + 3 + btnDialogNextControl.Size.Height : active.Size.Height + 3;
940 pnlDialog.Size = new Size(Width, Height);
942 pnlDialog.Left = pnlDialog.Parent.ClientSize.Width - Width;
944 btnDialogNextControl.Top = btnDialogNextControl.Parent.ClientSize.Height - btnDialogNextControl.Size.Height;
945 btnDialogNextControl.Left = btnDialogNextControl.Parent.ClientSize.Width - btnDialogNextControl.Size.Width;
947 btnDialogNextControl.BringToFront();
950 public void AddNotification(Control control)
954 BeginInvoke(new MethodInvoker(delegate()
956 AddNotification(control);
962 Control active = TabsConsole.FindFocusedControl(this);
964 FormFlash.StartFlash(this);
965 pnlDialog.Visible = true;
966 pnlDialog.BringToFront();
968 foreach (Control existing in notifications)
970 existing.Visible = false;
973 notifications.Add(control);
974 control.Visible = true;
975 control.Anchor = AnchorStyles.Top | AnchorStyles.Left;
978 pnlDialog.Controls.Add(control);
979 ResizeNotificationByControl(control);
981 btnDialogNextControl.Visible = notifications.Count > 1;
989 public void RemoveNotification(Control control)
991 pnlDialog.Controls.Remove(control);
992 notifications.Remove(control);
995 if (notifications.HasNext)
997 pnlDialog.Visible = true;
998 Control active = notifications.Next;
999 active.Visible = true;
1000 ResizeNotificationByControl(active);
1004 pnlDialog.Visible = false;
1007 btnDialogNextControl.Visible = notifications.Count > 1;
1010 private void btnDialogNextControl_Click(object sender, EventArgs e)
1012 foreach (Control existing in notifications)
1014 existing.Visible = false;
1017 if (notifications.HasNext)
1019 pnlDialog.Visible = true;
1020 Control active = notifications.Next;
1021 active.Visible = true;
1022 ResizeNotificationByControl(active);
1026 pnlDialog.Visible = false;
1030 #endregion Notifications
1032 #region Menu click handlers
1034 private void tmnuStatusAway_Click(object sender, EventArgs e)
1036 instance.State.SetAway(tmnuStatusAway.Checked);
1039 private void tmnuHelpReadme_Click(object sender, EventArgs e)
1041 System.Diagnostics.Process.Start(Application.StartupPath + @"\Readme.txt");
1044 private void tmnuStatusBusy_Click(object sender, EventArgs e)
1046 instance.State.SetBusy(tmnuStatusBusy.Checked);
1049 private void tmnuControlFly_Click(object sender, EventArgs e)
1051 instance.State.SetFlying(tmnuControlFly.Checked);
1054 private void tmnuControlAlwaysRun_Click(object sender, EventArgs e)
1056 instance.State.SetAlwaysRun(tmnuControlAlwaysRun.Checked);
1059 private void tmnuPrefs_Click(object sender, EventArgs e)
1061 (new frmSettings(instance)).ShowDialog();
1064 private void tbtnAppearance_Click(object sender, EventArgs e)
1066 client.Appearance.RequestSetAppearance(false);
1069 private void importObjectToolStripMenuItem_Click(object sender, EventArgs e)
1071 PrimDeserializer.ImportFromFile(client);
1074 private void autopilotToolStripMenuItem_Click(object sender, EventArgs e)
1078 ap = new AutoPilot(client);
1080 ap.InsertWaypoint(new Vector3(66, 163, 21));
1081 ap.InsertWaypoint(new Vector3(66, 98, 21));
1083 ap.InsertWaypoint(new Vector3(101, 98, 21));
1084 ap.InsertWaypoint(new Vector3(101, 45, 21));
1085 ap.InsertWaypoint(new Vector3(93, 27, 21));
1086 ap.InsertWaypoint(new Vector3(106, 12, 21));
1087 ap.InsertWaypoint(new Vector3(123, 24, 21));
1088 ap.InsertWaypoint(new Vector3(114, 45, 21));
1089 ap.InsertWaypoint(new Vector3(114, 98, 21));
1091 ap.InsertWaypoint(new Vector3(130, 98, 21));
1092 ap.InsertWaypoint(new Vector3(130, 163, 21));
1094 ap.InsertWaypoint(new Vector3(64, 68, 21));
1095 ap.InsertWaypoint(new Vector3(65, 20, 21));
1096 ap.InsertWaypoint(new Vector3(33, 23, 21));
1097 ap.InsertWaypoint(new Vector3(17, 39, 21));
1098 ap.InsertWaypoint(new Vector3(17, 62, 21));
1102 if (AutoPilotActive)
1104 AutoPilotActive = false;
1109 AutoPilotActive = true;
1115 private void cleanCacheToolStripMenuItem_Click(object sender, EventArgs e)
1117 ThreadPool.QueueUserWorkItem(sync => client.Assets.Cache.Clear());
1118 instance.Names.CleanCache();
1121 private void rebakeTexturesToolStripMenuItem_Click(object sender, EventArgs e)
1123 client.Appearance.RequestSetAppearance(true);
1126 public void MapToCurrentLocation()
1128 if (MapTab != null && client.Network.Connected)
1131 WorldMap.DisplayLocation(client.Network.CurrentSim.Name,
1132 (int)client.Self.SimPosition.X,
1133 (int)client.Self.SimPosition.Y,
1134 (int)client.Self.SimPosition.Z);
1138 private void standToolStripMenuItem_Click(object sender, EventArgs e)
1140 instance.State.SetSitting(false, UUID.Zero);
1143 private void groundSitToolStripMenuItem_Click(object sender, EventArgs e)
1145 client.Self.SitOnGround();
1148 private void newWindowToolStripMenuItem_Click(object sender, EventArgs e)
1150 try { System.Diagnostics.Process.Start(Application.ExecutablePath); }
1151 catch (Exception) { }
1154 private void tmnuExit_Click(object sender, EventArgs e)
1159 private void tlblRegionInfo_Click(object sender, EventArgs e)
1161 if (WorldMap != null && client.Network.Connected)
1167 private void scriptEditorToolStripMenuItem_Click(object sender, EventArgs e)
1169 ScriptEditor se = new ScriptEditor(instance);
1170 se.Dock = DockStyle.Fill;
1174 private void tmnuSetHome_Click(object sender, EventArgs e)
1176 client.Self.SetHome();
1179 private void tmnuCreateLandmark_Click(object sender, EventArgs e)
1181 string location = string.Format(", {0} ({1}, {2}, {3})",
1182 client.Network.CurrentSim.Name,
1183 (int)client.Self.SimPosition.X,
1184 (int)client.Self.SimPosition.Y,
1185 (int)client.Self.SimPosition.Z
1188 string name = tlblParcel.Text;
1189 int maxLen = 63 - location.Length;
1191 if (name.Length > maxLen)
1192 name = name.Substring(0, maxLen);
1196 client.Inventory.RequestCreateItem(
1197 client.Inventory.FindFolderForType(AssetType.Landmark),
1199 tlblParcel.ToolTipText,
1202 InventoryType.Landmark,
1204 (bool success, InventoryItem item) =>
1208 BeginInvoke(new MethodInvoker(() =>
1210 Landmark ln = new Landmark(instance, (InventoryLandmark)item);
1211 ln.Dock = DockStyle.Fill;
1220 private void timerWorldClock_Tick(object sender, EventArgs e)
1222 lblTime.Text = instance.GetWorldTime().ToString("h:mm tt", System.Globalization.CultureInfo.InvariantCulture);
1225 private void reportBugsToolStripMenuItem_Click(object sender, EventArgs e)
1227 ProcessLink("http://jira.openmetaverse.org/browse/RAD");
1230 private void aboutRadegastToolStripMenuItem_Click(object sender, EventArgs e)
1232 (new frmAbout(instance)).ShowDialog();
1235 #region Update Checking
1236 private UpdateChecker updateChecker = null;
1237 private bool ManualUpdateCheck = false;
1239 public void StartUpdateCheck(bool userInitiated)
1241 ManualUpdateCheck = userInitiated;
1243 if (updateChecker != null)
1245 if (ManualUpdateCheck)
1246 tabsConsole.DisplayNotificationInChat("Update check already in progress.");
1250 if (ManualUpdateCheck)
1251 tabsConsole.DisplayNotificationInChat("Checking for updates...", ChatBufferTextStyle.StatusBlue);
1252 updateChecker = new UpdateChecker();
1253 updateChecker.OnUpdateInfoReceived += new UpdateChecker.UpdateInfoCallback(OnUpdateInfoReceived);
1254 updateChecker.StartCheck();
1257 private void checkForUpdatesToolStripMenuItem_Click(object sender, EventArgs e)
1259 tabsConsole.SelectTab("chat");
1260 StartUpdateCheck(true);
1263 void OnUpdateInfoReceived(object sender, UpdateCheckerArgs e)
1267 BeginInvoke(new MethodInvoker(() => OnUpdateInfoReceived(sender, e)));
1273 if (ManualUpdateCheck)
1274 tabsConsole.DisplayNotificationInChat("Error: Failed connecting to the update site.", ChatBufferTextStyle.StatusBlue);
1278 if (!ManualUpdateCheck && e.Info.DisplayMOTD)
1280 tabsConsole.DisplayNotificationInChat(e.Info.MOTD, ChatBufferTextStyle.StatusBlue);
1283 if (e.Info.UpdateAvailable)
1285 tabsConsole.DisplayNotificationInChat("New version available at " + e.Info.DownloadSite, ChatBufferTextStyle.Alert);
1289 if (ManualUpdateCheck)
1290 tabsConsole.DisplayNotificationInChat("Your version is up to date.", ChatBufferTextStyle.StatusBlue);
1294 updateChecker.Dispose();
1295 updateChecker = null;
1299 private void ToggleHidden(string tabName)
1301 if (!tabsConsole.TabExists(tabName)) return;
1303 RadegastTab tab = tabsConsole.Tabs[tabName];
1322 private void tbtnFriends_Click(object sender, EventArgs e)
1324 ToggleHidden("friends");
1327 private void tbtnInventory_Click(object sender, EventArgs e)
1329 ToggleHidden("inventory");
1332 private void tbtnSearch_Click(object sender, EventArgs e)
1334 ToggleHidden("search");
1337 private void tbtnGroups_Click(object sender, EventArgs e)
1339 ToggleHidden("groups");
1342 private void tbtnVoice_Click(object sender, EventArgs e)
1344 ToggleHidden("voice");
1347 private void tbtnMedia_Click(object sender, EventArgs e)
1349 if (tabsConsole.TabExists("media"))
1351 ToggleHidden("media");
1355 RadegastTab tab = tabsConsole.AddTab("media", "Media", mediaConsole);
1356 tab.AllowClose = false;
1357 tab.AllowHide = true;
1362 private void debugConsoleToolStripMenuItem_Click(object sender, EventArgs e)
1364 if (tabsConsole.TabExists("debug"))
1366 ToggleHidden("debug");
1370 RadegastTab tab = tabsConsole.AddTab("debug", "Debug", new DebugConsole(instance));
1371 tab.AllowClose = false;
1372 tab.AllowHide = true;
1377 private void tbnObjects_Click(object sender, EventArgs e)
1379 if (tabsConsole.TabExists("objects"))
1381 RadegastTab tab = tabsConsole.Tabs["objects"];
1385 ((ObjectsConsole)tab.Control).RefreshObjectList();
1394 RadegastTab tab = tabsConsole.AddTab("objects", "Objects", new ObjectsConsole(instance));
1395 tab.AllowClose = true;
1396 tab.AllowDetach = true;
1398 tab.AllowHide = false;
1400 ((ObjectsConsole)tab.Control).RefreshObjectList();
1404 private void tbtnMap_Click(object sender, EventArgs e)
1406 if (MapTab == null) return; // too soon!
1408 ToggleHidden("map");
1410 MapToCurrentLocation();
1413 private void disconnectToolStripMenuItem_Click(object sender, EventArgs e)
1415 if (client.Network.Connected)
1416 client.Network.RequestLogout();
1419 private void reconnectToolStripMenuItem_Click(object sender, EventArgs e)
1421 if (!client.Network.Connected)
1423 instance.Reconnect();
1427 private frmKeyboardShortcuts keyboardShortcutsForm = null;
1428 private void keyboardShortcutsToolStripMenuItem_Click(object sender, EventArgs e)
1430 if (keyboardShortcutsForm != null)
1432 keyboardShortcutsForm.Focus();
1436 keyboardShortcutsForm = new frmKeyboardShortcuts(instance);
1438 keyboardShortcutsForm.Disposed += (object senderx, EventArgs ex) =>
1440 if (components != null)
1442 components.Remove(keyboardShortcutsForm);
1444 keyboardShortcutsForm = null;
1447 keyboardShortcutsForm.Show(this);
1448 keyboardShortcutsForm.Top = Top + 100;
1449 keyboardShortcutsForm.Left = Left + 100;
1451 if (components != null)
1453 components.Add(keyboardShortcutsForm);
1458 // Menu item for testing out stuff
1459 private void testToolStripMenuItem_Click(object sender, EventArgs e)
1463 private void reloadInventoryToolStripMenuItem_Click(object sender, EventArgs e)
1465 if (tabsConsole.TabExists("inventory"))
1467 ((InventoryConsole)tabsConsole.Tabs["inventory"].Control).ReloadInventory();
1468 tabsConsole.Tabs["inventory"].Select();
1472 private void btnLoadScript_Click(object sender, EventArgs e)
1474 if (!TabConsole.TabExists("plugin_manager"))
1476 TabConsole.AddTab("plugin_manager", "Plugins", new PluginsTab(instance));
1478 TabConsole.Tabs["plugin_manager"].Select();
1481 private void frmMain_Resize(object sender, EventArgs e)
1483 if (WindowState == FormWindowState.Minimized && instance.GlobalSettings["minimize_to_tray"].AsBoolean())
1485 if (TabConsole.TabExists("scene_window") && !TabConsole.Tabs["scene_window"].Detached)
1487 TabConsole.Tabs["scene_window"].Close();
1489 ShowInTaskbar = false;
1490 trayIcon.Visible = true;
1491 FormBorderStyle = FormBorderStyle.SizableToolWindow;
1495 private void treyIcon_MouseDoubleClick(object sender, MouseEventArgs e)
1497 WindowState = FormWindowState.Normal;
1498 ShowInTaskbar = true;
1499 trayIcon.Visible = false;
1500 FormBorderStyle = FormBorderStyle.Sizable;
1503 private void ctxTreyRestore_Click(object sender, EventArgs e)
1505 treyIcon_MouseDoubleClick(this, null);
1508 private void ctxTreyExit_Click(object sender, EventArgs e)
1510 tmnuExit_Click(this, EventArgs.Empty);
1513 private void tmnuTeleportHome_Click(object sender, EventArgs e)
1515 TabConsole.DisplayNotificationInChat("Teleporting home...");
1516 client.Self.RequestTeleport(UUID.Zero);
1519 private void stopAllAnimationsToolStripMenuItem_Click(object sender, EventArgs e)
1521 instance.State.StopAllAnimations();
1524 public void DisplayRegionParcelConsole()
1526 if (tabsConsole.TabExists("current region info"))
1528 tabsConsole.Tabs["current region info"].Select();
1529 (tabsConsole.Tabs["current region info"].Control as RegionInfo).UpdateDisplay();
1533 tabsConsole.AddTab("current region info", "Region info", new RegionInfo(instance));
1534 tabsConsole.Tabs["current region info"].Select();
1538 private void regionParcelToolStripMenuItem_Click(object sender, EventArgs e)
1540 DisplayRegionParcelConsole();
1543 private void tlblParcel_Click(object sender, EventArgs e)
1545 DisplayRegionParcelConsole();
1548 private void changeMyDisplayNameToolStripMenuItem_Click(object sender, EventArgs e)
1550 if (!client.Avatars.DisplayNamesAvailable())
1552 tabsConsole.DisplayNotificationInChat("This grid does not support display names.", ChatBufferTextStyle.Error);
1556 var dlg = new DisplayNameChange(instance);
1560 private void muteListToolStripMenuItem_Click(object sender, EventArgs e)
1562 if (!tabsConsole.TabExists("mute list console"))
1564 tabsConsole.AddTab("mute list console", "Mute list", new MuteList(instance));
1566 tabsConsole.Tabs["mute list console"].Select();
1569 private void uploadImageToolStripMenuItem_Click(object sender, EventArgs e)
1571 if (!tabsConsole.TabExists("image upload console"))
1573 tabsConsole.AddTab("image upload console", "Upload image", new ImageUploadConsole(instance));
1575 tabsConsole.Tabs["image upload console"].Select();
1579 private void myAttachmentsToolStripMenuItem_Click(object sender, EventArgs e)
1581 Avatar av = client.Network.CurrentSim.ObjectsAvatars.Find((Avatar a) => { return a.ID == client.Self.AgentID; });
1585 tabsConsole.DisplayNotificationInChat("Unable to find my avatar!", ChatBufferTextStyle.Error);
1589 if (!instance.TabConsole.TabExists("AT: " + av.ID.ToString()))
1591 instance.TabConsole.AddTab("AT: " + av.ID.ToString(), "My Attachments", new AttachmentTab(instance, av));
1593 instance.TabConsole.SelectTab("AT: " + av.ID.ToString());
1597 private void tsb3D_Click(object sender, EventArgs e)
1599 if (instance.TabConsole.TabExists("scene_window"))
1601 instance.TabConsole.Tabs["scene_window"].Select();
1605 var control = new Rendering.SceneWindow(instance);
1606 control.Dock = DockStyle.Fill;
1607 instance.TabConsole.AddTab("scene_window", "Scene Viewer", control);
1608 instance.TabConsole.Tabs["scene_window"].Floater = false;
1609 instance.TabConsole.Tabs["scene_window"].CloseOnDetachedClose = true;
1610 control.RegisterTabEvents();
1612 if (instance.GlobalSettings["scene_window_docked"])
1614 instance.TabConsole.Tabs["scene_window"].Select();
1618 instance.TabConsole.Tabs["scene_window"].Detach(instance);
1623 private void loginToolStripMenuItem_Click(object sender, EventArgs e)
1625 TabConsole.InitializeMainTab();
1626 TabConsole.Tabs["login"].Select();