Click here to Skip to main content
15,897,371 members
Articles / Desktop Programming / Windows Forms

i00 Spell Check and Control Extensions - No Third Party Components Required!

Rate me:
Please Sign up or sign in to vote.
4.95/5 (117 votes)
11 Jan 2014Ms-PL16 min read 1.4M   22   266  
Simple to use, open source Spell Checker for .NET
'i00 .Net Spell Check Data Grid View
'©i00 Productions All rights reserved
'Created by Kris Bennett
'----------------------------------------------------------------------------------------------------
'All property in this file is and remains the property of i00 Productions, regardless of its usage,
'unless stated otherwise in writing from i00 Productions.
'
'Anyone wishing to use this code in their projects may do so, however are required to leave a post on
'VBForums (under: http://www.vbforums.com/showthread.php?p=4075093) stating that they are doing so.
'A simple "I am using i00 Spell check in my project" will surffice.
'
'i00 is not and shall not be held accountable for any damages directly or indirectly caused by the
'use or miss-use of this product.  This product is only a component and thus is intended to be used 
'as part of other software, it is not a complete software package, thus i00 Productions is not 
'responsible for any legal ramifications that software using this product breaches.

Partial Public Class SpellCheckDataGridView
    Inherits SpellCheckControlBase
    Implements iTestHarness

#Region "Setup"

    'Called when the control is loaded
    Overrides Sub Load()
        Dim DataGridView = TryCast(Me.Control, DataGridView)
        If DataGridView IsNot Nothing Then
            'Add control specific event handlers
            AddHandler DataGridView.CellPainting, AddressOf DataGridView_CellPainting
            RepaintControl()
        End If
    End Sub

    'Lets the EnableSpellCheck() know what ControlTypes we can spellcheck
    Public Overrides ReadOnly Property ControlTypes() As IEnumerable(Of System.Type)
        Get
            Return New System.Type() {GetType(DataGridView)}
        End Get
    End Property

    'Repaint control when settings are changed
    Private Sub SpellCheckDataGridView_SettingsChanged(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.SettingsChanged
        RepaintControl()
    End Sub

#End Region

#Region "Underlying Control"

    'Make underlying control appear nicer in property grids
    <System.ComponentModel.Category("Control")> _
    <System.ComponentModel.Description("The DataGridView associated with the SpellCheckDataGridView object")> _
    <System.ComponentModel.DisplayName("Data Grid View")> _
    Public Overrides ReadOnly Property Control() As System.Windows.Forms.Control
        Get
            Return MyBase.Control
        End Get
    End Property

    'Quick access to underlying DataGridView object
    Private ReadOnly Property parentDataGridView() As DataGridView
        Get
            Return TryCast(MyBase.Control, DataGridView)
        End Get
    End Property

#End Region

#Region "Painting"

    Public Overrides Sub RepaintControl()
        If parentDataGridView IsNot Nothing Then
            parentDataGridView.Invalidate()
        End If
    End Sub

    Private Sub DataGridView_CellPainting(ByVal sender As Object, ByVal e As System.Windows.Forms.DataGridViewCellPaintingEventArgs)
        If OKToDraw = False Then Exit Sub

        Dim NewWords As New Dictionary(Of String, Dictionary.SpellCheckWordError)
        Dim oSender = TryCast(sender, DataGridView)
        If e.RowIndex >= 0 AndAlso e.ColumnIndex >= 0 AndAlso oSender IsNot Nothing Then

            If TypeOf oSender.Columns(e.ColumnIndex) Is DataGridViewTextBoxColumn AndAlso oSender.Columns(e.ColumnIndex).ReadOnly = False AndAlso e.FormattedValue IsNot Nothing Then

                Dim CellBounds = New Rectangle(e.CellBounds.X, e.CellBounds.Y, e.CellBounds.Width - 1, e.CellBounds.Height - 1)
                e.Handled = True
                e.Graphics.SmoothingMode = Drawing2D.SmoothingMode.HighQuality

                If (e.PaintParts And DataGridViewPaintParts.Background) = DataGridViewPaintParts.Background Then
                    e.PaintBackground(e.ClipBounds, True)
                End If
                e.Graphics.SetClip(CellBounds)

                If (e.PaintParts And DataGridViewPaintParts.ContentForeground) = DataGridViewPaintParts.ContentForeground Then

                  

                    e.Graphics.TranslateTransform(CellBounds.Left, CellBounds.Top)

                    'spell check bit :D
                    'changed words makes words like "kris'" > "kris'"
                    Dim ChangedWordText = Replace(e.FormattedValue.ToString, vbCrLf, vbCr)
                    ChangedWordText = Replace(ChangedWordText, vbLf, vbCr)
                    Dim ChangedWords = Dictionary.Formatting.RemoveWordBreaks(ChangedWordText)

                    Dim Flags = DrawingFunctions.Text.TextRendererMeasure.TextFlagsFrom(e.CellStyle)

                    Dim WordBounds = DrawingFunctions.Text.TextRendererMeasure.Measure(e.FormattedValue.ToString, e.CellStyle.Font, CellBounds.Size, Flags)
                    Dim NewWordBounds As New DrawingFunctions.Text.TextRendererMeasure.WordBounds With {.TextMargin = WordBounds.TextMargin}

                    Dim FalseFlags = Flags
                    Dim RemoveFlags() As TextFormatFlags = {TextFormatFlags.Top, TextFormatFlags.Left, TextFormatFlags.Bottom, TextFormatFlags.Right, TextFormatFlags.HorizontalCenter, TextFormatFlags.VerticalCenter}
                    For Each item In RemoveFlags
                        If (FalseFlags And item) = item Then
                            FalseFlags = FalseFlags Xor item
                        End If
                    Next

                    For Each item In WordBounds
                        'use the changed word instead :)
                        item.Word = ChangedWords.Substring(item.LetterIndex, item.Word.Length).Trim

                        'Dim Newwords = Split(item.Word, " ")

                        If item.Word.Contains(" ") Then
                            'need to break this word into parts :(

                            Dim NewSubWordBounds = DrawingFunctions.Text.TextRendererMeasure.Measure(item.Word, e.CellStyle.Font, New Size(Integer.MaxValue, Integer.MaxValue), FalseFlags)
                            For Each iNewSubWordBounds In NewSubWordBounds
                                iNewSubWordBounds.Bounds.X += item.Bounds.X - WordBounds.TextMargin
                                iNewSubWordBounds.Bounds.Y += item.Bounds.Y
                            Next
                            NewWordBounds.AddRange(NewSubWordBounds)
                        Else
                            'only one word here
                            NewWordBounds.Add(item)
                        End If

                    Next

                    For Each item In NewWordBounds

                        Dim WordState As Dictionary.SpellCheckWordError = Dictionary.SpellCheckWordError.SpellError
                        If dictCache.ContainsKey(item.Word) Then
                            'load from cache
                            WordState = dictCache(item.Word)
                        Else
                            If NewWords.ContainsKey(item.Word) = False Then
                                NewWords.Add(item.Word, Dictionary.SpellCheckWordError.OK)
                            End If

                            WordState = Dictionary.SpellCheckWordError.OK
                        End If

                        Dim eArgs = New SpellCheckCustomPaintEventArgs With {.Graphics = e.Graphics, .Word = item.Word, .Bounds = item.Bounds, .WordState = WordState}
                        OnSpellCheckErrorPaint(eArgs)
                        If eArgs.DrawDefault Then
                            Select Case WordState
                                Case Dictionary.SpellCheckWordError.Ignore
                                    If DrawIgnored() Then
                                        Using p As New Pen(Settings.IgnoreColor)
                                            e.Graphics.DrawLine(p, item.Bounds.X, item.Bounds.Bottom + 1, item.Bounds.Right, item.Bounds.Bottom + 1)
                                        End Using
                                    End If
                                Case Dictionary.SpellCheckWordError.CaseError
                                    DrawingFunctions.DrawWave(e.Graphics, New Point(item.Bounds.X, item.Bounds.Bottom), New Point(item.Bounds.Right, item.Bounds.Bottom), Settings.CaseMistakeColor)
                                Case Dictionary.SpellCheckWordError.SpellError
                                    DrawingFunctions.DrawWave(e.Graphics, New Point(item.Bounds.X, item.Bounds.Bottom), New Point(item.Bounds.Right, item.Bounds.Bottom), Settings.MistakeColor)
                            End Select
                        End If
                    Next
                    'end spell check bit :D

                    e.Graphics.ResetTransform()
                    Dim CellForeColor = e.CellStyle.ForeColor
                    If (e.State And DataGridViewElementStates.Selected) = DataGridViewElementStates.Selected Then
                        CellForeColor = e.CellStyle.SelectionForeColor
                    End If
                    TextRenderer.DrawText(e.Graphics, e.FormattedValue.ToString, e.CellStyle.Font, CellBounds, CellForeColor, Flags)
                End If

                e.Graphics.ResetClip()
            End If
        End If

        If NewWords.Count > 0 Then
            AddWordsToCache(NewWords)
        End If

    End Sub

#End Region

#Region "Test Harness"

    Private Class GridViewData
        Dim mc_GridViewExample As String
        <System.ComponentModel.DisplayName("Data Grid View Example")> _
        Public Property GridViewExample() As String
            Get
                Return mc_GridViewExample
            End Get
            Set(ByVal value As String)
                mc_GridViewExample = value
            End Set
        End Property
        Public Sub New(ByVal GridViewExample As String)
            Me.mc_GridViewExample = GridViewExample
        End Sub
        Public Sub New()

        End Sub
    End Class

    Public Function SetupControl(ByVal Control As System.Windows.Forms.Control) As Control Implements iTestHarness.SetupControl
        Dim DataGridView = TryCast(Control, DataGridView)
        If DataGridView IsNot Nothing Then

            Dim pnlHolder As New Panel
            Dim BindingNavigator As New BindingNavigator

            Dim GridData As New System.ComponentModel.BindingList(Of GridViewData)
            GridData.Add(New GridViewData("This is a grid view example to demonistrate that i00 Spell Check can be used in grids!"))
            GridData.Add(New GridViewData("So comeon and edit a cell!"))
            DataGridView.DefaultCellStyle.WrapMode = DataGridViewTriState.False
            DataGridView.Dock = DockStyle.Fill
            DataGridView.BorderStyle = BorderStyle.None

            AddHandler DataGridView.VisibleChanged, AddressOf TestHarness_VisibleChanged

            BindingNavigator.Dock = DockStyle.Bottom
            BindingNavigator.AddStandardItems()

            Dim bs As New BindingSource
            bs.DataSource = GridData
            bs.AllowNew = True
            DataGridView.DataSource = bs
            BindingNavigator.BindingSource = bs

            pnlHolder.Controls.Add(DataGridView)
            pnlHolder.Controls.Add(BindingNavigator)

            Return pnlHolder
        Else
            Return Nothing
        End If
    End Function

    Private Sub TestHarness_VisibleChanged(ByVal sender As System.Object, ByVal e As EventArgs)
        Dim DataGridView = TryCast(Control, DataGridView)
        If DataGridView IsNot Nothing AndAlso DataGridView.Visible = True Then
            DataGridView.AutoResizeColumns()
        End If
    End Sub

#End Region

End Class

By viewing downloads associated with this article you agree to the Terms of Service and the article's licence.

If a file you wish to view isn't highlighted, and is a text file (not binary), please let us know and we'll add colourisation support for it.

License

This article, along with any associated source code and files, is licensed under The Microsoft Public License (Ms-PL)


Written By
i00
Software Developer (Senior) i00 Productions
Australia Australia
I hope you enjoy my code. It's yours to use for free, but if you do wish to say thank you then a donation is always appreciated.
You can donate here.

Comments and Discussions