mirror of
https://github.com/ankitects/anki.git
synced 2025-09-18 14:02:21 -04:00

* Fix footer moving upwards
* Fix column detection
Was broken because escaped line breaks were not considered.
Also removes delimiter detection on `#columns:` line. User must use tabs
or set delimiter beforehand.
* Add CSV preview
* Parse `#tags column:`
* Optionally export deck and notetype with CSV
* Avoid clones in CSV export
* Prevent bottom of page appearing under footer (dae)
* Increase padding to 1em (dae)
With 0.5em, when a vertical scrollbar is shown, it sits right next to
the right edge of the content, making it look like there's no right
margin.
* Experimental changes to make table fit+scroll (dae)
- limit individual cells to 15em, and show ellipses when truncated
- limit total table width to body width, so that inner table is shown
with scrollbar
- use class rather than id - ids are bad practice in Svelte components,
as more than one may be displayed on a single page
* Skip importing foreign notes with filtered decks
Were implicitly imported into the default deck before.
Also some refactoring to fetch deck ids and names beforehand.
* Hide spacer below hidden field mapping
* Fix guid being replaced when updating note
* Fix dupe identity check
Canonify tags before checking if dupe is identical, but only add update
tags later if appropriate.
* Fix deck export for notes with missing card 1
* Fix note lines starting with `#`
csv crate doesn't support escaping a leading comment char. :(
* Support import/export of guids
* Strip HTML from preview rows
* Fix initially set deck if current is filtered
* Make isHtml toggle reactive
* Fix `html_to_text_line()` stripping sound names
* Tweak export option labels
* Switch to patched rust-csv fork
Fixes writing lines starting with `#`, so revert 5ece10ad05
.
* List column options with first column field
* Fix flag for exports with HTML stripped
148 lines
5.1 KiB
Svelte
148 lines
5.1 KiB
Svelte
<!--
|
|
Copyright: Ankitects Pty Ltd and contributors
|
|
License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
-->
|
|
<script lang="ts">
|
|
import Col from "../components/Col.svelte";
|
|
import Container from "../components/Container.svelte";
|
|
import Row from "../components/Row.svelte";
|
|
import Spacer from "../components/Spacer.svelte";
|
|
import * as tr from "../lib/ftl";
|
|
import {
|
|
Decks,
|
|
Generic,
|
|
ImportExport,
|
|
importExport,
|
|
Notetypes,
|
|
} from "../lib/proto";
|
|
import DeckSelector from "./DeckSelector.svelte";
|
|
import DelimiterSelector from "./DelimiterSelector.svelte";
|
|
import DupeResolutionSelector from "./DupeResolutionSelector.svelte";
|
|
import FieldMapper from "./FieldMapper.svelte";
|
|
import Header from "./Header.svelte";
|
|
import HtmlSwitch from "./HtmlSwitch.svelte";
|
|
import { getColumnOptions, getCsvMetadata } from "./lib";
|
|
import NotetypeSelector from "./NotetypeSelector.svelte";
|
|
import Preview from "./Preview.svelte";
|
|
import StickyFooter from "./StickyFooter.svelte";
|
|
import Tags from "./Tags.svelte";
|
|
|
|
export let path: string;
|
|
export let notetypeNameIds: Notetypes.NotetypeNameId[];
|
|
export let deckNameIds: Decks.DeckNameId[];
|
|
|
|
export let delimiter: ImportExport.CsvMetadata.Delimiter;
|
|
export let forceDelimiter: boolean;
|
|
export let forceIsHtml: boolean;
|
|
export let isHtml: boolean;
|
|
export let globalTags: string[];
|
|
export let updatedTags: string[];
|
|
export let columnLabels: string[];
|
|
export let tagsColumn: number;
|
|
export let guidColumn: number;
|
|
export let preview: Generic.StringList[];
|
|
// Protobuf oneofs. Exactly one of these pairs is expected to be set.
|
|
export let notetypeColumn: number | null;
|
|
export let globalNotetype: ImportExport.CsvMetadata.MappedNotetype | null;
|
|
export let deckId: number | null;
|
|
export let deckColumn: number | null;
|
|
|
|
let dupeResolution: ImportExport.ImportCsvRequest.DupeResolution;
|
|
let lastNotetypeId = globalNotetype?.id;
|
|
|
|
$: columnOptions = getColumnOptions(
|
|
columnLabels,
|
|
preview[0].vals,
|
|
notetypeColumn,
|
|
deckColumn,
|
|
tagsColumn,
|
|
guidColumn,
|
|
);
|
|
$: getCsvMetadata(path, delimiter, undefined, isHtml).then((meta) => {
|
|
columnLabels = meta.columnLabels;
|
|
preview = meta.preview;
|
|
});
|
|
$: if (globalNotetype?.id !== lastNotetypeId) {
|
|
lastNotetypeId = globalNotetype?.id;
|
|
getCsvMetadata(path, delimiter, globalNotetype?.id).then((meta) => {
|
|
globalNotetype = meta.globalNotetype ?? null;
|
|
});
|
|
}
|
|
|
|
async function onImport(): Promise<void> {
|
|
await importExport.importCsv(
|
|
ImportExport.ImportCsvRequest.create({
|
|
path,
|
|
dupeResolution,
|
|
metadata: ImportExport.CsvMetadata.create({
|
|
delimiter,
|
|
forceDelimiter,
|
|
isHtml,
|
|
forceIsHtml,
|
|
globalTags,
|
|
updatedTags,
|
|
columnLabels,
|
|
tagsColumn,
|
|
guidColumn,
|
|
notetypeColumn,
|
|
globalNotetype,
|
|
deckColumn,
|
|
deckId,
|
|
}),
|
|
}),
|
|
);
|
|
}
|
|
</script>
|
|
|
|
<Container class="csv-page">
|
|
<Row --cols={2}>
|
|
<Col --col-size={1} breakpoint="md">
|
|
<Container>
|
|
<Header heading={tr.importingFile()} />
|
|
<Spacer --height="1.5rem" />
|
|
<DelimiterSelector bind:delimiter disabled={forceDelimiter} />
|
|
<HtmlSwitch bind:isHtml disabled={forceIsHtml} />
|
|
<Preview {columnOptions} {preview} />
|
|
</Container>
|
|
</Col>
|
|
<Col --col-size={1} breakpoint="md">
|
|
<Container>
|
|
<Header heading={tr.importingImportOptions()} />
|
|
<Spacer --height="1.5rem" />
|
|
{#if globalNotetype}
|
|
<NotetypeSelector
|
|
{notetypeNameIds}
|
|
bind:notetypeId={globalNotetype.id}
|
|
/>
|
|
{/if}
|
|
{#if deckId}
|
|
<DeckSelector {deckNameIds} bind:deckId />
|
|
{/if}
|
|
<DupeResolutionSelector bind:dupeResolution />
|
|
<Tags bind:globalTags bind:updatedTags />
|
|
</Container>
|
|
</Col>
|
|
<Col --col-size={1} breakpoint="md">
|
|
<Container>
|
|
<Header heading={tr.importingFieldMapping()} />
|
|
<Spacer --height="1.5rem" />
|
|
<FieldMapper {columnOptions} bind:globalNotetype bind:tagsColumn />
|
|
</Container>
|
|
</Col>
|
|
</Row>
|
|
<StickyFooter {path} {onImport} />
|
|
</Container>
|
|
|
|
<style lang="scss">
|
|
:global(.csv-page) {
|
|
--gutter-inline: 0.25rem;
|
|
|
|
:global(.row) {
|
|
// rows have negative margins by default
|
|
--bs-gutter-x: 0;
|
|
// ensure equal spacing between tall rows like
|
|
// dropdowns, and short rows like checkboxes
|
|
min-height: 3em;
|
|
}
|
|
}
|
|
</style>
|