第一个 Angular 项目 - 添加路由

第一个 Angular 项目 - 添加路由

前置项目是 第一个 Angular 项目 - 添加服务,之前的切换页面使用的是 ngIf 对渲染的组件进行判断,从而完成渲染。这一步的打算是添加路由,同时添加 edit recipe 的功能(同样通过路由实现)

用到的内容为:


配置基础路由

这里修改的文件有以下文件:

其中:

  • header
  • recipe-details
  • recipe-item

三个 HTML Template 修改的内容为删除 a 标签中的 href 属性,并添加 style="cursor: pointer" 维持原本的样式,这里不多赘述

其他修改如下:

  • app-routing

    typescript 复制代码
    const appRoutes: Routes = [
      { path: '', redirectTo: '/recipes', pathMatch: 'full' },
      { path: 'recipes', component: RecipesComponent },
      { path: 'shopping-list', component: ShoppingListComponent },
    ];
    
    @NgModule({
      imports: [RouterModule.forRoot(appRoutes)],
      exports: [RouterModule],
    })
    export class AppRoutingModule {}

    这里主要的修改在 routing 两篇笔记中都提过,最低层次的配置需求需要包含路径和对应的组件,redirectTo 中使用 pathMatch 因为 Angular 的路由是通过前缀匹配的,如果不添加这个属性,那么从首页登陆就会 infinite loop,从而报错

  • app V 层

    删除原本的 ngIf 部分,替换为 <router-outlet></router-outlet>

  • app module

    imports 中新增 AppRoutingModule

实现后效果如下:

添加子路由

设置如下:

typescript 复制代码
const appRoutes: Routes = [
  {
    path: 'recipes',
    component: RecipesComponent,
    children: [
      { path: '', component: RecipeStartComponent },
      { path: 'new', component: RecipeEditComponent },
      { path: ':id', component: RecipeDetailComponent },
      { path: ':id/edit', component: RecipeEditComponent },
    ],
  },
];

实现后效果如下:

这里基本上没什么特别难的,唯一需要注意的就是 :idnew 的顺序。:id 也算是一个 wildcard,它会将所有的参数都默认为变量名为 id 的值,并且传到 Params 里去

修改获取 recipe

这里是 recipe details 的实现,修改如下:

typescript 复制代码
export class RecipeDetailComponent implements OnInit {
  recipeId: string;
  activeRecipe: Recipe;

  constructor(
    private ingredientService: IngredientService,
    private recipeService: RecipeService,
    private activatedRouter: ActivatedRoute
  ) {}

  ngOnInit(): void {
    this.activatedRouter.params.subscribe((params: Params) => {
      this.recipeId = params.id;
      this.activeRecipe = this.recipeService.getRecipeById(this.recipeId);
    });
  }
}

service 中的实现就是返回 array 中的数据就是了,这里实现不多赘述

其他需要做的地方就是 recipe-list 中传值的 onSelect,以及对应的数据传输(@Input & @Output)

动态拼接路由

当前项目还是一个 dummy 项目,所以没法做到直接从 recipe 中获取 id,所以这里要做的就是从 ngFor 中将 index 传到下一个组件,并且在下一个组件中获取 id:

  • recipe list 修改:

    html 复制代码
    <app-recipe-item
      *ngFor="let recipe of recipes; let i = index"
      [recipe]="recipe"
      [id]="i + 1"
    ></app-recipe-item>
  • recipe item 修改:

    html 复制代码
    <a
      style="cursor: pointer"
      [routerLink]="[id]"
      routerLinkActive="active"
      class="list-group-item clearfix"
    ></a>

对应的 @Input 绑定就不实现了,这部分内容在 Angular 基础 - 自定义事件 & 自定义属性 中;ngFor 的复习在 Angular 基础 - 指令(directives)

此时的实现如下:

可以看到和之前的实现基本没有变化,除了 URL 路径的变动

edit recipe 实现

具体的功能会等到学完表单再实现,不过这里通过 ngOnInit 简单的判断一下当前页面是新建页面还是编辑页面:

typescript 复制代码
@Component({
  selector: 'app-recipe-edit',
  templateUrl: './recipe-edit.component.html',
  styleUrl: './recipe-edit.component.css',
})
export class RecipeEditComponent implements OnInit {
  id: number;
  editMode = false;

  constructor(private route: ActivatedRoute) {}

  ngOnInit() {
    this.route.params.subscribe((params: Params) => {
      this.id = +params.id;
      this.editMode = isNaN(this.id);
    });
  }
}

这里的逻辑处理相对而言比较暴力,简单的说就是如果传进来的参数不是 new------有 new 走不进 :id------并且无法转成数字,那么就肯定不是编辑模式,这种情况下默认转为新建模式。

具体的逻辑处理可以写很细,这里只是大致实现一下

new 和 edit 的路由实现

这个部分比较简单,Angular 似乎对 routerLink 得绑定没什么限制,注意一下绝对路径/相对路径就能实现了:

⚠️:另一个实现方式是绑定 onclick handler,随后通过 this.router.navigate 进行实现。二者都可以,鉴于现在并没有什么复杂的逻辑需求,我就直接绑定 routerLink

相关推荐
禅思院10 小时前
前端部署“三层漏斗”完全指南:从CI/CD到自动回滚的工程化实战【开题】
前端·架构·前端框架
快乐肚皮11 小时前
深入理解Loop Engineering
前端·后端
半个落月11 小时前
从递归到快速排序:用 JavaScript 把分治思想讲明白
javascript·算法·面试
风骏时光牛马11 小时前
VHDL十大经典基础功能设计实例代码合集
前端
小兔崽子去哪了11 小时前
Vue3 + Pinia 集成 IGV.js 实现 BAM 文件在线浏览
javascript·vue.js·后端
hunterandroid11 小时前
Notification 通知:从基础到渠道适配
前端
孟陬11 小时前
Claude Code 巧思 `Ctrl+S` 暂存键
前端·后端
PedroQue9912 小时前
V1.6.1性能优化:高频路径提速与代码精简
前端·uni-app
猩猩程序员12 小时前
将 LiteLLM 迁移到 Rust —— 构建最快、最轻量的 AI Gateway
前端
小月土星12 小时前
JavaScript 快速排序:从 pivot、双指针到分治思想
javascript·算法·面试