1
1
document . addEventListener ( 'DOMContentLoaded' , async ( ) => {
2
+ const defaultModel = 'gpt-4o-mini' ;
3
+
2
4
const maxPromptBytes = 8192 ;
3
5
const customPromptsCounter = document . getElementById ( 'customPromptsCounter' ) ;
4
6
@@ -12,10 +14,14 @@ document.addEventListener('DOMContentLoaded', async () => {
12
14
13
15
// Profile options
14
16
const name = document . getElementById ( 'name' ) ;
15
- const model = document . getElementById ( 'model' ) ;
16
17
const customPrompts = document . getElementById ( 'customPrompts' ) ;
17
18
const isDefault = document . getElementById ( 'default' ) ;
18
19
20
+ // Model-related widgets
21
+ const refreshModelsBtn = document . getElementById ( 'refresh-models-btn' ) ;
22
+ const saveProfileBtn = document . getElementById ( 'save-profile-btn' ) ;
23
+ const modelSelect = document . getElementById ( 'model' ) ;
24
+
19
25
let config ;
20
26
let currentProfile ;
21
27
@@ -57,7 +63,7 @@ document.addEventListener('DOMContentLoaded', async () => {
57
63
58
64
function buildDefaultProfile ( ) {
59
65
return {
60
- model : 'gpt-3.5-turbo-16k' ,
66
+ model : defaultModel ,
61
67
customPrompts : [ ] ,
62
68
} ;
63
69
}
@@ -79,7 +85,7 @@ document.addEventListener('DOMContentLoaded', async () => {
79
85
80
86
// Profile options
81
87
const name = document . getElementById ( 'name' ) . value . trim ( ) ;
82
- const model = document . getElementById ( 'model' ) . value . trim ( ) ;
88
+ const model = modelSelect . value . trim ( ) ;
83
89
const customPrompts = document . getElementById ( 'customPrompts' ) . value . split ( '\n' ) ;
84
90
const isDefault = document . getElementById ( 'default' ) . checked ;
85
91
@@ -127,7 +133,7 @@ document.addEventListener('DOMContentLoaded', async () => {
127
133
128
134
await chrome . storage . sync . set ( config ) ;
129
135
await reloadConfig ( ) ;
130
- await selectProfile ( name ) ;
136
+ selectProfile ( name ) ;
131
137
132
138
window . scrollTo ( 0 , 0 ) ;
133
139
showSuccess ( 'Settings saved.' ) ;
@@ -153,7 +159,7 @@ document.addEventListener('DOMContentLoaded', async () => {
153
159
await chrome . storage . sync . set ( config ) ;
154
160
155
161
showSuccess ( `Profile "${ currentProfile } " deleted.` ) ;
156
- await selectProfile ( config . defaultProfile ) ;
162
+ selectProfile ( config . defaultProfile ) ;
157
163
}
158
164
}
159
165
@@ -184,7 +190,7 @@ document.addEventListener('DOMContentLoaded', async () => {
184
190
185
191
async function reloadConfig ( ) {
186
192
const profileKeys = ( await chrome . storage . sync . get ( 'profiles' ) ) . profiles . map ( ( name ) => `profile__${ name } ` ) ;
187
- config = await chrome . storage . sync . get ( [ 'apiKey' , 'defaultProfile' , 'debug' , 'profiles' , ...profileKeys ] ) ;
193
+ config = await chrome . storage . sync . get ( [ 'apiKey' , 'defaultProfile' , 'debug' , 'models' , ' profiles', ...profileKeys ] ) ;
188
194
console . log ( 'Config' , config ) ;
189
195
190
196
if ( config . profiles === undefined ) {
@@ -214,7 +220,113 @@ document.addEventListener('DOMContentLoaded', async () => {
214
220
// Populate the profile selector dropdown
215
221
sortedProfileNames . forEach ( addOption ) ;
216
222
217
- await selectProfile ( currentProfile ) ;
223
+ // Populate the model selector dropdown if possible
224
+ if ( config . models && config . models . length > 0 ) {
225
+ populateModelOptions ( config . models ) ;
226
+ modelSelect . disabled = false ;
227
+ saveProfileBtn . disabled = false ;
228
+ } else {
229
+ modelSelect . disabled = true ;
230
+ saveProfileBtn . disabled = true ;
231
+ }
232
+
233
+ selectProfile ( currentProfile ) ;
234
+ }
235
+
236
+ function populateModelOptions ( models ) {
237
+ // Clear existing options
238
+ modelSelect . innerHTML = '' ;
239
+
240
+ // Populate the models dropdown
241
+ models . forEach ( ( modelName ) => {
242
+ const option = new Option ( modelName , modelName ) ;
243
+ modelSelect . add ( option ) ;
244
+ } ) ;
245
+ }
246
+
247
+ async function fetchAvailableModels ( apiKey ) {
248
+ const headers = {
249
+ Authorization : `Bearer ${ apiKey } ` ,
250
+ } ;
251
+
252
+ try {
253
+ const response = await fetch ( 'https://api.openai.com/v1/models' , {
254
+ method : 'GET' ,
255
+ headers : headers ,
256
+ } ) ;
257
+
258
+ if ( ! response . ok ) {
259
+ console . error ( 'Error fetching models:' , response ) ;
260
+ throw new Error ( `Error fetching models: ${ response . statusText } ` ) ;
261
+ }
262
+
263
+ const data = await response . json ( ) ;
264
+
265
+ models = data . data
266
+ // We only want the model IDs
267
+ . map ( ( model ) => model . id )
268
+ // Filter out models that are not GPT-3 or GPT-4
269
+ . filter ( ( model ) => model . startsWith ( 'gpt-' ) )
270
+ // Filter out models matching `-\d\d\d\d`
271
+ . filter ( ( model ) => ! model . match ( / - \d \d \d \d / ) ) ;
272
+
273
+ models . sort ( ) ;
274
+
275
+ return models ;
276
+ } catch ( error ) {
277
+ console . error ( error ) ;
278
+ return [ ] ;
279
+ }
280
+ }
281
+
282
+ async function refreshAvailableModels ( ) {
283
+ // Disable the button to prevent multiple clicks
284
+ refreshModelsBtn . disabled = true ;
285
+ refreshModelsBtn . textContent = 'Refreshing...' ;
286
+
287
+ // Store the currently selected model
288
+ const currentModel = modelSelect . value ;
289
+
290
+ try {
291
+ const apiKeyValue = apiKey . value . trim ( ) ;
292
+
293
+ if ( ! apiKeyValue ) {
294
+ showError ( 'Please enter your OpenAI API key before refreshing models.' ) ;
295
+ return ;
296
+ }
297
+
298
+ const models = await fetchAvailableModels ( apiKeyValue ) ;
299
+
300
+ if ( models . length === 0 ) {
301
+ showError ( 'No models retrieved. Please check your API key and try again.' ) ;
302
+ return ;
303
+ }
304
+
305
+ // Store models in config
306
+ config . models = models ;
307
+ await chrome . storage . sync . set ( config ) ;
308
+
309
+ // Populate the models dropdown
310
+ populateModelOptions ( models ) ;
311
+
312
+ // Enable the models select and save button
313
+ modelSelect . disabled = false ;
314
+ saveProfileBtn . disabled = false ;
315
+
316
+ showSuccess ( 'Available models have been refreshed.' ) ;
317
+
318
+ // Restore the previously selected model... if it still exists.
319
+ if ( models . includes ( currentModel ) ) {
320
+ modelSelect . value = currentModel ;
321
+ } else {
322
+ modelSelect . value = defaultModel ;
323
+ }
324
+ } catch ( error ) {
325
+ showError ( `Failed to refresh models: ${ error . message } ` ) ;
326
+ } finally {
327
+ refreshModelsBtn . disabled = false ;
328
+ refreshModelsBtn . textContent = 'Refresh available models' ;
329
+ }
218
330
}
219
331
220
332
function addOption ( name ) {
@@ -233,7 +345,7 @@ document.addEventListener('DOMContentLoaded', async () => {
233
345
profileSelector . value = profile ;
234
346
235
347
name . value = profile ;
236
- model . value = data . model || 'gpt-3.5-turbo-16k' ;
348
+ modelSelect . value = data . model || defaultModel ;
237
349
customPrompts . value = data . customPrompts . join ( '\n' ) || '' ;
238
350
isDefault . checked = profile === config . defaultProfile ;
239
351
@@ -357,6 +469,11 @@ document.addEventListener('DOMContentLoaded', async () => {
357
469
}
358
470
} ) ;
359
471
472
+ // Event listener for the refresh models button
473
+ refreshModelsBtn . addEventListener ( 'click' , async ( ) => {
474
+ await refreshAvailableModels ( ) ;
475
+ } ) ;
476
+
360
477
// Powers the display of the custom prompts byte counter
361
478
customPrompts . addEventListener ( 'input' , updateCustomPromptsCounter ) ;
362
479
0 commit comments