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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
|
// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROME_BROWSER_SPELLCHECKER_H__
#define CHROME_BROWSER_SPELLCHECKER_H__
#include <vector>
#include "chrome/browser/spellcheck_worditerator.h"
#include "base/task.h"
#include "unicode/uscript.h"
class Hunspell;
class PrefService;
class Profile;
class MessageLoop;
class URLRequestContext;
// The Browser's Spell Checker. It checks and suggests corrections.
//
// This object is not threadsafe. In normal usage (not unit tests) it lives on
// the I/O thread of the browser. It is threadsafe refcounted so that I/O thread
// and the profile on the main thread (which gives out references to it) can
// keep it. However, all usage of this must be on the I/O thread.
//
// This object should also be deleted on the I/O thread only. It owns a
// reference to URLRequestContext which in turn owns the cache, etc. and must be
// deleted on the I/O thread itself.
class SpellChecker : public base::RefCountedThreadSafe<SpellChecker> {
public:
// Creates the spellchecker by reading dictionaries from the given directory,
// and defaulting to the given language. Both strings must be provided.
//
// The request context is used to download dictionaries if they do not exist.
// This can be NULL if you don't want this (like in tests).
// The |custom_dictionary_file_name| should be left blank so that Spellchecker
// can figure out the custom dictionary file. It is non empty only for unit
// testing.
SpellChecker(const std::wstring& dict_dir,
const std::wstring& language,
URLRequestContext* request_context,
const std::wstring& custom_dictionary_file_name);
static void RegisterUserPrefs(PrefService* prefs);
// Only delete on the I/O thread (see above).
~SpellChecker();
// SpellCheck a word.
// Returns true if spelled correctly, false otherwise.
// If the spellchecker failed to initialize, always returns true.
// In addition, finds the suggested words for a given word
// and puts them into |*optional_suggestions|.
// If the word is spelled correctly, the vector is empty.
// If optional_suggestions is NULL, suggested words will not be looked up.
// Note that Doing suggest lookups can be slow.
bool SpellCheckWord(const wchar_t* in_word,
int in_word_len,
int* misspelling_start,
int* misspelling_len,
std::vector<std::wstring>* optional_suggestions);
// Add custom word to the dictionary, which means:
// a) Add it to the current hunspell object for immediate use,
// b) Add the word to a file in disk for custom dictionary.
void AddWord(const std::wstring& word);
private:
// Download dictionary files when required.
class DictionaryDownloadController;
SpellChecker::SpellChecker();
// Initializes the Hunspell Dictionary.
bool Initialize();
// After |hunspell_| is initialized, this function is called to add custom
// words from the custom dictionary to the |hunspell_|
void AddCustomWordsToHunspell();
void set_file_is_downloading(bool value);
// Memory maps the given .bdic file. On success, it will return true and will
// place the data and lenght into the given out parameters.
bool MapBdictFile(const unsigned char** data, size_t* length);
// Returns whether or not the given word is a contraction of valid words
// (e.g. "word:word").
bool IsValidContraction(const std::wstring& word);
// Return the file name of the dictionary, including the path and the version
// numbers.
std::wstring GetVersionedFileName(const std::wstring& language,
const std::wstring& dict_dir);
// Path to the spellchecker file.
std::wstring bdict_file_name_;
// Path to the custom dictionary file.
std::wstring custom_dictionary_file_name_;
// We memory-map the BDict file for spellchecking. These are the handles
// necessary for that.
HANDLE bdict_file_;
HANDLE bdict_mapping_;
const unsigned char* bdict_mapped_data_;
// The hunspell dictionary in use.
Hunspell *hunspell_;
// Represents character attributes used for filtering out characters which
// are not supported by this SpellChecker object.
SpellcheckCharAttribute character_attributes_;
// Flag indicating whether we've tried to initialize. If we've already
// attempted initialiation, we won't retry to avoid failure loops.
bool tried_to_init_;
#ifndef NDEBUG
// This object must only be used on the same thread. However, it is normally
// created on the UI thread. This checks calls to SpellCheckWord and the
// destructor to make sure we're only ever running on the same thread.
//
// This will be NULL if it is not initialized yet (not initialized in the
// constructor since that's on a different thread.
MessageLoop* worker_loop_;
#endif
// Flag indicating whether we've tried to download dictionary files. If we've
// already attempted download, we won't retry to avoid failure loops.
bool tried_to_download_;
// File Thread Message Loop.
MessageLoop* file_loop_;
// UI Thread Message Loop - this will be used as a proxy to access io loop.
MessageLoop* ui_loop_;
// Used for requests. MAY BE NULL which means don't try to download.
URLRequestContext* url_request_context_;
// DictionaryDownloadController object to download dictionary if required.
scoped_refptr<DictionaryDownloadController> ddc_dic_;
// Set when the dictionary file is currently downloading.
bool dic_is_downloading_;
// Used for generating callbacks to spellchecker, since spellchecker is a
// non-reference counted object. The callback is generated by generating tasks
// using NewRunableMethod on these objects.
ScopedRunnableMethodFactory<SpellChecker> dic_download_state_changer_factory_;
DISALLOW_EVIL_CONSTRUCTORS(SpellChecker);
};
#endif // #ifndef CHROME_BROWSER_SPELLCHECKER_H__
|