UI/FormRegionPrivacyStatus.cs
author Dean Looyengoed
Tue, 29 Mar 2016 18:33:18 +0200
changeset 737 c905c76679d1
parent 730 87a202808393
child 740 cc736d688b23
permissions -rw-r--r--
Implement an in-line cancel handshake button in the manage privacy status identities list.
Dean@159
     1
´╗┐using pEpCOMServerAdapterLib;
Dean@159
     2
using System;
vb@133
     3
using System.Collections.Generic;
Dean@445
     4
using System.ComponentModel;
Dean@438
     5
using System.Drawing;
Dean@159
     6
using System.Runtime.InteropServices;
vb@133
     7
using System.Windows.Forms;
Dean@159
     8
using Color = System.Drawing.Color;
vb@133
     9
using Outlook = Microsoft.Office.Interop.Outlook;
vb@133
    10
vb@133
    11
namespace pEp
vb@133
    12
{
Dean@159
    13
    /// <summary>
Dean@312
    14
    /// Partial class for the privacy status form region that is displayed below every outlook message.
Dean@159
    15
    /// </summary>
Dean@312
    16
    partial class FormRegionPrivacyStatus
vb@133
    17
    {
vb@133
    18
        #region Form Region Factory
vb@133
    19
vb@133
    20
        [Microsoft.Office.Tools.Outlook.FormRegionMessageClass(Microsoft.Office.Tools.Outlook.FormRegionMessageClassAttribute.Note)]
Dean@312
    21
        [Microsoft.Office.Tools.Outlook.FormRegionName("pEp.FormRegionPrivacyStatus")]
Dean@312
    22
        public partial class FormRegionPrivacyStatusFactory
vb@133
    23
        {
vb@133
    24
            // Occurs before the form region is initialized.
vb@133
    25
            // To prevent the form region from appearing, set e.Cancel to true.
vb@133
    26
            // Use e.OutlookItem to get a reference to the current Outlook item.
Dean@312
    27
            private void FormRegionPrivacyStatus_FormRegionInitializing(object sender, Microsoft.Office.Tools.Outlook.FormRegionInitializingEventArgs e)
vb@133
    28
            {
Dean@327
    29
                WindowFormRegionCollection formRegions;
Dean@327
    30
Dean@327
    31
                /* There is a Microsoft bug at least in Outlook 2013 and Windows 8.1
Dean@327
    32
                 * This bug causes multiple PrivacyStatus form regions to appear stacked on top of each other.
Dean@327
    33
                 * To trigger this bug, on an unencrypted server, click reply to compose an in-line response.
Dean@327
    34
                 * Then click to another tab such as People or Tasks. Then click back on the mail tab to view 
Dean@327
    35
                 * the original email again. Two form regions will be visible: 
Dean@327
    36
                 * (1) for the status during the in-line reply and 
Dean@327
    37
                 * (2) for the status of only the received message.
Dean@327
    38
                 * 
Dean@327
    39
                 * To fix this bug, any existing form regions are found and closed when initializing a new privacy status
Dean@327
    40
                 * form region.
Dean@327
    41
                 */
Dean@327
    42
Dean@327
    43
                try
Dean@327
    44
                {
Dean@327
    45
                    formRegions = Globals.FormRegions[Globals.ThisAddIn.Application.ActiveExplorer()];
Dean@327
    46
                    if (formRegions.FormRegionPrivacyStatus != null)
Dean@327
    47
                    {
Dean@327
    48
                        /* Note: there seems to be no way to actually close the form region.
Dean@327
    49
                         * Therefore, this is as close as possible to actually closing it.
Dean@327
    50
                         * The actual form regions will be cleaned up as soon as another email is selected.
Dean@327
    51
                         */
Dean@327
    52
                        formRegions.FormRegionPrivacyStatus.OutlookFormRegion.Visible = false;
Dean@327
    53
                    }
Dean@327
    54
                }
Dean@327
    55
                catch { }
Dean@327
    56
Dean@327
    57
                return;
vb@133
    58
            }
vb@133
    59
        }
vb@133
    60
vb@133
    61
        #endregion
vb@133
    62
Dean@159
    63
        /* Notes:
Dean@179
    64
         * 
Dean@159
    65
         * Use this.OutlookItem to get a reference to the current Outlook item.
Dean@159
    66
         * Use this.OutlookFormRegion to get a reference to the form region.
Dean@179
    67
         * 
Dean@179
    68
         * UI State Managment:
Dean@179
    69
         * 
Dean@179
    70
         * The UI state is almost entirely set from the associated mail item data.
Dean@179
    71
         * However, a separate state class is maintained to represent the UI as some separation is needed.
Dean@179
    72
         * This logical separation MUST be maintained throughout the code.
Dean@179
    73
         * Specific cases are noted where possible.
Dean@179
    74
         * 
Dean@356
    75
         * The separate privacy status manager form state is also managed here.
Dean@259
    76
         * 
Dean@159
    77
         */
Dean@159
    78
Dean@680
    79
        private CryptableMailItem       associatedMailItem = null;
Dean@680
    80
        private bool                    eventsAreConnected = false;
Dean@680
    81
        private bool                    initialized        = false;
Dean@680
    82
        private FormManagePrivacyStatus managerForm        = null;
Dean@680
    83
        private bool                    refreshOngoing     = false;
Dean@259
    84
Dean@250
    85
        // Data or controls not stored in the UI
Dean@179
    86
        private _pEp_color stateUIColorRating; // WARNING: Do NOT use this directly
vb@133
    87
Dean@159
    88
        /**************************************************************
Dean@159
    89
         * 
Dean@159
    90
         * Methods
Dean@159
    91
         * 
Dean@159
    92
         *************************************************************/
Dean@159
    93
Dean@171
    94
        /// <summary>
Dean@438
    95
        /// Determines if this form region is running in an inspector window.
Dean@438
    96
        /// If not true, it is assumed to be within an explorer window.
Dean@438
    97
        /// </summary>
Dean@438
    98
        /// <returns>True if an inspector window, false if within an explorer.</returns>
Dean@438
    99
        private bool IsWithinInspector()
Dean@438
   100
        {
Dean@438
   101
            bool isWithinInspector = false;
Dean@438
   102
            Outlook.Explorer temp = null;
Dean@438
   103
Dean@438
   104
            try
Dean@438
   105
            {
Dean@438
   106
                // Just try to cast to an explorer class
Dean@438
   107
                temp = (Outlook.Explorer)this.OutlookFormRegion.Parent;
Dean@438
   108
                isWithinInspector = false;
Dean@438
   109
            }
Dean@438
   110
            catch
Dean@438
   111
            {
Dean@438
   112
                isWithinInspector = true;
Dean@438
   113
            }
Dean@438
   114
            finally
Dean@438
   115
            {
Dean@438
   116
                if (temp != null)
Dean@438
   117
                {
Dean@438
   118
                    Marshal.ReleaseComObject(temp);
Dean@438
   119
                    temp = null;
Dean@438
   120
                }
Dean@438
   121
            }
Dean@438
   122
Dean@438
   123
            return (isWithinInspector);
Dean@438
   124
        }
Dean@438
   125
Dean@438
   126
        /// <summary>
Dean@170
   127
        /// Sets the cryptable mail item associated with this encryption status panel.
Dean@170
   128
        /// The associated mail item can then be accessed through its local variable.
Dean@170
   129
        /// </summary>
Dean@170
   130
        private void SetAssociatedMailItem()
Dean@170
   131
        {
Dean@170
   132
            bool errorOccurred = false;
Dean@170
   133
            Outlook.MailItem omi = null;
Dean@170
   134
Dean@170
   135
            // Null check
Dean@170
   136
            if (!errorOccurred)
Dean@170
   137
            {
Dean@170
   138
                try
Dean@170
   139
                {
Dean@170
   140
                    if (this.OutlookItem == null)
Dean@170
   141
                    {
Dean@170
   142
                        errorOccurred = true;
Dean@170
   143
                    }
Dean@170
   144
                }
Dean@170
   145
                catch (COMException)
Dean@170
   146
                {
Dean@170
   147
                    errorOccurred = true;
Dean@170
   148
                }
Dean@170
   149
            }
Dean@170
   150
Dean@170
   151
            // Attempt to get and cast the outlook mail item
Dean@170
   152
            if (!errorOccurred)
Dean@170
   153
            {
Dean@170
   154
                try
Dean@170
   155
                {
Dean@170
   156
                    omi = (Outlook.MailItem)this.OutlookItem;
Dean@170
   157
                }
Dean@170
   158
                catch
Dean@170
   159
                {
Dean@170
   160
                    errorOccurred = true;
Dean@170
   161
                }
Dean@170
   162
            }
Dean@170
   163
Dean@170
   164
            // Finally set the associated mail item
Dean@170
   165
            if ((errorOccurred) ||
Dean@170
   166
                (omi == null))
Dean@170
   167
            {
Dean@170
   168
                this.associatedMailItem = null;
Dean@170
   169
            }
Dean@170
   170
            else
Dean@170
   171
            {
Dean@170
   172
                // Check if the associated mail item has already been set
Dean@170
   173
                if (this.associatedMailItem != null)
Dean@170
   174
                {
Dean@170
   175
                    // Only re-set the mail item if the EntryID has changed
Dean@170
   176
                    if (this.associatedMailItem.EntryID != omi.EntryID)
Dean@170
   177
                    {
Dean@170
   178
                        this.associatedMailItem = new CryptableMailItem(omi);
Dean@170
   179
                    }
Dean@170
   180
                }
Dean@170
   181
                else
Dean@170
   182
                {
Dean@170
   183
                    this.associatedMailItem = new CryptableMailItem(omi);
Dean@170
   184
                }
Dean@170
   185
            }
Dean@170
   186
Dean@170
   187
            return;
Dean@170
   188
        }
Dean@170
   189
Dean@179
   190
        /// <summary>
Dean@672
   191
        /// Completes the handshake process when the identity of a partner was previously marked as mistrusted.
Dean@281
   192
        /// </summary>
Dean@646
   193
        /// <param name="myself">The personal identity to complete the handshake with.</param>
Dean@646
   194
        /// <param name="partner">The identity of the partner to complete the handshake with.</param>
Dean@672
   195
        private void DoHandshakeForMistrustedKey(PEPIdentity myself,
Dean@672
   196
                                                 PEPIdentity partner)
Dean@281
   197
        {
Dean@281
   198
            DialogResult result;
Dean@281
   199
Dean@281
   200
            result = MessageBox.Show(this.ParentForm,
Dean@672
   201
                                     pEp.Properties.Resources.Message_WarningMistrustedKey,
Dean@378
   202
                                     pEp.Properties.Resources.Message_TitleConfirmOperation,
Dean@423
   203
                                     MessageBoxButtons.YesNo);
Dean@281
   204
Dean@423
   205
            if (result == DialogResult.Yes)
Dean@281
   206
            {
Dean@646
   207
                this.DoHandshake(myself, partner);
Dean@281
   208
            }
Dean@281
   209
Dean@281
   210
            return;
Dean@281
   211
        }
Dean@281
   212
Dean@281
   213
        /// <summary>
Dean@281
   214
        /// Completes the handshake process where the identity of a partner is confirmed.
Dean@179
   215
        /// </summary>
Dean@646
   216
        /// <param name="myself">The personal identity to complete the handshake with.</param>
Dean@646
   217
        /// <param name="partner">The identity of the partner to complete the handshake with.</param>
Dean@646
   218
        private void DoHandshake(PEPIdentity myself,
Dean@646
   219
                                 PEPIdentity partner)
vb@133
   220
        {
Dean@155
   221
            string ownFpr;
Dean@155
   222
            string partnerFpr;
Dean@363
   223
            string trustwordsShort;
Dean@363
   224
            string trustwordsFull;
Dean@155
   225
            string[] lines;
Dean@179
   226
            DialogResult result;
Dean@155
   227
            FormHandshake handshakeDialog;
Dean@155
   228
            FormHandshake.State state = new FormHandshake.State();
markus@201
   229
Dean@660
   230
            Global.AddToLog("DoHandshake: handshake started.");
vb@139
   231
Dean@363
   232
            // Process identities
Dean@646
   233
            if (partner.Username == "")
Dean@363
   234
            {
Dean@646
   235
                partner.Username = partner.Address;
vb@133
   236
            }
Dean@646
   237
            partner.UserID = Globals.ThisAddIn.GetUserID(partner.Address);
Dean@363
   238
Dean@363
   239
            // Calculate and add trustwords
Dean@646
   240
            this.CalcTrustwords(myself, partner, out trustwordsShort, out trustwordsFull);
Dean@646
   241
            state.Myself = myself.Copy();
Dean@646
   242
            state.Partner = partner.Copy();
Dean@363
   243
            state.TrustwordsShort = trustwordsShort;
Dean@363
   244
            state.TrustwordsFull = trustwordsFull;
Dean@363
   245
Dean@363
   246
            // Calculate and add fingerprint
Dean@646
   247
            ownFpr = this.ToQuadruple(myself.Fingerprint);
Dean@646
   248
            partnerFpr = this.ToQuadruple(partner.Fingerprint);
vb@133
   249
Dean@646
   250
            if (myself.Fingerprint.CompareTo(partner.Fingerprint) > 0)
vb@133
   251
            {
Dean@155
   252
                lines = new string[3];
Dean@378
   253
                lines[0] = pEp.Properties.Resources.PrivacyStatus_TrustwordsPartnerShortPartner + partnerFpr;
Dean@378
   254
                lines[2] = pEp.Properties.Resources.PrivacyStatus_TrustwordsPartnerShortMyself + ownFpr;
Dean@155
   255
                state.Fingerprint = lines;
vb@133
   256
            }
vb@133
   257
            else
vb@133
   258
            {
Dean@155
   259
                lines = new string[3];
Dean@378
   260
                lines[0] = pEp.Properties.Resources.PrivacyStatus_TrustwordsOwnShortMyself + ownFpr;
Dean@378
   261
                lines[2] = pEp.Properties.Resources.PrivacyStatus_TrustwordsOwnShortPartner + partnerFpr;
Dean@155
   262
                state.Fingerprint = lines;
vb@133
   263
            }
vb@133
   264
Dean@155
   265
            // Create and show handshake dialog
Dean@155
   266
            handshakeDialog = new FormHandshake(state);
Dean@179
   267
            handshakeDialog.StartPosition = FormStartPosition.CenterScreen;
Dean@179
   268
            result = handshakeDialog.ShowDialog(this);
Dean@155
   269
Dean@646
   270
            this.ProcessDoHandshakeResult(result, partner);
Dean@356
   271
Dean@356
   272
            return;
Dean@356
   273
        }
Dean@356
   274
Dean@356
   275
        /// <summary>
Dean@356
   276
        /// Processes the result of the do handshake dialog after a user makes a selection.
Dean@356
   277
        /// </summary>
Dean@356
   278
        /// <param name="result">The result of the handshake dialog selection.</param>
Dean@356
   279
        /// <param name="partner">The identity of the partner to handshake with.</param>
Dean@356
   280
        private void ProcessDoHandshakeResult(DialogResult result,
Dean@384
   281
                                              PEPIdentity partner)
Dean@356
   282
        {
Dean@384
   283
            pEp_identity_s identityPartner = ThisAddIn.pEp.update_identity(partner.ToCOMType());
vb@133
   284
vb@133
   285
            switch (result)
vb@133
   286
            {
vb@133
   287
                case DialogResult.Yes:
Dean@279
   288
                    {
Dean@672
   289
                        // Check if key was previously mistrusted -- warning to user must be displayed earlier
Dean@672
   290
                        if (identityPartner.comm_type == _pEp_comm_type.pEp_ct_mistrusted)
Dean@281
   291
                        {
Dean@281
   292
                            ThisAddIn.pEp.key_reset_trust(ref identityPartner);
Dean@281
   293
                        }
Dean@281
   294
Dean@279
   295
                        identityPartner = ThisAddIn.pEp.trust_personal_key(ref identityPartner);
Dean@279
   296
Dean@279
   297
                        this.UpdateUIFromMailItem();
Dean@279
   298
Dean@279
   299
                        // Update the manager form state
Dean@279
   300
                        if (this.managerForm != null)
Dean@279
   301
                        {
Dean@279
   302
                            this.managerForm.CopyStateToUI(this.GetManagerState());
Dean@279
   303
                        }
Dean@279
   304
Dean@279
   305
                        break;
Dean@279
   306
                    }
vb@133
   307
                case DialogResult.No:
Dean@279
   308
                    {
Dean@279
   309
                        ThisAddIn.pEp.key_compromized(ref identityPartner);
Dean@279
   310
Dean@279
   311
                        this.UpdateUIFromMailItem();
Dean@279
   312
Dean@279
   313
                        // Update the manager form state
Dean@279
   314
                        if (this.managerForm != null)
Dean@279
   315
                        {
Dean@279
   316
                            this.managerForm.CopyStateToUI(this.GetManagerState());
Dean@279
   317
                        }
Dean@279
   318
Dean@279
   319
                        break;
Dean@279
   320
                    }
vb@133
   321
            }
Dean@279
   322
Dean@279
   323
            return;
vb@133
   324
        }
vb@133
   325
Dean@179
   326
        /// <summary>
Dean@179
   327
        /// Reverses any past handshake confirmation by unconfirming the given identity partner.
Dean@179
   328
        /// </summary>
Dean@384
   329
        /// <param name="partner">The identity of the partner to unconfirm.</param>
Dean@384
   330
        private void UndoHandshake(PEPIdentity partner)
vb@133
   331
        {
Dean@384
   332
            pEp_identity_s identityPartner = ThisAddIn.pEp.update_identity(partner.ToCOMType());
Dean@280
   333
            ThisAddIn.pEp.key_reset_trust(ref identityPartner);
Dean@279
   334
Dean@179
   335
            this.UpdateUIFromMailItem();
Dean@279
   336
Dean@279
   337
            // Update the manager form state
Dean@279
   338
            if (this.managerForm != null)
Dean@279
   339
            {
Dean@279
   340
                this.managerForm.CopyStateToUI(this.GetManagerState());
Dean@279
   341
            }
Dean@279
   342
Dean@279
   343
            return;
vb@133
   344
        }
vb@133
   345
Dean@161
   346
        /// <summary>
Dean@363
   347
        /// Calculates both the short and full trustwords between the given personal identity and the identity partner.
Dean@363
   348
        /// </summary>
Dean@646
   349
        /// <param name="myself">The personal identity.</param>
Dean@646
   350
        /// <param name="partner">The identity of the partner.</param>
Dean@363
   351
        /// <param name="trustwordsShort">The short version of trustwords.</param>
Dean@363
   352
        /// <param name="trustwordsFull">The full/long version of trustwords.</param>
Dean@646
   353
        private void CalcTrustwords(PEPIdentity myself,
Dean@646
   354
                                    PEPIdentity partner,
Dean@363
   355
                                    out string trustwordsShort,
Dean@363
   356
                                    out string trustwordsFull)
Dean@363
   357
        {
Dean@363
   358
            string myShort;
Dean@363
   359
            string myLong;
Dean@363
   360
            string partnerShort;
Dean@363
   361
            string partnerLong;
Dean@363
   362
Dean@674
   363
            myShort = ThisAddIn.pEp.trustwords(myself.Fingerprint, max_words: 5);
Dean@674
   364
            myLong = ThisAddIn.pEp.trustwords(myself.Fingerprint);
Dean@674
   365
            partnerShort = ThisAddIn.pEp.trustwords(partner.Fingerprint, max_words: 5);
Dean@674
   366
            partnerLong = ThisAddIn.pEp.trustwords(partner.Fingerprint);
Dean@363
   367
Dean@646
   368
            if (myself.Fingerprint.CompareTo(partner.Fingerprint) > 0)
Dean@363
   369
            {
Dean@730
   370
                trustwordsShort = partnerShort + Environment.NewLine + myShort;
Dean@730
   371
                trustwordsFull = partnerLong + Environment.NewLine + myLong;
Dean@363
   372
            }
Dean@363
   373
            else
Dean@363
   374
            {
Dean@730
   375
                trustwordsShort = myShort + Environment.NewLine + partnerShort;
Dean@730
   376
                trustwordsFull = myLong + Environment.NewLine + partnerLong;
Dean@363
   377
            }
Dean@363
   378
Dean@363
   379
            return;
Dean@363
   380
        }
Dean@363
   381
Dean@363
   382
        /// <summary>
Dean@161
   383
        /// Formats the given text string as separated 4-character groups.
Dean@161
   384
        /// Example: 49422235FC99585B891C --> 4942 2235 FC99 585B 891C
Dean@161
   385
        /// </summary>
Dean@161
   386
        /// <param name="text">The text to format in 4-character groups.</param>
Dean@161
   387
        /// <returns>The re-formatted string.</returns>
Dean@161
   388
        private string ToQuadruple(string text)
Dean@159
   389
        {
Dean@159
   390
            List<string> result = new List<string>();
Dean@159
   391
Dean@161
   392
            if (text != null)
Dean@159
   393
            {
Dean@161
   394
                for (int i = 0; i < text.Length; i += 4)
Dean@159
   395
                {
Dean@161
   396
                    try
Dean@161
   397
                    {
Dean@161
   398
                        result.Add(text.Substring(i, 4));
Dean@161
   399
                    }
Dean@161
   400
                    catch (ArgumentOutOfRangeException)
Dean@161
   401
                    {
Dean@161
   402
                        result.Add(text.Substring(i));
Dean@161
   403
                        break;
Dean@161
   404
                    }
Dean@159
   405
                }
Dean@159
   406
            }
Dean@159
   407
Dean@159
   408
            return String.Join(" ", result);
Dean@159
   409
        }
markus@201
   410
Dean@196
   411
        /// <summary>
Dean@279
   412
        /// Builds a new manager form state using this encryption state/mail item current state.
Dean@279
   413
        /// </summary>
Dean@279
   414
        /// <returns>A new manager form state.</returns>
Dean@313
   415
        private FormManagePrivacyStatus.State GetManagerState()
Dean@279
   416
        {
Dean@737
   417
            int currIndex = 0;
Dean@527
   418
            bool isMyself;
Dean@664
   419
            bool forceUnencrypted;
Dean@364
   420
            string trustwordsShort;
Dean@364
   421
            string trustwordsFull;
Dean@384
   422
            List<PEPIdentity> identities;
Dean@384
   423
            PEPIdentity myIdentity;
Dean@384
   424
            PEPIdentity incomingIdent;
Dean@259
   425
            _pEp_color identityPartnerColor;
Dean@259
   426
            SelectionItem item;
Dean@312
   427
            PrivacyState state = this.CopyUIToState();
Dean@313
   428
            FormManagePrivacyStatus.State managerState = new FormManagePrivacyStatus.State();
Dean@259
   429
Dean@332
   430
            // Resolve all recipients -- this ensures the identities list is correctly populated
Dean@332
   431
            this.associatedMailItem.ResolveAllRecipients();
Dean@332
   432
Dean@279
   433
            managerState.VisualState = state.Copy();
Dean@279
   434
            managerState.IsIncoming = this.associatedMailItem.IsIncoming;
Dean@259
   435
Dean@653
   436
            if (managerState.IsIncoming)
Dean@259
   437
            {
Dean@384
   438
                identities = new List<PEPIdentity>();
Dean@259
   439
Dean@259
   440
                // Add only one identity
Dean@638
   441
                if (this.associatedMailItem.From != null)
Dean@638
   442
                {
Dean@638
   443
                    incomingIdent = new PEPIdentity();
Dean@638
   444
                    incomingIdent.Address = this.associatedMailItem.From.Address;
Dean@638
   445
                    incomingIdent.Username = this.associatedMailItem.From.Username;
Dean@638
   446
                    incomingIdent.UserID = Globals.ThisAddIn.GetUserID(incomingIdent.Address);
Dean@259
   447
Dean@638
   448
                    identities.Add(incomingIdent);
Dean@638
   449
                }
Dean@259
   450
            }
Dean@259
   451
            else
Dean@259
   452
            {
Dean@395
   453
                identities = this.associatedMailItem.Recipients;
Dean@259
   454
            }
Dean@259
   455
Dean@366
   456
            // Get my identity
Dean@633
   457
            myIdentity = this.associatedMailItem.Myself;
Dean@366
   458
Dean@279
   459
            // Add identities
Dean@394
   460
            identities = PEPIdentity.ToFlatList(identities);
Dean@279
   461
            managerState.Identities.Clear();
Dean@384
   462
            foreach (PEPIdentity ident in identities)
Dean@259
   463
            {
Dean@737
   464
                // Needed so index can be used within anonymous methods without getting into the 'outer variable trap' of currIndex
Dean@737
   465
                int index = currIndex;
Dean@737
   466
Dean@648
   467
                if (ident.IsValid)
Dean@485
   468
                {
Dean@485
   469
                    pEp_identity_s partnerIdentity_s = ThisAddIn.pEp.update_identity(ident.ToCOMType());
Dean@485
   470
                    PEPIdentity partnerIdentity = new PEPIdentity(partnerIdentity_s);
Dean@485
   471
                    identityPartnerColor = ThisAddIn.pEp.identity_color(partnerIdentity_s);
Dean@664
   472
                    forceUnencrypted = Globals.ThisAddIn.ContactIsForceUnencrypted(partnerIdentity.Address);
Dean@259
   473
Dean@527
   474
                    // Check if identity is myself
Dean@713
   475
                    isMyself = PEPIdentity.GetIsMyIdentity(partnerIdentity.Address);
Dean@724
   476
Dean@485
   477
                    item = new SelectionItem();
Dean@485
   478
                    item.TextLine1 = partnerIdentity.Username;
Dean@485
   479
                    item.TextLine2 = partnerIdentity.Address;
Dean@259
   480
Dean@485
   481
                    // Don't show both the user name and address if they are the same
Dean@485
   482
                    if ((item.TextLine1 != null) &&
Dean@485
   483
                        (item.TextLine2 != null) &&
Dean@485
   484
                        (item.TextLine1 == item.TextLine2))
Dean@485
   485
                    {
Dean@485
   486
                        item.IsTwoTextLinesVisible = false;
Dean@485
   487
                    }
Dean@332
   488
Dean@527
   489
                    // Calculate trustwords
Dean@527
   490
                    this.CalcTrustwords(myIdentity,
Dean@527
   491
                                        partnerIdentity,
Dean@527
   492
                                        out trustwordsShort,
Dean@527
   493
                                        out trustwordsFull);
Dean@527
   494
Dean@485
   495
                    // Set image
Dean@664
   496
                    if ((forceUnencrypted) &&
Dean@653
   497
                        (managerState.IsIncoming == false))
Dean@485
   498
                    {
Dean@641
   499
                        // Show the force unencrypted image indicating this is set by the user not the engine
Dean@642
   500
                        item.ItemImage = pEp.Properties.Resources.ImageForceUnencOn;
Dean@641
   501
                    }
Dean@641
   502
                    else
Dean@641
   503
                    {
Dean@641
   504
                        switch (PrivacyState.ConvertRatingToPrivacyColor(identityPartnerColor))
Dean@641
   505
                        {
Dean@641
   506
                            case PrivacyState.PrivacyColor.Green:
Dean@641
   507
                                item.ItemImage = pEp.Properties.Resources.ImagePrivacyStatusGreen;
Dean@641
   508
                                break;
Dean@641
   509
                            case PrivacyState.PrivacyColor.Yellow:
Dean@641
   510
                                item.ItemImage = pEp.Properties.Resources.ImagePrivacyStatusYellow;
Dean@641
   511
                                break;
Dean@641
   512
                            case PrivacyState.PrivacyColor.Red:
Dean@641
   513
                                item.ItemImage = pEp.Properties.Resources.ImagePrivacyStatusRed;
Dean@641
   514
                                break;
Dean@641
   515
                            case PrivacyState.PrivacyColor.NoColor:
Dean@641
   516
                                item.ItemImage = pEp.Properties.Resources.ImagePrivacyStatusNoColor;
Dean@641
   517
                                break;
Dean@641
   518
                            default:
Dean@641
   519
                                item.ItemImage = null;
Dean@641
   520
                                break;
Dean@641
   521
                        }
Dean@485
   522
                    }
Dean@259
   523
Dean@485
   524
                    // Set button
Dean@653
   525
                    if ((isMyself) ||
Dean@664
   526
                        ((forceUnencrypted) &&
Dean@653
   527
                         (managerState.IsIncoming == false)))
Dean@653
   528
                    {
Dean@653
   529
                        item.IsButtonVisible = false;
Dean@653
   530
                        item.IsExpandable = false;
Dean@653
   531
                    }
Dean@653
   532
                    else
Dean@366
   533
                    {
Dean@485
   534
                        if (partnerIdentity.CommunicationType >= _pEp_comm_type.pEp_ct_confirmed_encryption)
Dean@485
   535
                        {
Dean@485
   536
                            // Undo handshake
Dean@485
   537
                            item.TextButton = pEp.Properties.Resources.PrivacyStatus_StopTrusting;
Dean@485
   538
                            item.IsButtonVisible = true;
Dean@646
   539
                            item.ButtonOnClick = (x, y) => { this.UndoHandshake(partnerIdentity.Copy()); };
Dean@356
   540
Dean@485
   541
                            item.IsExpandable = false;
Dean@485
   542
                        }
Dean@485
   543
                        else if (partnerIdentity.CommunicationType >= _pEp_comm_type.pEp_ct_unconfirmed_encryption &&
Dean@485
   544
                                 partnerIdentity.CommunicationType < _pEp_comm_type.pEp_ct_confirmed_encryption)
Dean@485
   545
                        {
Dean@485
   546
                            // Do handshake
Dean@737
   547
                            item.TextButton = pEp.Properties.Resources.PrivacyStatus_Handshake;
Dean@485
   548
                            item.IsButtonVisible = true;
Dean@646
   549
                            item.ButtonOnClick = (x, y) => { this.DoHandshake(myIdentity.Copy(), partnerIdentity.Copy()); };
Dean@356
   550
Dean@485
   551
                            item.IsExpandable = true;
Dean@485
   552
                            item.ExpandedText = Properties.Resources.PrivacyStatus_TrustwordDesc + "\n\n" +
Dean@485
   553
                                                trustwordsShort;
Dean@447
   554
Dean@485
   555
                            item.ExpandedButton1Text = Properties.Resources.PrivacyStatus_SelectTrust;
Dean@646
   556
                            item.ExpandedButton1OnClick = (x, y) => { this.ProcessDoHandshakeResult(DialogResult.Yes, partnerIdentity.Copy()); };
Dean@485
   557
                            item.ExpandedButton1BackColor = PrivacyState.COLOR_GREEN;
Dean@485
   558
                            item.ExpandedButton1ForeColor = Color.White;
Dean@447
   559
Dean@737
   560
                            item.ExpandedButton2Text = Properties.Resources.PrivacyStatus_SelectCancel;
Dean@737
   561
                            item.ExpandedButton2OnClick = (x, y) =>
Dean@737
   562
                                {
Dean@737
   563
                                    FormManagePrivacyStatus.State activeManagerState;
Dean@737
   564
Dean@737
   565
                                    if (this.managerForm != null)
Dean@737
   566
                                    {
Dean@737
   567
                                        activeManagerState = this.managerForm.CopyUIToState();
Dean@737
   568
                
Dean@737
   569
                                        if ((index >= 0) &&
Dean@737
   570
                                            (index < activeManagerState.Identities.Count))
Dean@737
   571
                                        {
Dean@737
   572
                                            item = activeManagerState.Identities[index];
Dean@737
   573
                                            item.TextButton   = pEp.Properties.Resources.PrivacyStatus_Handshake;
Dean@737
   574
                                            item.IsExpanded   = false;
Dean@737
   575
                                            item.IsExpandable = false;
Dean@737
   576
                                        }
Dean@737
   577
                                    }
Dean@737
   578
                                };
Dean@737
   579
Dean@737
   580
                            item.ExpandedButton3Text = Properties.Resources.PrivacyStatus_SelectMistrust;
Dean@737
   581
                            item.ExpandedButton3OnClick = (x, y) => { this.ProcessDoHandshakeResult(DialogResult.No, partnerIdentity.Copy()); };
Dean@737
   582
                            item.ExpandedButton3BackColor = PrivacyState.COLOR_RED;
Dean@737
   583
                            item.ExpandedButton3ForeColor = Color.White;
Dean@485
   584
                        }
Dean@672
   585
                        else if (partnerIdentity.CommunicationType == _pEp_comm_type.pEp_ct_mistrusted)
Dean@485
   586
                        {
Dean@485
   587
                            // Redo handshake with confirmation
Dean@485
   588
                            item.TextButton = pEp.Properties.Resources.PrivacyStatus_Handshake;
Dean@485
   589
                            item.IsButtonVisible = true;
Dean@672
   590
                            item.ButtonOnClick = (x, y) => { this.DoHandshakeForMistrustedKey(myIdentity.Copy(), partnerIdentity.Copy()); };
Dean@641
   591
Dean@641
   592
                            item.IsExpandable = false;
Dean@485
   593
                        }
Dean@485
   594
                        else
Dean@485
   595
                        {
Dean@485
   596
                            item.IsButtonVisible = false;
Dean@641
   597
                            item.IsExpandable = false;
Dean@485
   598
                        }
Dean@366
   599
                    }
Dean@281
   600
                }
Dean@485
   601
                else // Invalid identity
Dean@259
   602
                {
Dean@485
   603
                    item = new SelectionItem();
Dean@485
   604
                    item.TextLine1 = ident.Username;
Dean@485
   605
                    item.TextLine2 = ident.Address;
Dean@485
   606
                    item.ItemImage = null;
Dean@485
   607
                    item.IsTwoTextLinesVisible = true;
Dean@259
   608
                    item.IsButtonVisible = false;
Dean@259
   609
                }
Dean@259
   610
Dean@279
   611
                managerState.Identities.Add(item);
Dean@737
   612
                currIndex++;
Dean@259
   613
            }
Dean@259
   614
Dean@737
   615
            // Attempt to select and expand the first identity requiring a handshake
Dean@375
   616
            for (int i = 0; i < managerState.Identities.Count; i++)
Dean@375
   617
            {
Dean@737
   618
                // Determine if it's a handshake identity by checking if it's expandable
Dean@375
   619
                if (managerState.Identities[i].IsExpandable)
Dean@375
   620
                {
Dean@737
   621
                    managerState.Identities[i].TextButton = Properties.Resources.PrivacyStatus_HandshakeAdvanced;
Dean@737
   622
                    managerState.Identities[i].IsExpanded = true;
Dean@375
   623
                    managerState.SelectedIdentityIndex = i;
Dean@375
   624
                    break;
Dean@375
   625
                }
Dean@375
   626
            }
Dean@375
   627
Dean@279
   628
            return (managerState);
Dean@279
   629
        }
Dean@279
   630
Dean@279
   631
        /// <summary>
Dean@279
   632
        /// Builds the latest state of the encryption status manager then shows the UI.
Dean@279
   633
        /// </summary>
Dean@279
   634
        private void BuildAndShowManager()
Dean@279
   635
        {
Dean@279
   636
            DialogResult result;
Dean@313
   637
            FormManagePrivacyStatus form;
Dean@313
   638
            FormManagePrivacyStatus.State stateOut;
Dean@279
   639
Dean@483
   640
            try
Dean@483
   641
            {
Dean@483
   642
                // Show the form
Dean@483
   643
                form = new FormManagePrivacyStatus();
Dean@483
   644
                form.StartPosition = FormStartPosition.CenterScreen;
Dean@259
   645
Dean@483
   646
                this.managerForm = form;
Dean@483
   647
                result = form.ShowDialog(this.ParentForm, this.GetManagerState(), out stateOut); // Must show as dialog to block code
Dean@483
   648
            }
Dean@483
   649
            catch (Exception ex)
Dean@483
   650
            {
Dean@660
   651
                Global.StopAndSendCrashReport(ex);
Dean@483
   652
            }
Dean@259
   653
Dean@259
   654
            return;
Dean@259
   655
        }
Dean@259
   656
Dean@259
   657
        /// <summary>
Dean@179
   658
        /// Updates the status of the UI state based on the associated mail item.
Dean@179
   659
        /// </summary>
Dean@622
   660
        public void UpdateUIFromMailItem()
Dean@159
   661
        {
Dean@312
   662
            PrivacyState state;
Dean@664
   663
            bool? forceUnencryptedProperty;
Dean@159
   664
Dean@173
   665
            if (this.associatedMailItem != null)
Dean@159
   666
            {
Dean@664
   667
                forceUnencryptedProperty = this.associatedMailItem.ForceUnencrypted;
Dean@342
   668
Dean@664
   669
                if ((forceUnencryptedProperty != null) &&
Dean@664
   670
                    ((bool)forceUnencryptedProperty == true))
Dean@159
   671
                {
Dean@342
   672
                    // Force unencrypted
Dean@312
   673
                    state = new PrivacyState(_pEp_color.pEp_rating_unencrypted);
Dean@680
   674
                    this.CopyStateToUI(state);
Dean@159
   675
                }
Dean@159
   676
                else
Dean@159
   677
                {
Dean@680
   678
                    // Start the color rating calculation/decryption process
Dean@680
   679
                    this.associatedMailItem.StartDecryption();
Dean@159
   680
                }
Dean@159
   681
            }
Dean@159
   682
Dean@179
   683
            return;
Dean@159
   684
        }
Dean@159
   685
Dean@709
   686
        /// <summary>
Dean@709
   687
        /// Clears the associated unencrypted preview and displays the given note (if any).
Dean@709
   688
        /// </summary>
Dean@709
   689
        /// <param name="note">The note to diplsay.</param>
Dean@709
   690
        private void ClearPreview(string note = null)
Dean@709
   691
        {
Dean@709
   692
            WindowFormRegionCollection formRegions = Globals.FormRegions[Globals.ThisAddIn.Application.ActiveExplorer()];
Dean@709
   693
Dean@709
   694
            if ((formRegions != null) &&
Dean@709
   695
                (formRegions.FormRegionPreviewUnencrypted != null) &&
Dean@709
   696
                (formRegions.FormRegionPreviewUnencrypted.Visible))
Dean@709
   697
            {
Dean@709
   698
                formRegions.FormRegionPreviewUnencrypted.ClearMessage();
Dean@709
   699
                formRegions.FormRegionPreviewUnencrypted.SetNote(note);
Dean@709
   700
            }
Dean@709
   701
Dean@709
   702
            return;
Dean@709
   703
        }
Dean@709
   704
Dean@172
   705
        #region StateMethods
Dean@172
   706
Dean@172
   707
        /// <summary>
Dean@172
   708
        /// Connects or disconnects all control events from the UI.
Dean@172
   709
        /// </summary>
Dean@172
   710
        /// <param name="connect">True to connect events, false to disconnect.</param>
Dean@172
   711
        private void ConnectEvents(bool connect)
Dean@172
   712
        {
Dean@172
   713
            // Connect events only if not already connected
markus@201
   714
            if ((connect == true) &&
Dean@172
   715
                (this.eventsAreConnected == false))
Dean@172
   716
            {
Dean@312
   717
                this.ButtonPrivacyStatus.Click += this.ButtonPrivacyStatus_Click;
Dean@312
   718
                this.ButtonPrivacyStatus.MouseUp += this.ButtonPrivacyStatus_MouseUp;
Dean@179
   719
Dean@172
   720
                this.eventsAreConnected = true;
Dean@172
   721
            }
Dean@172
   722
            // Always attempt to disconnect
Dean@172
   723
            else if (connect == false)
Dean@172
   724
            {
Dean@312
   725
                this.ButtonPrivacyStatus.Click -= this.ButtonPrivacyStatus_Click;
Dean@312
   726
                this.ButtonPrivacyStatus.MouseUp -= this.ButtonPrivacyStatus_MouseUp;
Dean@179
   727
Dean@172
   728
                this.eventsAreConnected = false;
Dean@172
   729
            }
Dean@172
   730
Dean@172
   731
            return;
Dean@172
   732
        }
Dean@172
   733
Dean@172
   734
        /// <summary>
Dean@172
   735
        /// Copies the given state to the UI.
Dean@172
   736
        /// Events are turned off until the process is complete.
Dean@172
   737
        /// </summary>
Dean@172
   738
        /// <param name="state">The state to set to the UI.</param>
Dean@312
   739
        private void CopyStateToUI(PrivacyState state)
Dean@172
   740
        {
Dean@172
   741
            this.ConnectEvents(false);
markus@201
   742
Dean@172
   743
            ///////////////////////////////////////////////////////////
Dean@172
   744
            // Set UI data
Dean@172
   745
            ///////////////////////////////////////////////////////////
Dean@172
   746
Dean@179
   747
            // Save UI state maintained outside of controls
Dean@312
   748
            this.stateUIColorRating = state.PrivacyStatus;
Dean@179
   749
Dean@312
   750
            this.ButtonPrivacyStatus.BackColor = state.BackgroundColor;
Dean@312
   751
            this.ButtonPrivacyStatus.ForeColor = state.ForegroundColor;
Dean@312
   752
            this.ButtonPrivacyStatus.Text = state.ShortText;
Dean@172
   753
Dean@670
   754
            // Update the manager form state
Dean@670
   755
            if (this.managerForm != null)
Dean@670
   756
            {
Dean@670
   757
                this.managerForm.CopyStateToUI(this.GetManagerState());
Dean@670
   758
            }
Dean@670
   759
Dean@172
   760
            this.ConnectEvents(true);
Dean@172
   761
Dean@172
   762
            return;
Dean@172
   763
        }
Dean@172
   764
Dean@172
   765
        /// <summary>
Dean@172
   766
        /// Copies the UI to a new state.
Dean@172
   767
        /// </summary>
Dean@172
   768
        /// <returns>The state of the UI.</returns>
Dean@312
   769
        private PrivacyState CopyUIToState()
Dean@172
   770
        {
Dean@312
   771
            return (new PrivacyState(this.stateUIColorRating));
Dean@172
   772
        }
Dean@172
   773
Dean@172
   774
        #endregion
Dean@172
   775
Dean@159
   776
        /**************************************************************
Dean@159
   777
         * 
Dean@159
   778
         * Event Handling
Dean@159
   779
         * 
Dean@159
   780
         *************************************************************/
Dean@159
   781
Dean@159
   782
        /// <summary>
Dean@680
   783
        /// Event handler that is called when the form region is displayed.
Dean@680
   784
        /// This is called each time the form region looses then regains visiblity 
Dean@680
   785
        /// (for example an other email is selected then back to this one).
Dean@159
   786
        /// </summary>
Dean@312
   787
        private void FormRegionPrivacyStatus_FormRegionShowing(object sender, System.EventArgs e)
Dean@159
   788
        {
Dean@680
   789
            // Do not allow initialization more than once
Dean@680
   790
            if (initialized == false)
Dean@680
   791
            {
Dean@680
   792
                this.SetAssociatedMailItem();
Dean@173
   793
Dean@680
   794
                // Set background color for Office 2013
Dean@159
   795
                try
Dean@159
   796
                {
Dean@680
   797
                    if (Globals.ThisAddIn.Application.Version.Substring(0, 2) == "15")
Dean@159
   798
                    {
Dean@680
   799
                        if (this.IsWithinInspector())
Dean@680
   800
                        {
Dean@680
   801
                            this.BackColor = SystemColors.ButtonFace;
Dean@680
   802
                        }
Dean@680
   803
                        else
Dean@680
   804
                        {
Dean@680
   805
                            this.BackColor = SystemColors.Window;
Dean@680
   806
                        }
Dean@159
   807
                    }
Dean@159
   808
                }
Dean@159
   809
                catch { }
Dean@680
   810
Dean@680
   811
                // Connect cryptable mail item events
Dean@680
   812
                if (this.associatedMailItem != null)
Dean@680
   813
                {
Dean@680
   814
                    try
Dean@680
   815
                    {
Dean@680
   816
                        this.associatedMailItem.PropertyChanged += MailItem_PropertyChanged;
Dean@680
   817
                        this.associatedMailItem.DecryptionComplete += MailItem_DecryptionComplete;
Dean@709
   818
                        this.associatedMailItem.GetMirrorComplete += MailItem_GetMirrorComplete;
Dean@682
   819
                        this.associatedMailItem.Send += MailItem_Send;
Dean@680
   820
Dean@680
   821
                        if (this.associatedMailItem.IsInEncryptedStore &&
Dean@680
   822
                            this.associatedMailItem.IsPGPEncrypted)
Dean@680
   823
                        {
Dean@680
   824
                            this.associatedMailItem.Open += MailItem_Open;
Dean@680
   825
                        }
Dean@680
   826
                    }
Dean@680
   827
                    catch { }
Dean@680
   828
                }
Dean@680
   829
Dean@680
   830
                this.CopyStateToUI(new PrivacyState());
Dean@683
   831
                this.TimerRefresh.Tick += TimerRefresh_Tick;
Dean@680
   832
                this.initialized = true;
Dean@179
   833
            }
Dean@159
   834
Dean@683
   835
            // Call the timer tick method manually to refresh data with no delay
Dean@199
   836
            this.TimerRefresh_Tick(null, new EventArgs());
Dean@179
   837
Dean@179
   838
            return;
Dean@159
   839
        }
Dean@159
   840
Dean@159
   841
        /// <summary>
Dean@159
   842
        /// Event handler for when the form region is closed.
Dean@159
   843
        /// </summary>
Dean@312
   844
        private void FormRegionPrivacyStatus_FormRegionClosed(object sender, System.EventArgs e)
Dean@159
   845
        {
Dean@199
   846
            // Disconnect cryptable mail item events
Dean@190
   847
            if (this.associatedMailItem != null)
Dean@179
   848
            {
Dean@179
   849
                try
Dean@179
   850
                {
Dean@445
   851
                    this.associatedMailItem.PropertyChanged -= MailItem_PropertyChanged;
Dean@680
   852
                    this.associatedMailItem.DecryptionComplete -= MailItem_DecryptionComplete;
Dean@709
   853
                    this.associatedMailItem.GetMirrorComplete -= MailItem_GetMirrorComplete;
markus@201
   854
                    this.associatedMailItem.Open -= MailItem_Open;
Dean@682
   855
                    this.associatedMailItem.Send -= MailItem_Send;
Dean@179
   856
                }
Dean@179
   857
                catch { }
Dean@179
   858
            }
Dean@179
   859
Dean@682
   860
            // Stop and disconnect the refresh timer
Dean@682
   861
            this.TimerRefresh.Stop();
Dean@682
   862
            this.TimerRefresh.Enabled = false;
Dean@682
   863
            this.TimerRefresh.Tick -= TimerRefresh_Tick;
Dean@682
   864
Dean@179
   865
            return;
Dean@179
   866
        }
Dean@179
   867
Dean@179
   868
        /// <summary>
Dean@179
   869
        /// Event handler called after the refresh timer has elapsed.
Dean@179
   870
        /// </summary>
Dean@179
   871
        private void TimerRefresh_Tick(object sender, EventArgs e)
Dean@179
   872
        {
Dean@199
   873
            bool tryAgain = false;
Dean@684
   874
            bool markForDownload = false;
Dean@179
   875
            this.TimerRefresh.Enabled = false; // Only once
Dean@684
   876
            Outlook.OlDownloadState dlState;
markus@201
   877
Dean@709
   878
            /* The Refresh/UI_Update process is a little more complicated here than other forms.
Dean@709
   879
             * There are the following components:
Dean@709
   880
             *   1. TimerRefresh_Tick
Dean@709
   881
             *        This is the main timer tick event handler called any time
Dean@709
   882
             *        a refresh was requested and hasn't been run yet. A refresh
Dean@709
   883
             *        is requested either at initialization or when a property changes
Dean@709
   884
             *        (such as MailItem_PropertyChanged). It is on a timer as many 
Dean@709
   885
             *        property change events could occur rapidy, but only one refresh should
Dean@709
   886
             *        occur for performance reasons.
Dean@709
   887
             * 
Dean@709
   888
             *   2. UpdateUIFromMailItem
Dean@709
   889
             *        This will re-calculate the mail item's color rating and then update the UI.
Dean@709
   890
             *        However, if a mail item is not forcefully unencrypted, it will just call StartDecryption.
Dean@709
   891
             * 
Dean@709
   892
             *   3. MailItem_DecryptionComplete
Dean@709
   893
             *        When decryption of the mail item is complete, this even handler will be called.
Dean@709
   894
             *        This will update the UI with the latest color rating then call StartGetMirror.
Dean@709
   895
             *        The CopyStateToUI method is used which means any open privacy status form will also
Dean@709
   896
             *        be updated.
Dean@709
   897
             * 
Dean@709
   898
             *   4. MailItem_GetMirrorComplete
Dean@709
   899
             *        This is the final step in updating the UI, after a mirror is located, it's contents
Dean@709
   900
             *        will be shown to the unencrypted preview.
Dean@709
   901
             * 
Dean@709
   902
             * The general calling sequence is as shown above 1->4 with each component calling the next.
Dean@709
   903
             * However, for methods that update the mail item directly, commonly only 2->4 is needed.
Dean@709
   904
             */
Dean@709
   905
Dean@179
   906
            // Ensure the tick method is not called more than once
Dean@179
   907
            if (refreshOngoing == false)
Dean@179
   908
            {
Dean@179
   909
                this.refreshOngoing = true;
Dean@179
   910
Dean@179
   911
                if (this.associatedMailItem != null)
Dean@179
   912
                {
Dean@684
   913
                    // Attempt to get the download state
Dean@684
   914
                    try
Dean@684
   915
                    {
Dean@684
   916
                        dlState = this.associatedMailItem.DownloadState;
Dean@684
   917
                    }
Dean@684
   918
                    catch (Exception ex)
Dean@684
   919
                    {
Dean@684
   920
                        Global.AddToLog("TimerRefresh_Tick: Get DownloadState failed, " + ex.ToString());
Dean@684
   921
Dean@684
   922
                        // Assume everything is downloaded, but try to download again as well
Dean@684
   923
                        dlState = Outlook.OlDownloadState.olFullItem;
Dean@684
   924
                        markForDownload = true;
Dean@684
   925
                    }
Dean@684
   926
Dean@684
   927
                    if (dlState == Outlook.OlDownloadState.olFullItem)
Dean@179
   928
                    {
Dean@709
   929
                        // Clear any existing preview and update
Dean@709
   930
                        this.ClearPreview(Properties.Resources.Message_Opening);
Dean@179
   931
                        this.UpdateUIFromMailItem();
Dean@179
   932
                    }
Dean@179
   933
                    else
Dean@179
   934
                    {
Dean@684
   935
                        markForDownload = true;
Dean@684
   936
                    }
Dean@684
   937
Dean@684
   938
                    if (markForDownload)
Dean@684
   939
                    {
Dean@684
   940
                        // Try to mark the message for full download
Dean@684
   941
                        try
Dean@684
   942
                        {
Dean@684
   943
                            this.associatedMailItem.MarkForDownload = Outlook.OlRemoteStatus.olMarkedForDownload;
Dean@684
   944
                            tryAgain = true;
Dean@684
   945
                        }
Dean@684
   946
                        catch (Exception ex)
Dean@684
   947
                        {
Dean@684
   948
                            Global.AddToLog("TimerRefresh_Tick: MarkForDownload failed, " + ex.ToString());
Dean@684
   949
                        }
Dean@179
   950
                    }
Dean@179
   951
                }
Dean@179
   952
Dean@199
   953
                // Set the timer to refresh again later automatically
Dean@199
   954
                if (tryAgain)
Dean@199
   955
                {
Dean@199
   956
                    this.TimerRefresh.Interval = 100;
Dean@199
   957
                    this.TimerRefresh.Enabled = true;
Dean@199
   958
                }
Dean@199
   959
Dean@179
   960
                this.refreshOngoing = false;
Dean@179
   961
            }
Dean@175
   962
Dean@159
   963
            return;
Dean@159
   964
        }
Dean@159
   965
Dean@169
   966
        /// <summary>
Dean@680
   967
        /// Event handler for when the decryption process is completed in the associated mail item.
Dean@709
   968
        /// This will then update the form region UI and the privacy status window as needed.
Dean@680
   969
        /// </summary>
Dean@704
   970
        private void MailItem_DecryptionComplete(object sender, CryptableMailItem.DecryptionEventArgs e)
Dean@680
   971
        {
Dean@680
   972
            try
Dean@680
   973
            {
Dean@680
   974
                // Marshal code back to UI thread as necessary
Dean@680
   975
                this.Invoke(new Action(() =>
Dean@680
   976
                    {
Dean@680
   977
                        this.CopyStateToUI(new PrivacyState(e.ColorRating));
Dean@709
   978
Dean@709
   979
                        /* Create the unencrypted preview if the mail item is encrypted and
Dean@709
   980
                         * it is in an encrypted (untrusted) store
Dean@709
   981
                         * 
Dean@709
   982
                         * This is done here because FormRegionPrivacyStatus has the cryptable mail item and
Dean@709
   983
                         * it also is initialized after FormRegionPreviewUnencrypted.
Dean@709
   984
                         */
Dean@709
   985
                        if (this.associatedMailItem.IsInEncryptedStore &&
Dean@709
   986
                            this.associatedMailItem.IsPGPEncrypted)
Dean@709
   987
                        {
Dean@709
   988
                            this.associatedMailItem.StartGetMirror();
Dean@709
   989
                        }
Dean@719
   990
                        else
Dean@719
   991
                        {
Dean@719
   992
                            this.ClearPreview();
Dean@719
   993
                        }
Dean@680
   994
                    }));
Dean@680
   995
            }
Dean@680
   996
            catch (Exception ex)
Dean@680
   997
            {
Dean@680
   998
                Global.AddToLog("MailItem_DecryptionComplete: Error setting UI state, " + ex.ToString());
Dean@680
   999
            }
Dean@680
  1000
Dean@680
  1001
            return;
Dean@680
  1002
        }
Dean@680
  1003
Dean@680
  1004
        /// <summary>
Dean@709
  1005
        /// Event handler for when the get mirror locating process is complete for the associated mail item.
Dean@709
  1006
        /// This will then update the unencrypted preview in the UI.
Dean@709
  1007
        /// </summary>
Dean@709
  1008
        private void MailItem_GetMirrorComplete(object sender, CryptableMailItem.GetMirrorEventArgs e)
Dean@709
  1009
        {
Dean@709
  1010
            try
Dean@709
  1011
            {
Dean@709
  1012
                // Marshal code back to UI thread as necessary
Dean@709
  1013
                this.Invoke(new Action(() =>
Dean@709
  1014
                    {
Dean@709
  1015
                        WindowFormRegionCollection formRegions = Globals.FormRegions[Globals.ThisAddIn.Application.ActiveExplorer()];
Dean@709
  1016
Dean@709
  1017
                        if ((formRegions != null) &&
Dean@709
  1018
                            (formRegions.FormRegionPreviewUnencrypted != null) &&
Dean@709
  1019
                            (formRegions.FormRegionPreviewUnencrypted.Visible))
Dean@709
  1020
                        {
Dean@709
  1021
                            formRegions.FormRegionPreviewUnencrypted.SetMessage(e.Mirror);
Dean@709
  1022
                        }
Dean@709
  1023
                    }));
Dean@709
  1024
            }
Dean@709
  1025
            catch (Exception ex)
Dean@709
  1026
            {
Dean@709
  1027
                Global.AddToLog("MailItem_GetMirrorComplete: Error displaying preview, " + ex.ToString());
Dean@709
  1028
            }
Dean@709
  1029
Dean@709
  1030
            return;
Dean@709
  1031
        }
Dean@709
  1032
Dean@709
  1033
        /// <summary>
Dean@169
  1034
        /// Event handler for when a mail item is being opened in an inspector.
Dean@180
  1035
        /// See: https://msdn.microsoft.com/en-us/library/office/ff865989.aspx
Dean@169
  1036
        /// </summary>
Dean@169
  1037
        /// <param name="cancel">Whether to cancel the event: Value is False when the event occurs. 
Dean@169
  1038
        /// If the event procedure sets this argument to True, the open operation is not completed 
Dean@169
  1039
        /// and the inspector is not displayed.</param>
Dean@169
  1040
        private void MailItem_Open(ref bool cancel)
Dean@159
  1041
        {
Dean@720
  1042
            if (this.associatedMailItem != null && this.associatedMailItem.DisplayMirror())
Dean@169
  1043
            {
Dean@169
  1044
                cancel = true;
Dean@169
  1045
            }
Dean@169
  1046
Dean@169
  1047
            return;
Dean@159
  1048
        }
Dean@159
  1049
Dean@169
  1050
        /// <summary>
Dean@682
  1051
        /// Event handler for when a mail item is sent.
Dean@682
  1052
        /// See: https://msdn.microsoft.com/en-us/library/office/ff865379.aspx
Dean@682
  1053
        /// </summary>
Dean@682
  1054
        /// <param name="cancel">Whether to cancel the event: Value is False when the event occurs. 
Dean@682
  1055
        /// If the event procedure sets this argument to True, the send operation is not completed 
Dean@682
  1056
        /// and the inspector is left open.</param>
Dean@682
  1057
        private void MailItem_Send(ref bool cancel)
Dean@682
  1058
        {
Dean@682
  1059
            // Stop and disconnect the refresh timer
Dean@682
  1060
            // This is necessary so an ongoing refresh doesn't try to access a mail item as it's being moved
Dean@682
  1061
            this.TimerRefresh.Stop();
Dean@682
  1062
            this.TimerRefresh.Enabled = false;
Dean@682
  1063
            this.TimerRefresh.Tick -= TimerRefresh_Tick;
Dean@682
  1064
Dean@682
  1065
            return;
Dean@682
  1066
        }
Dean@682
  1067
Dean@682
  1068
        /// <summary>
Dean@169
  1069
        /// Event handler for when a mail item property is changed.
Dean@169
  1070
        /// See: https://msdn.microsoft.com/en-us/library/office/ff866739.aspx
Dean@169
  1071
        /// </summary>
Dean@169
  1072
        /// <param name="propertyName">The name of the property that was changed.</param>
Dean@445
  1073
        private void MailItem_PropertyChanged(object sender, PropertyChangedEventArgs e)
Dean@159
  1074
        {
Dean@445
  1075
            switch (e.PropertyName.ToUpper())
Dean@159
  1076
            {
Dean@179
  1077
                case "TO":
Dean@179
  1078
                    // Start the refresh timer
Dean@179
  1079
                    this.TimerRefresh.Enabled = true;
Dean@159
  1080
                    break;
Dean@159
  1081
                // Outlook bug: there are always both events, so one is enough
Dean@159
  1082
                //case "CC":
Dean@179
  1083
                //    // Start the refresh timer
Dean@179
  1084
                //    this.TimerRefresh.Enabled = true;
Dean@159
  1085
                //    break;
Dean@159
  1086
            }
Dean@169
  1087
Dean@169
  1088
            return;
Dean@159
  1089
        }
Dean@159
  1090
Dean@179
  1091
        /// <summary>
Dean@312
  1092
        /// Event handler for when the privacy status button is clicked.
Dean@179
  1093
        /// </summary>
Dean@312
  1094
        private void ButtonPrivacyStatus_Click(object sender, EventArgs e)
vb@133
  1095
        {
Dean@310
  1096
            this.BuildAndShowManager();
Dean@179
  1097
            return;
vb@133
  1098
        }
vb@133
  1099
Dean@179
  1100
        /// <summary>
Dean@312
  1101
        /// Event handler for when a mouse button is released over the privacy status button.
Dean@179
  1102
        /// </summary>
Dean@312
  1103
        private void ButtonPrivacyStatus_MouseUp(object sender, MouseEventArgs e)
vb@133
  1104
        {
Dean@179
  1105
            return;
vb@133
  1106
        }
vb@133
  1107
    }
vb@133
  1108
}