Compare commits

...

5 Commits

Author SHA1 Message Date
7e5b878a9c chore: add release workflow with auto changelog
Some checks failed
Release / build (push) Failing after 2m24s
2026-04-11 02:24:44 +08:00
90320808e4 Feature:
增加Gitea自动发布二进制版本
2026-04-11 02:17:00 +08:00
003815239d feat: implement copy selected text with Ctrl+C 2026-04-11 01:53:29 +08:00
d885ca1317 fix: disable Ctrl+C exit, copy works via textarea native 2026-04-11 01:49:08 +08:00
74fa5a52ab feat: add copy (Ctrl+C) and exit (Ctrl+D) shortcuts 2026-04-11 01:44:34 +08:00
4 changed files with 95 additions and 6 deletions

View File

@@ -0,0 +1,56 @@
name: Release
on:
push:
tags:
- 'v*'
jobs:
build:
runs-on: ubuntu-latest
container:
image: hub.gaomia.site/titor/bun:latest
steps:
- name: Install git
run: apt-get update && apt-get install -y git
- name: Checkout
run: |
git clone https://${{ secrets.release_token }}@hub.gaomia.site/titor/mio.git /workspace/mio
cd /workspace/mio
- name: Create source archive
run: |
git archive --format=zip -o mio-source.zip HEAD
- name: Release
env:
GITEA_TOKEN: ${{ secrets.release_token }}
run: |
cd /workspace/mio
TAG_NAME="${GITHUB_REF#refs/tags/}"
PREV_TAG=$(git describe --tags --abbrev=0 HEAD^ 2>/dev/null || echo "")
if [ -n "$PREV_TAG" ]; then
CHANGES=$(git log --pretty=format:"- %s (%h)" "$PREV_TAG..HEAD" 2>/dev/null || echo "")
else
CHANGES=$(git log --pretty=format:"- %s (%h)" -n 20 2>/dev/null || echo "")
fi
if [ -n "$CHANGES" ]; then
RELEASE_BODY="## Changes\n\n$CHANGES"
else
RELEASE_BODY="Release $TAG_NAME"
fi
RELEASE_RESPONSE=$(curl -s -X POST "https://hub.gaomia.site/api/v1/repos/titor/mio/releases" \
-H "Authorization: token ${GITEA_TOKEN}" \
-H "Content-Type: application/json" \
-d "{\"tag_name\":\"${TAG_NAME}\",\"name\":\"${TAG_NAME}\",\"body\":\"${RELEASE_BODY}\"}")
RELEASE_ID=$(echo "$RELEASE_RESPONSE" | grep -o '"id":[0-9]*' | head -1 | cut -d: -f2)
curl -s -X POST "https://hub.gaomia.site/api/v1/repos/titor/mio/releases/${RELEASE_ID}/assets" \
-H "Authorization: token ${GITEA_TOKEN}" \
-F "attachment=@mio-source.zip"

View File

@@ -17,6 +17,8 @@ All notable changes to this project will be documented in this file.
- **Multi-select delete**: Double-click to enter selection mode, click to toggle selection, batch delete support - **Multi-select delete**: Double-click to enter selection mode, click to toggle selection, batch delete support
- **Title truncation**: Long titles in the list display with ellipsis - **Title truncation**: Long titles in the list display with ellipsis
- **Time display**: Format "time · month/day/year", e.g., "12:45 · 4/11/2026" - **Time display**: Format "time · month/day/year", e.g., "12:45 · 4/11/2026"
- **Copy**: In edit mode, use Shift+Arrow to select text, then Ctrl+C to copy
- **Exit**: Press Ctrl+D to exit
### Technical ### Technical
@@ -33,4 +35,11 @@ All notable changes to this project will be documented in this file.
```bash ```bash
npm run dev # Development mode npm run dev # Development mode
npm run build # Build to mio.exe npm run build # Build to mio.exe
``` ```
### Keyboard Shortcuts
- `Shift+Arrow` - Select text in edit mode
- `Ctrl+C` - Copy selected text (in edit mode)
- `Ctrl+D` - Exit application
- `Esc` - Exit application (alternative)

View File

@@ -1,10 +1,12 @@
import { useState, useEffect, useRef } from "react"; import { useState, useEffect, useRef } from "react";
import type { TextareaRenderable } from "@opentui/core";
import { db, type Note } from "../db"; import { db, type Note } from "../db";
interface NoteEditorProps { interface NoteEditorProps {
note: Note | null; note: Note | null;
onUpdate: () => void; onUpdate: () => void;
onDelete: (id: number) => void; onDelete: (id: number) => void;
onCopySelected: () => string;
} }
function renderMarkdown(content: string): any { function renderMarkdown(content: string): any {
@@ -58,11 +60,11 @@ function renderMarkdown(content: string): any {
return elements; return elements;
} }
export function NoteEditor({ note, onUpdate, onDelete }: NoteEditorProps) { export function NoteEditor({ note, onUpdate, onDelete, onCopySelected }: NoteEditorProps) {
const [isEditing, setIsEditing] = useState(false); const [isEditing, setIsEditing] = useState(false);
const [title, setTitle] = useState(""); const [title, setTitle] = useState("");
const [content, setContent] = useState(""); const [content, setContent] = useState("");
const textareaRef = useRef<{ plainText: string } | null>(null); const textareaRef = useRef<TextareaRenderable | null>(null);
useEffect(() => { useEffect(() => {
if (note) { if (note) {
@@ -75,6 +77,14 @@ export function NoteEditor({ note, onUpdate, onDelete }: NoteEditorProps) {
setIsEditing(false); setIsEditing(false);
}, [note?.id]); }, [note?.id]);
useEffect(() => {
if (onCopySelected) {
(window as any).__getSelectedText = () => {
return textareaRef.current?.getSelectedText() ?? "";
};
}
}, [onCopySelected]);
const handleSave = async () => { const handleSave = async () => {
if (note) { if (note) {
const finalContent = textareaRef.current?.plainText ?? content; const finalContent = textareaRef.current?.plainText ?? content;
@@ -85,6 +95,10 @@ export function NoteEditor({ note, onUpdate, onDelete }: NoteEditorProps) {
} }
}; };
const getSelectedText = (): string => {
return textareaRef.current?.getSelectedText() ?? "";
};
if (!note) { if (!note) {
return ( return (
<box flexGrow={1} justifyContent="center" alignItems="center"> <box flexGrow={1} justifyContent="center" alignItems="center">

View File

@@ -77,10 +77,20 @@ function App() {
const selectedNote = selectedId ? notes.find((n) => n.id === selectedId) : null; const selectedNote = selectedId ? notes.find((n) => n.id === selectedId) : null;
const handleCopySelected = (): string => {
return (window as any).__getSelectedText?.() ?? "";
};
useKeyboard((key) => { useKeyboard((key) => {
if (key.name === "escape") { if (key.ctrl && key.name === "d") {
process.exit(0); process.exit(0);
} }
if (key.ctrl && key.name === "c") {
const text = handleCopySelected();
if (text) {
process.stdout.write('\x1b]52;c;' + btoa(text) + '\x07');
}
}
}); });
return ( return (
@@ -98,11 +108,11 @@ function App() {
/> />
</box> </box>
<box flexGrow={1}> <box flexGrow={1}>
<NoteEditor note={selectedNote || null} onUpdate={handleUpdate} onDelete={handleDelete} /> <NoteEditor note={selectedNote || null} onUpdate={handleUpdate} onDelete={handleDelete} onCopySelected={handleCopySelected} />
</box> </box>
</box> </box>
); );
} }
const renderer = await createCliRenderer(); const renderer = await createCliRenderer({ exitOnCtrlC: false });
createRoot(renderer).render(<App />); createRoot(renderer).render(<App />);