|
|
@@ -1,224 +0,0 @@
|
|
|
-import React, { useCallback, useState } from 'react';
|
|
|
-
|
|
|
-import { useTranslation } from 'react-i18next';
|
|
|
-import {
|
|
|
- Modal, ModalHeader, ModalBody, ModalFooter, Form, FormGroup, Label, Input,
|
|
|
-} from 'reactstrap';
|
|
|
-
|
|
|
-import { toastError, toastSuccess } from '~/client/util/toastr';
|
|
|
-import { AiAssistantAccessScope } from '~/features/openai/interfaces/ai-assistant';
|
|
|
-import type { IPageForItem } from '~/interfaces/page';
|
|
|
-import type { PopulatedGrantedGroup } from '~/interfaces/page-grant';
|
|
|
-import { usePageSelectModal } from '~/stores/modal';
|
|
|
-import loggerFactory from '~/utils/logger';
|
|
|
-
|
|
|
-import type { SelectedPage } from '../../../interfaces/selected-page';
|
|
|
-import { createAiAssistant } from '../../services/ai-assistant';
|
|
|
-import { useAiAssistantManegementModal } from '../../stores/ai-assistant';
|
|
|
-import { SelectedPageList } from '../Common/SelectedPageList';
|
|
|
-
|
|
|
-import { AccessScopeDropdown } from './AccessScopeDropdown';
|
|
|
-
|
|
|
-import styles from './AiAssistantManegementModal.module.scss';
|
|
|
-
|
|
|
-const moduleClass = styles['grw-ai-assistant-manegement'] ?? '';
|
|
|
-
|
|
|
-const logger = loggerFactory('growi:openai:client:components:AiAssistantManegementModal');
|
|
|
-
|
|
|
-const AiAssistantManegementModalSubstance = (): JSX.Element => {
|
|
|
- const { open: openPageSelectModal } = usePageSelectModal();
|
|
|
-
|
|
|
- const [selectedPages, setSelectedPages] = useState<SelectedPage[]>([]);
|
|
|
- const [selectedAccessScope, setSelectedAccessScope] = useState<AiAssistantAccessScope>(AiAssistantAccessScope.OWNER);
|
|
|
- const [selectedUserGroupsForAccessScope, setSelectedUserGroupsForAccessScope] = useState<PopulatedGrantedGroup[]>([]);
|
|
|
-
|
|
|
- const clickAccessScopeItemHandler = useCallback((accessScope: AiAssistantAccessScope) => {
|
|
|
- setSelectedAccessScope(accessScope);
|
|
|
- }, []);
|
|
|
-
|
|
|
- const selectUserGroupsForAccessScopeHandler = useCallback((targetUserGroup: PopulatedGrantedGroup) => {
|
|
|
- const selectedUserGroupIds = selectedUserGroupsForAccessScope.map(userGroup => userGroup.item._id);
|
|
|
- if (selectedUserGroupIds.includes(targetUserGroup.item._id)) {
|
|
|
- // if selected, remove it
|
|
|
- setSelectedUserGroupsForAccessScope(selectedUserGroupsForAccessScope.filter(userGroup => userGroup.item._id !== targetUserGroup.item._id));
|
|
|
- }
|
|
|
- else {
|
|
|
- // if not selected, add it
|
|
|
- setSelectedUserGroupsForAccessScope([...selectedUserGroupsForAccessScope, targetUserGroup]);
|
|
|
- }
|
|
|
- }, [selectedUserGroupsForAccessScope]);
|
|
|
-
|
|
|
- const clickOpenPageSelectModalHandler = useCallback(() => {
|
|
|
- const onSelected = (page: IPageForItem, isIncludeSubPage: boolean) => {
|
|
|
- const selectedPageIds = selectedPages.map(selectedPage => selectedPage.page._id);
|
|
|
- if (page._id != null && !selectedPageIds.includes(page._id)) {
|
|
|
- setSelectedPages([...selectedPages, { page, isIncludeSubPage }]);
|
|
|
- }
|
|
|
- };
|
|
|
-
|
|
|
- openPageSelectModal({ onSelected, isHierarchicalSelectionMode: true });
|
|
|
- }, [openPageSelectModal, selectedPages]);
|
|
|
-
|
|
|
- const clickRmoveSelectedPageHandler = useCallback((pageId: string) => {
|
|
|
- setSelectedPages(selectedPages.filter(selectedPage => selectedPage.page._id !== pageId));
|
|
|
- }, [selectedPages]);
|
|
|
-
|
|
|
- const clickCreateAiAssistantHandler = useCallback(async() => {
|
|
|
- try {
|
|
|
- const pagePathPatterns = selectedPages
|
|
|
- .map(selectedPage => (selectedPage.isIncludeSubPage ? `${selectedPage.page.path}/*` : selectedPage.page.path))
|
|
|
- .filter((path): path is string => path !== undefined && path !== null);
|
|
|
-
|
|
|
- const grantedGroupsForAccessScope = selectedUserGroupsForAccessScope.map(group => ({
|
|
|
- type: group.type,
|
|
|
- item: group.item._id, // Convert UserGroupDocument to Ref<IUserGroup>
|
|
|
- }));
|
|
|
-
|
|
|
- await createAiAssistant({
|
|
|
- name: 'test',
|
|
|
- description: 'test',
|
|
|
- additionalInstruction: 'test',
|
|
|
- pagePathPatterns,
|
|
|
- shareScope: 'publicOnly',
|
|
|
- accessScope: selectedAccessScope,
|
|
|
- grantedGroupsForAccessScope: selectedAccessScope === AiAssistantAccessScope.GROUPS ? grantedGroupsForAccessScope : undefined,
|
|
|
- });
|
|
|
- toastSuccess('アシスタントを作成しました');
|
|
|
- }
|
|
|
- catch (err) {
|
|
|
- toastError('アシスタントの作成に失敗しました');
|
|
|
- logger.error(err);
|
|
|
- }
|
|
|
- }, [selectedPages, selectedUserGroupsForAccessScope, selectedAccessScope]);
|
|
|
-
|
|
|
- return (
|
|
|
- <div className="px-4">
|
|
|
- <ModalBody>
|
|
|
- <Form>
|
|
|
- <FormGroup className="mb-4">
|
|
|
- <Label className="mb-2 ">アシスタント名</Label>
|
|
|
- <Input
|
|
|
- type="text"
|
|
|
- placeholder="アシスタント名を入力"
|
|
|
- className="border rounded"
|
|
|
- />
|
|
|
- </FormGroup>
|
|
|
-
|
|
|
- <FormGroup className="mb-4">
|
|
|
- <div className="d-flex align-items-center mb-2">
|
|
|
- <Label className="mb-0">アシスタントの種類</Label>
|
|
|
- <span className="ms-1 fs-5 material-symbols-outlined text-secondary">help</span>
|
|
|
- </div>
|
|
|
- <div className="d-flex gap-4">
|
|
|
- <FormGroup check>
|
|
|
- <Input type="checkbox" defaultChecked />
|
|
|
- <Label check>ナレッジアシスタント</Label>
|
|
|
- </FormGroup>
|
|
|
- <FormGroup check>
|
|
|
- <Input type="checkbox" />
|
|
|
- <Label check>エディタアシスタント</Label>
|
|
|
- </FormGroup>
|
|
|
- <FormGroup check>
|
|
|
- <Input type="checkbox" />
|
|
|
- <Label check>ラーニングアシスタント</Label>
|
|
|
- </FormGroup>
|
|
|
- </div>
|
|
|
- </FormGroup>
|
|
|
-
|
|
|
- <FormGroup className="mb-4">
|
|
|
- <div className="d-flex align-items-center mb-2">
|
|
|
- <Label className="mb-0">ページのアクセス権限</Label>
|
|
|
- <span className="ms-1 fs-5 material-symbols-outlined text-secondary">help</span>
|
|
|
- </div>
|
|
|
- <AccessScopeDropdown
|
|
|
- selectedAccessScope={selectedAccessScope}
|
|
|
- selectedUserGroup={selectedUserGroupsForAccessScope}
|
|
|
- onSelectAccessScope={clickAccessScopeItemHandler}
|
|
|
- onSelectUserGroup={selectUserGroupsForAccessScopeHandler}
|
|
|
- />
|
|
|
- </FormGroup>
|
|
|
-
|
|
|
- <FormGroup className="mb-4">
|
|
|
- <div className="d-flex align-items-center mb-2">
|
|
|
- <Label className="mb-0">参照するページ</Label>
|
|
|
- <span className="ms-1 fs-5 material-symbols-outlined text-secondary">help</span>
|
|
|
- </div>
|
|
|
- <SelectedPageList selectedPages={selectedPages} onRemove={clickRmoveSelectedPageHandler} />
|
|
|
- <button
|
|
|
- type="button"
|
|
|
- className="btn btn-outline-primary d-flex align-items-center gap-1"
|
|
|
- onClick={clickOpenPageSelectModalHandler}
|
|
|
- >
|
|
|
- <span>+</span>
|
|
|
- 追加する
|
|
|
- </button>
|
|
|
- </FormGroup>
|
|
|
-
|
|
|
- <FormGroup>
|
|
|
- <div className="d-flex align-items-center mb-2">
|
|
|
- <Label className="mb-0 me-2">アシスタントへの指示</Label>
|
|
|
- <label className="form-label form-check-label">
|
|
|
- <span className="badge text-bg-danger mt-2">
|
|
|
- 必須
|
|
|
- </span>
|
|
|
- </label>
|
|
|
- </div>
|
|
|
- <Input
|
|
|
- type="textarea"
|
|
|
- placeholder="アシスタントに実行して欲しい内容を具体的に記入してください"
|
|
|
- className="border rounded"
|
|
|
- rows={4}
|
|
|
- />
|
|
|
- </FormGroup>
|
|
|
-
|
|
|
- <FormGroup>
|
|
|
- <div className="d-flex align-items-center mb-2">
|
|
|
- <Label className="mb-0 me-2">アシスタントのメモ</Label>
|
|
|
- <label className="form-label form-check-label">
|
|
|
- <span className="badge text-bg-secondary mt-2">
|
|
|
- 必須
|
|
|
- </span>
|
|
|
- </label>
|
|
|
- </div>
|
|
|
- <Input
|
|
|
- type="textarea"
|
|
|
- placeholder="内容や用途のメモを表示させることができます"
|
|
|
- className="border rounded"
|
|
|
- rows={4}
|
|
|
- />
|
|
|
- <p className="mt-1 text-muted">メモ内容はアシスタントには影響しません。</p>
|
|
|
- </FormGroup>
|
|
|
- </Form>
|
|
|
- </ModalBody>
|
|
|
-
|
|
|
- <ModalFooter className="border-0 pt-0 mb-3">
|
|
|
- <button type="button" className="btn btn-outline-secondary" onClick={() => {}}>キャンセル</button>
|
|
|
- <button type="button" className="btn btn-primary" onClick={clickCreateAiAssistantHandler}>作成</button>
|
|
|
- </ModalFooter>
|
|
|
- </div>
|
|
|
- );
|
|
|
-};
|
|
|
-
|
|
|
-
|
|
|
-export const AiAssistantManegementModal = (): JSX.Element => {
|
|
|
- const { t } = useTranslation();
|
|
|
-
|
|
|
- const { data: aiAssistantManegementModalData, close: closeAiAssistantManegementModal } = useAiAssistantManegementModal();
|
|
|
-
|
|
|
- const isOpened = aiAssistantManegementModalData?.isOpened ?? false;
|
|
|
-
|
|
|
- return (
|
|
|
- <Modal size="lg" isOpen={isOpened} toggle={closeAiAssistantManegementModal} className={moduleClass} scrollable>
|
|
|
-
|
|
|
- <ModalHeader tag="h4" toggle={closeAiAssistantManegementModal} className="pe-4">
|
|
|
- <span className="growi-custom-icons growi-ai-assistant-icon me-3 fs-4">ai_assistant</span>
|
|
|
- <span className="fw-bold">新規アシスタントの追加</span> {/* TODO i18n */}
|
|
|
- </ModalHeader>
|
|
|
-
|
|
|
- { isOpened && (
|
|
|
- <AiAssistantManegementModalSubstance />
|
|
|
- ) }
|
|
|
-
|
|
|
- </Modal>
|
|
|
- );
|
|
|
-};
|