wordbreaker.c 5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
/*
 *    Word splitter Implementation
 *
 * Copyright 2006 Mike McCormack
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
 */

#define COBJMACROS


#include <stdarg.h>

#include "windef.h"
#include "winbase.h"
#include "winuser.h"
#include "ole2.h"
30
#include "indexsrv.h"
31 32 33 34 35 36
#include "wine/debug.h"

WINE_DEFAULT_DEBUG_CHANNEL(infosoft);

typedef struct tag_wordbreaker_impl
{
37
    IWordBreaker IWordBreaker_iface;
38 39 40
    LONG ref;
} wordbreaker_impl;

41 42 43 44 45
static inline wordbreaker_impl *impl_from_IWordBreaker(IWordBreaker *iface)
{
    return CONTAINING_RECORD(iface, wordbreaker_impl, IWordBreaker_iface);
}

46 47 48
static HRESULT WINAPI wb_QueryInterface( IWordBreaker *iface,
        REFIID riid, LPVOID *ppvObj)
{
49
    wordbreaker_impl *This = impl_from_IWordBreaker(iface);
50 51 52 53 54 55 56 57

    TRACE("(%p)->(%s)\n",This,debugstr_guid(riid));

    *ppvObj = NULL;

    if (IsEqualIID(riid, &IID_IUnknown) ||
        IsEqualIID(riid, &IID_IWordBreaker))
    {
58
        *ppvObj = &This->IWordBreaker_iface;
59 60 61 62 63 64 65 66 67
        return S_OK;
    }

    TRACE("-- E_NOINTERFACE\n");
    return E_NOINTERFACE;
}

static ULONG WINAPI wb_AddRef( IWordBreaker *iface )
{
68
    wordbreaker_impl *This = impl_from_IWordBreaker(iface);
69 70 71 72 73
    return InterlockedIncrement(&This->ref);
}

static ULONG WINAPI wb_Release(IWordBreaker *iface)
{
74
    wordbreaker_impl *This = impl_from_IWordBreaker(iface);
75 76 77 78 79 80 81 82 83 84 85 86
    LONG refcount;

    refcount = InterlockedDecrement(&This->ref);
    if (!refcount)
        HeapFree(GetProcessHeap(), 0, This);

    return refcount;
}

static HRESULT WINAPI wb_Init( IWordBreaker *iface,
        BOOL fQuery, ULONG ulMaxTokenSize, BOOL *pfLicense )
{
87
    TRACE("%d %u\n", fQuery, ulMaxTokenSize);
88 89 90 91
    *pfLicense = FALSE;
    return S_OK;
}

92
static HRESULT call_sink( IWordSink *pWordSink, TEXT_SOURCE *ts, UINT len )
93
{
94
    HRESULT r;
95

96 97
    if (!len)
        return S_OK;
98

99
    TRACE("%d %s\n", len, debugstr_w(&ts->awcBuffer[ts->iCur]));
100

101 102 103 104 105
    r = IWordSink_PutWord( pWordSink, len, &ts->awcBuffer[ts->iCur], len, ts->iCur );
    ts->iCur += len;

    return r;
}
106

107 108 109 110 111
static HRESULT WINAPI wb_BreakText( IWordBreaker *iface,
         TEXT_SOURCE *ts, IWordSink *pWordSink, IPhraseSink *pPhraseSink)
{
    UINT len, state = 0;
    WCHAR ch;
112

113
    TRACE("%p %p %p\n", ts, pWordSink, pPhraseSink);
114

115 116 117 118 119 120 121 122 123 124 125 126 127 128
    if (pPhraseSink)
        FIXME("IPhraseSink won't be called\n");

    do
    {
        len = 0;
        while ((ts->iCur + len) < ts->iEnd)
        {
            ch = ts->awcBuffer[ts->iCur + len];

            switch (state)
            {
            case 0: /* skip spaces and punctuation */

129
                if (!ch || iswpunct(ch) || iswspace(ch))
130 131 132 133 134 135 136
                    ts->iCur ++;
                else
                    state = 1;
                break;

            case 1: /* find the end of the word */

137
                if (ch && !iswpunct(ch) && !iswspace(ch))
138 139 140 141 142 143 144 145 146 147 148 149 150 151
                    len++;
                else
                {
                    call_sink( pWordSink, ts, len );
                    len = 0;
                    state = 0;
                }
                break;

            }
        }
        call_sink( pWordSink, ts, len );

    } while (S_OK == ts->pfnFillTextBuffer( ts ));
152 153 154 155 156 157 158 159 160

    return S_OK;
}

static HRESULT WINAPI wb_ComposePhrase( IWordBreaker *iface,
         const WCHAR *pwcNoun, ULONG cwcNoun,
         const WCHAR *pwcModifier, ULONG cwcModifier,
         ULONG ulAttachmentType, WCHAR *pwcPhrase, ULONG *pcwcPhrase)
{
161
    FIXME("%p %u %p %u %u %p %p\n", pwcNoun, cwcNoun,
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
          pwcModifier, cwcModifier, ulAttachmentType, pwcPhrase, pcwcPhrase);
    return S_OK;
}

static HRESULT WINAPI wb_GetLicenseToUse( IWordBreaker *iface, const WCHAR **ppwcsLicense )
{
    FIXME("%p\n", ppwcsLicense);
    *ppwcsLicense = NULL;
    return S_OK;
}

static const IWordBreakerVtbl wordbreaker_vtbl =
{
    wb_QueryInterface,
    wb_AddRef,
    wb_Release,
    wb_Init,
    wb_BreakText,
    wb_ComposePhrase,
    wb_GetLicenseToUse,
};

184
DECLSPEC_HIDDEN HRESULT WINAPI wb_Constructor(IUnknown* pUnkOuter, REFIID riid, LPVOID *ppvObject)
185 186 187 188
{
    wordbreaker_impl *This;
    IWordBreaker *wb;

189
    TRACE("%p %s %p\n", pUnkOuter, debugstr_guid(riid), ppvObject);
190 191 192 193 194 195

    This = HeapAlloc(GetProcessHeap(), 0, sizeof *This);
    if (!This)
        return E_OUTOFMEMORY;

    This->ref = 1;
196
    This->IWordBreaker_iface.lpVtbl = &wordbreaker_vtbl;
197

198
    wb = &This->IWordBreaker_iface;
199 200 201

    return IWordBreaker_QueryInterface(wb, riid, ppvObject);
}