File size: 6,217 Bytes
9705b6c
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
import { useEffect, useState } from 'react';
import TextareaAutosize from 'react-textarea-autosize';
import { useUpdateTokenCountMutation, TUpdateTokenCountResponse } from 'librechat-data-provider';
import type { TSettingsProps } from '~/common';
import { Label, Checkbox, SelectDropDown } from '~/components/ui';
import { cn, defaultTextProps, removeFocusOutlines } from '~/utils/';
import { useLocalize, useDebounce } from '~/hooks';

export default function Settings({ conversation, setOption, readonly }: TSettingsProps) {
  const localize = useLocalize();
  const [tokenCount, setTokenCount] = useState(0);
  const debouncedContext = useDebounce(conversation?.context?.trim() ?? '', 250);
  const updateTokenCountMutation = useUpdateTokenCountMutation();

  useEffect(() => {
    if (!debouncedContext || debouncedContext === '') {
      setTokenCount(0);
      return;
    }

    const handleTextChange = (context: string) => {
      updateTokenCountMutation.mutate(
        { text: context },
        {
          onSuccess: (data: TUpdateTokenCountResponse) => {
            setTokenCount(data.count);
          },
        },
      );
    };

    handleTextChange(debouncedContext);
    // eslint-disable-next-line react-hooks/exhaustive-deps
  }, [debouncedContext]);

  if (!conversation) {
    return null;
  }
  const { context, systemMessage, jailbreak, toneStyle } = conversation;
  const showSystemMessage = jailbreak;

  const setContext = setOption('context');
  const setSystemMessage = setOption('systemMessage');
  const setJailbreak = setOption('jailbreak');
  const setToneStyle = (value: string) => setOption('toneStyle')(value.toLowerCase());

  return (
    <div className="grid gap-6 sm:grid-cols-2">
      <div className="col-span-1 flex flex-col items-center justify-start gap-6">
        <div className="grid w-full items-center gap-2">
          <Label htmlFor="toneStyle-dropdown" className="text-left text-sm font-medium">
            {localize('com_endpoint_tone_style')}{' '}
            <small className="opacity-40">({localize('com_endpoint_default_creative')})</small>
          </Label>
          <SelectDropDown
            id="toneStyle-dropdown"
            title={''}
            value={`${toneStyle?.charAt(0).toUpperCase()}${toneStyle?.slice(1)}`}
            setValue={setToneStyle}
            availableValues={['Creative', 'Fast', 'Balanced', 'Precise']}
            disabled={readonly}
            className={cn(defaultTextProps, 'flex w-full resize-none', removeFocusOutlines)}
            containerClassName="flex w-full resize-none"
          />
        </div>
        <div className="grid w-full items-center gap-2">
          <Label htmlFor="context" className="text-left text-sm font-medium">
            {localize('com_endpoint_context')}{' '}
            <small className="opacity-40">({localize('com_endpoint_default_blank')})</small>
          </Label>
          <TextareaAutosize
            id="context"
            disabled={readonly}
            value={context || ''}
            onChange={(e) => setContext(e.target.value ?? null)}
            placeholder={localize('com_endpoint_bing_context_placeholder')}
            className={cn(
              defaultTextProps,
              'flex max-h-[300px] min-h-[100px] w-full resize-none px-3 py-2',
            )}
          />
          <small className="mb-5 text-black dark:text-white">{`${localize(
            'com_endpoint_token_count',
          )}: ${tokenCount}`}</small>
        </div>
      </div>
      <div className="col-span-1 flex flex-col items-center justify-start gap-6">
        <div className="grid w-full items-center gap-2">
          <Label htmlFor="jailbreak" className="text-left text-sm font-medium">
            {localize('com_endpoint_bing_enable_sydney')}{' '}
            <small className="opacity-40">({localize('com_endpoint_default_false')})</small>
          </Label>
          <div className="flex h-[40px] w-full items-center space-x-3">
            <Checkbox
              id="jailbreak"
              disabled={readonly}
              checked={jailbreak}
              className="focus:ring-opacity-20 dark:border-gray-500 dark:bg-gray-700 dark:text-gray-50 dark:focus:ring-gray-600 dark:focus:ring-opacity-50 dark:focus:ring-offset-0"
              onCheckedChange={setJailbreak}
            />
            <label
              htmlFor="jailbreak"
              className="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70 dark:text-gray-50"
            >
              {localize('com_endpoint_bing_jailbreak')}{' '}
              <small>{localize('com_endpoint_bing_to_enable_sydney')}</small>
            </label>
          </div>
        </div>
        {showSystemMessage && (
          <div className="grid w-full items-center gap-2">
            <Label
              htmlFor="systemMessage"
              className="text-left text-sm font-medium"
              style={{ opacity: showSystemMessage ? '1' : '0' }}
            >
              <a
                href="https://github.com/danny-avila/LibreChat/blob/main/docs/features/bing_jailbreak.md#default-system-message-for-jailbreak-mode-sydney"
                target="_blank"
                className="text-blue-500 transition-colors duration-200 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-500"
                rel="noreferrer"
              >
                {localize('com_endpoint_system_message')}
              </a>{' '}
              <small className="opacity-40 dark:text-gray-50">
                ( {localize('com_endpoint_default_blank')})
              </small>
            </Label>

            <TextareaAutosize
              id="systemMessage"
              disabled={readonly}
              value={systemMessage || ''}
              onChange={(e) => setSystemMessage(e.target.value ?? null)}
              placeholder={localize('com_endpoint_bing_system_message_placeholder')}
              className={cn(
                defaultTextProps,
                'flex max-h-[300px] min-h-[100px] w-full resize-none px-3 py-2 placeholder:text-red-400',
              )}
            />
          </div>
        )}
      </div>
    </div>
  );
}