Faisal Amir b662c25007
fix: adjustment minor UI from qa feedback (#2963)
* feat: update minor ui from feedback

* fix: adjust spacing import model

* feat: edit title thread (#2964)

* feat: edit title thread

* fix: fix copies

* fix copies

* fix copies
2024-05-30 10:53:40 +07:00

203 lines
6.1 KiB
TypeScript

import { useCallback, useEffect, useState } from 'react'
import {
Model,
ModelEvent,
events,
joinPath,
openFileExplorer,
} from '@janhq/core'
import { Modal, Button, Input, ModalClose, TextArea } from '@janhq/joi'
import { atom, useAtom, useAtomValue, useSetAtom } from 'jotai'
import { Paperclip } from 'lucide-react'
import useImportModel, {
getImportModelStageAtom,
setImportModelStageAtom,
} from '@/hooks/useImportModel'
import { toGibibytes } from '@/utils/converter'
import { openFileTitle } from '@/utils/titleUtils'
import { janDataFolderPathAtom } from '@/helpers/atoms/AppConfig.atom'
import {
importingModelsAtom,
updateImportingModelAtom,
} from '@/helpers/atoms/Model.atom'
export const editingModelIdAtom = atom<string | undefined>(undefined)
const EditModelInfoModal = () => {
const importModelStage = useAtomValue(getImportModelStageAtom)
const importingModels = useAtomValue(importingModelsAtom)
const setImportModelStage = useSetAtom(setImportModelStageAtom)
const [editingModelId, setEditingModelId] = useAtom(editingModelIdAtom)
const [modelName, setModelName] = useState('')
const [modelId, setModelId] = useState('')
const [description, setDescription] = useState('')
const [tags, setTags] = useState<string[]>([])
const janDataFolder = useAtomValue(janDataFolderPathAtom)
const updateImportingModel = useSetAtom(updateImportingModelAtom)
const { updateModelInfo } = useImportModel()
const [modelPath, setModelPath] = useState<string>('')
const editingModel = importingModels.find(
(model) => model.importId === editingModelId
)
useEffect(() => {
if (editingModel && editingModel.modelId != null) {
setModelName(editingModel.name)
setModelId(editingModel.modelId)
setDescription(editingModel.description)
setTags(editingModel.tags)
}
}, [editingModel])
const onCancelClick = () => {
setImportModelStage('IMPORTING_MODEL')
setEditingModelId(undefined)
}
const onSaveClick = async () => {
if (!editingModel || !editingModel.modelId) return
const modelInfo: Partial<Model> = {
id: editingModel.modelId,
name: modelName,
description,
metadata: {
author: 'User',
tags,
size: 0,
},
}
await updateModelInfo(modelInfo)
events.emit(ModelEvent.OnModelsUpdate, {})
updateImportingModel(editingModel.importId, modelName, description, tags)
setImportModelStage('IMPORTING_MODEL')
setEditingModelId(undefined)
}
useEffect(() => {
const getModelPath = async () => {
const modelId = editingModel?.modelId
if (!modelId) return ''
const path = await joinPath([janDataFolder, 'models', modelId])
setModelPath(path)
}
getModelPath()
}, [janDataFolder, editingModel])
const onShowInFinderClick = useCallback(() => {
openFileExplorer(modelPath)
}, [modelPath])
if (!editingModel) {
setImportModelStage('IMPORTING_MODEL')
setEditingModelId(undefined)
return null
}
const onTagsChange = (e: React.ChangeEvent<HTMLInputElement>) => {
const tags = e.target.value.split(',')
setTags(tags)
}
return (
<Modal
open={importModelStage === 'EDIT_MODEL_INFO'}
onOpenChange={onCancelClick}
title="Edit Model Information"
content={
<div>
<div className="flex flex-row space-x-4 rounded-xl border border-[hsla(var(--app-border))] p-4">
<div className="flex h-10 w-10 items-center justify-center rounded-full">
<Paperclip color="#fff" />
</div>
<div className="flex flex-1 flex-col">
<p>{editingModel.name}</p>
<div className="flex flex-row">
<span className="mr-2">{toGibibytes(editingModel.size)}</span>
<div className="flex flex-row space-x-1">
<span className="font-semibold">Format:</span>
<span className="font-normal">
{editingModel.format.toUpperCase()}
</span>
</div>
</div>
<div className="mt-1 flex flex-row items-center space-x-2">
<span className="line-clamp-1 text-xs font-normal">
{modelPath}
</span>
<Button theme="ghost" onClick={onShowInFinderClick}>
{openFileTitle()}
</Button>
</div>
</div>
</div>
<form className="mt-4 flex flex-col space-y-4">
<div className="flex flex-col">
<label className="mb-1">Model Name</label>
<Input
value={modelName}
onChange={(e) => {
e.preventDefault()
setModelName(e.target.value)
}}
/>
</div>
<div className="flex flex-col">
<label className="mb-1">Model ID</label>
<Input
disabled
value={modelId}
onChange={(e) => {
e.preventDefault()
setModelId(e.target.value)
}}
/>
</div>
<div className="flex flex-col">
<label className="mb-1">Description</label>
<TextArea
value={description}
onChange={(e) => {
e.preventDefault()
setDescription(e.target.value)
}}
/>
</div>
<div className="flex flex-col">
<label className="mb-1">Tags</label>
<Input value={tags.join(',')} onChange={onTagsChange} />
</div>
</form>
<div className="mt-4 flex gap-x-2">
<ModalClose asChild onClick={onCancelClick}>
<Button theme="ghost">Cancel</Button>
</ModalClose>
<ModalClose asChild>
<Button autoFocus theme="primary" onClick={onSaveClick}>
Save
</Button>
</ModalClose>
</div>
</div>
}
/>
)
}
export default EditModelInfoModal