【Svelte从入门到精通】实战篇——TodoList之页面布局

搭建环境

首先使用vite搭建项目。

bash 复制代码
npm create vite@latest svelte-todolist -- --template svelte

安装tailwind,这一步读者可自行选择自己喜欢的样式工具。

bash 复制代码
npm install -D tailwindcss postcss autoprefixer
bash 复制代码
npx tailwindcss init -p

删除lib/Counter.svelte,清空App.svelte的内容,清空app.css的内容。

将tailwind的基本内容添加进app.css:

css 复制代码
@tailwind base;
@tailwind components;
@tailwind utilities;

修改tailwind.config.js的内容

javascript 复制代码
/** @type {import('tailwindcss').Config} */
export default {
  content: ["./src/**/*.{html,js,svelte}"],
  theme: {
    extend: {},
  },
  plugins: [],
}

然后我们往App.svelte里添加一行内容

html 复制代码
<div class="text-rose-400">app</div>

如无意外,相信读者朋友们能在页面上看到玫瑰色的app字样。

dribble上获取灵感,挑选了一些好看的颜色。在tailwind.config.js上配置:

javascript 复制代码
/** @type {import('tailwindcss').Config} */
export default {
  content: ["./src/**/*.{html,js,svelte}"],
  theme: {
    extend: {
      colors: {
        'color1': '#FDEEE1',
        'color2': '#E62737',
        'color3': '#FF7171',
        'color4': '#FFADA8',
      }
    },
  },
  plugins: [],
}

数据结构

首先我们需要确定待办列表的一个待办项的数据结构,目前笔者定义如下:

typescript 复制代码
interface Todo {
	id: number,
	text: string,
	done: boolean,
}

布局

我们将页面分成三部分,笔者在这里用不同的颜色进行了标注。分别是输入框、Tab切换以及待办列表展示。 整体使用flex布局。

基本框架

在App.svelte中

html 复制代码
<script>
  let divClass =
    "flex flex-col w-[500px] h-[700px] overflow-hidden rounded-lg mx-auto p-4 bg-black";
</script>

<div class={divClass}></div>

在页面上,看到一个居中的暖色块。

输入框

新建一个Input.svelte文件,内容如下

html 复制代码
<script>
  let inputClass = "flex-1 h-full border rounded-lg mr-4 px-4 caret-color2 focus:outline-color2";
  let buttonClass = "w-[100px] h-full rounded-lg flex items-center justify-center bg-color2 text-white flex-shrink-0 hover:cursor-pointer"
</script>

<div class="flex h-12">
  <input
    type="text"
    class={inputClass}
  />
  <button
    class={buttonClass}>Add</button
  >
</div>```

然后我们在App.svelte中引入该组件
```html
<script>
  import Input from './Input.svelte';

  let divClass =
    "flex flex-col w-[500px] h-[700px] overflow-hidden rounded-lg mx-auto p-4 bg-color1";
</script>

<div class={divClass}>
  <Input />
</div>

此时相信读者们能够看到以下内容:

Tabs

新建Tabs.svelte组件,添加内容:

html 复制代码
<script>
  let tabsClass = "grid grid-cols-3 gap-4 h-12 my-4";
  let tabClass =
    "h-12 flex items-center justify-center hover:cursor-pointer rounded-lg";
</script>

<div class={tabsClass}>
  <button class={tabClass}>Todo</button>
  <button class={tabClass}>Done</button>
  <button class={tabClass}>All</button>
</div>

App.svelte中引入该组件后,页面展示如下:

继续完善Tabs.svelte的内容:

html 复制代码
<script>
  let tabsClass = "grid grid-cols-3 gap-4 h-12 my-4";
  let tabClass =
    "h-12 flex items-center justify-center hover:cursor-pointer rounded-lg";

  let currentTab = 'todo'; // todo | done | all

  const changeTab = (tab) => currentTab = tab;
</script>

<div class={tabsClass}>
  <button class={tabClass} on:click={() => changeTab('todo')}>Todo</button>
  <button class={tabClass} on:click={() => changeTab('done')}>Done</button>
  <button class={tabClass} on:click={() => changeTab('all')}>All</button>
</div>
<div class="flex-1 overflow-auto">
  <slot>{currentTab}</slot>
</div>

笔者设置了currentTab变量,用来记录当前激活的是哪个tab,为三个tab按钮绑定更新tab的事件,然后添加了slot,用于待办列表的展示

列表

首先我们实现Item.svelte内的展示:

html 复制代码
<script>
  export let index = 0;
  export let name = "";
  export let done = false;

  let divClass =
    "flex items-center bg-white rounded-lg h-12 px-4 mb-4";
  let indexClass = "w-12 text-left flex-shrink-0";
  let nameClass = "flex-1 overflow-hidden text-ellipsis";
  let checkboxWrapClass = "w-12 flex-shrink-0 flex justify-end items-center";
</script>

<div class={divClass}>
  <span class={indexClass}>{index}</span>
  <span class={nameClass}>{name}</span>
  <span class={checkboxWrapClass}>
    <input
      type="checkbox"
      checked={done}
    />
  </span>
</div>

<style>
  input[type="checkbox"] {
    appearance: none;
    background-color: #fff;
    margin: 0;
    font: inherit;
    width: 1.15em;
    height: 1.15em;
    border: 0.15em solid currentColor;
    border-radius: 0.15em;
    transform: translateY(-0.075em);
    display: grid;
    place-content: center;
  }

  input[type="checkbox"]::before {
    content: "";
    width: 0.65em;
    height: 0.65em;
    transform: scale(0);
    transition: 120ms transform ease-in-out;
    box-shadow: inset 1em 1em #E62737;
  }

  input[type="checkbox"]:checked::before {
    transform: scale(1);
  }
</style>

待办项Item.svelte接收三个prop:index、name和done。 这里我们用自己的style重写了checkbox的样式,使其看上去更符合我们页面的主题。

接着我们实现List.svelte的内容:

html 复制代码
<script>
  import Item from './Item.svelte';

  let list = [{
    id: 1,
    text: 'hello',
    done: false
  }, {
    id: 2,
    text: 'world',
    done: true
  }];
</script>

{#each list as {text, done}, i}
  <Item index={i} name={text} done={done} />
  {:else}
    No data
{/each}

这里我们mock了两个待办项的假数据,一个状态未完成,另一个状态已完成。 最后,我们在App.svelte中引用List.svelte并使用它。

html 复制代码
<script>
  import Input from './Input.svelte';
  import Tabs from './Tabs.svelte';
  import List from './List.svelte';

  let divClass =
    "flex flex-col w-[500px] h-[700px] overflow-hidden rounded-lg mx-auto p-4 bg-color1";
</script>

<div class={divClass}>
  <Input />
  <Tabs>
    <List />
  </Tabs>
</div>

至此,页面上能够看见我们最开始设计的布局结构。

当然,此时的页面功能仍不完整,比如无法添加待办项、Tab切换没有样式、待办项勾选后没有放置到对应的Tab栏等等。在下一章,我们将完善剩余的功能。

小结

本章我们完成了:

  • TodoList项目环境的搭建
  • TailwindCss的使用
  • 一个待办项基本的数据结构
  • 页面布局的定义和初步实现
相关推荐
逐·風3 小时前
unity关于自定义渲染、内存管理、性能调优、复杂物理模拟、并行计算以及插件开发
前端·unity·c#
Devil枫3 小时前
Vue 3 单元测试与E2E测试
前端·vue.js·单元测试
尚梦4 小时前
uni-app 封装刘海状态栏(适用小程序, h5, 头条小程序)
前端·小程序·uni-app
GIS程序媛—椰子4 小时前
【Vue 全家桶】6、vue-router 路由(更新中)
前端·vue.js
前端青山5 小时前
Node.js-增强 API 安全性和性能优化
开发语言·前端·javascript·性能优化·前端框架·node.js
毕业设计制作和分享5 小时前
ssm《数据库系统原理》课程平台的设计与实现+vue
前端·数据库·vue.js·oracle·mybatis
从兄6 小时前
vue 使用docx-preview 预览替换文档内的特定变量
javascript·vue.js·ecmascript
清灵xmf7 小时前
在 Vue 中实现与优化轮询技术
前端·javascript·vue·轮询
大佩梨7 小时前
VUE+Vite之环境文件配置及使用环境变量
前端
GDAL7 小时前
npm入门教程1:npm简介
前端·npm·node.js