Skip to content

Commit

Permalink
feat(umi-plugin): add styles, info area, group feature for side menu (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
PeachScript committed Nov 27, 2019
1 parent 7748662 commit 8ba10bb
Show file tree
Hide file tree
Showing 7 changed files with 239 additions and 29 deletions.
1 change: 1 addition & 0 deletions packages/umi-plugin-father-doc/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@
"remark-parse": "^7.0.1",
"remark-rehype": "^5.0.0",
"sylvanas": "^0.4.2",
"umi": "^2.12.3",
"umi-build-dev": "^1.16.5",
"umi-plugin-react": "^1.13.1",
"umi-types": "^0.5.4",
Expand Down
28 changes: 21 additions & 7 deletions packages/umi-plugin-father-doc/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,9 @@ import getRouteConfig from './routes/getRouteConfig';
import getMenuFromRoutes from'./routes/getMenuFromRoutes';

export interface IFatherDocOpts {
name?: string;
title?: string;
logo?: URL;
desc?: string;
include?: string[];
routes?: {
path: IRoute['path'];
Expand All @@ -34,10 +35,17 @@ function docConfigPlugin() {

export default function (api: IApi, opts: IFatherDocOpts) {
// apply default options
opts = Object.assign({
name: require(path.join(api.paths.cwd, 'package.json')).name,
include: ['docs'],
}, (api.config as any).doc, opts);
opts = Object.assign(
{
title: require(path.join(api.paths.cwd, 'package.json')).name,
include: ['docs'],
},
{
routes: api.config.routes
},
(api.config as any).doc,
opts,
);

const routeConfig = getRouteConfig(api.paths, opts);

Expand All @@ -48,7 +56,7 @@ export default function (api: IApi, opts: IFatherDocOpts) {
api.registerPlugin({
id: require.resolve('umi-plugin-react'),
apply: require('umi-plugin-react').default,
opts: { title: { defaultTitle: opts.name } },
opts: { title: { defaultTitle: opts.title } },
});

// repalce default routes with generated routes
Expand Down Expand Up @@ -82,7 +90,13 @@ export default function (api: IApi, opts: IFatherDocOpts) {
module
}, { menu: { items: ${
JSON.stringify(getMenuFromRoutes(routeConfig[0].routes)).replace(/\"/g, '\'')
} }, ...props })`;
} }, title: '${
opts.title
}', logo: '${
opts.logo || ''
}', desc: '${
opts.desc || ''
}', ...props })`;
}

return ret;
Expand Down
58 changes: 53 additions & 5 deletions packages/umi-plugin-father-doc/src/routes/getMenuFromRoutes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,26 +2,74 @@ import path from 'path';
import { IApi } from 'umi-types';

export interface IMenuItem {
path: string;
/**
* menu 的路由,仅该 menu 不为 group 时才存在
*/
path?: string;
/**
* group 前缀,仅该 menu 为 group 时才存在
*/
prefix?: string;
title: string;
meta?: { [key: string]: any };
children: IMenuItem[];
children?: IMenuItem[];
}

export default (routes: IApi['routes']) : IMenuItem[] => {
const menu: IMenuItem[] = [];
let menu: IMenuItem[] = [];

// convert routes to menu
routes.forEach((route) => {

if (route.path) {
menu.push({
path: route.path,
title: route.meta.title || path.parse(route.component as string).name,
meta: route.meta,
children: [],
});
}
});

// group by menu frontmatter
const [ungroup, groupedMapping] = menu.reduce((result, item) => {
if (item.meta?.group?.title) {
const key = item.meta.group.title;

if (result[1][key]) {
result[1][key].push(item);
} else {
result[1][key] = [item];
}
} else {
result[0].push(item);
}

return result;
}, [[], {}]);

menu = ungroup.concat(
Object
.keys(groupedMapping)
.map(key => {

return {
title: key,
prefix: groupedMapping[key].find(item => item.meta.group.path).meta.group.path,
children: groupedMapping[key],
meta: {
// use child order as group order
order: groupedMapping[key].find(item => item.meta.order)?.meta?.order || undefined,
},
}
})
);

// sort menu
menu.sort((prev, next) => {
const prevOrder = typeof(prev.meta.order) === 'number' ? prev.meta.order : 0;
const nextOrder = typeof(next.meta.order) === 'number' ? next.meta.order : 0;

return (prevOrder === nextOrder) ? 0 : (nextOrder - prevOrder);
});

return menu;
}
7 changes: 6 additions & 1 deletion packages/umi-plugin-father-doc/src/routes/getRouteConfig.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ export default (paths: IApi['paths'], opts: IFatherDocOpts): IRoute[] => {
path: '/',
component: path.join(__dirname, '../themes/default/layout.js'),
routes: [],
title: opts.name,
title: opts.title,
}];

if (opts.routes) {
Expand Down Expand Up @@ -47,6 +47,11 @@ export default (paths: IApi['paths'], opts: IFatherDocOpts): IRoute[] => {
if (route.meta.title) {
route.title = route.meta.title;
}

// apply group path
if (route.meta.group?.path) {
route.path = path.join(route.meta.group.path, route.path);
}
});

return config;
Expand Down
104 changes: 102 additions & 2 deletions packages/umi-plugin-father-doc/src/themes/default/layout.less
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
@import './markdown.less';

@s-menu-width: 300px;
@s-menu-width: 260px;
@s-content-margin: 64px;
@img-logo: '';

body {
margin: 0;
Expand All @@ -23,7 +24,106 @@ body {
left: 0;
bottom: 0;
width: @s-menu-width;
border-right: 1px solid #eee;
background-color: #F2F5FA;
box-sizing: border-box;
overflow: auto;

// shadow
&::after {
content: '';
position: absolute;
top: 0;
right: 0;
bottom: 0;
display: block;
width: 20px;
background: linear-gradient(to right, rgba(0, 0, 0, 0), rgba(0, 0, 0, 0.03));
}

.menuHeader {
padding-top: 40px;
text-align: center;
border-bottom: 1px solid @c-border;

.logo {
display: inline-block;
width: 66px;
height: 65px;
background: url(@img-logo) no-repeat 0/contain;
}

h1 {
margin: 10px 0 0;
color: @c-heading;
font-weight: 500;
line-height: 1.40625;
}

p {
margin: 0 0 10px;
color: lighten(@c-secondary, 10%);
}
}

> ul {
list-style: none;
margin: 0;
padding: 8px 0;

> li {
> a {
@c-active-bg: #E8ECF4;

display: block;
padding: 0 28px;
color: @c-heading;
font-size: 16px;
line-height: 46px;
transition: background 0.3s;

&:hover {
background: lighten(@c-active-bg, 2%);
}

&:active {
background: lighten(@c-active-bg, 1%);
}

&:global(.active) {
color: @c-primary;
background: #E8ECF4;
}

// disable click for group link
&[data-group] {
pointer-events: none;;
}
}

ul {
list-style: none;
margin: 6px 0;
padding: 0 28px 0 42px;

li a {
font-size: 14px;
line-height: 2;
color: @c-secondary;

&:hover {
color: lighten(@c-primary, 5%);
}

&:active {
color: lighten(@c-primary, 3%);
}

&:global(.active) {
color: @c-primary;
}
}
}
}
}
}
}
65 changes: 51 additions & 14 deletions packages/umi-plugin-father-doc/src/themes/default/layout.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,38 +2,75 @@
/// <reference path="../typings/global.d.ts" />
/// <reference path="../typings/typings.d.ts" />

import React, { Component, MouseEvent } from 'react';
import { IMenuItem } from '../../routes/getMenuFromRoutes';
import React, { Component } from 'react';
import Link from 'umi/link';
import NavLink from 'umi/navlink';
import 'prismjs/themes/prism.css';
import { IMenuItem } from '../../routes/getMenuFromRoutes';
import styles from './layout.less';

export interface ILayoutProps {
title: string;
logo?: string;
desc?: string;
menu: {
items: IMenuItem[];
};
}

export default class Layout extends Component<ILayoutProps> {
handleMenuItemClick = (e: MouseEvent, path: string) => {
window.g_history.push(path);
e.preventDefault();
}

render () {
const { children, menu } = this.props;
const { children, menu, logo, title, desc } = this.props;

return (
<div className={styles.wrapper}>
<div className={styles.menu}>
<div className={styles.menuHeader}>
<Link
to="/"
className={styles.logo}
style={{
backgroundImage: logo && `url('${logo}')`,
}}
/>
<h1>{title}</h1>
<p>{desc}</p>
</div>
<ul>
{menu.items.map((item) => (
<li>
<a
href={item.path}
onClick={e => this.handleMenuItemClick(e, item.path)}
>
{item.title}
</a>
{
item.path
? (
// render single routes
<NavLink to={item.path} exact>
{item.title}
</NavLink>
)
: (
// render child routes
<>
{/* use NavLink for active, but disable click by css */}
<NavLink
to={item.prefix}
data-group
>
{item.title}
</NavLink>
{item.children && item.children.length && (
<ul>
<li>
{item.children.map((child) => (
<NavLink to={child.path} exact>
{child.title}
</NavLink>
))}
</li>
</ul>
)}
</>
)
}
</li>
))}
</ul>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -143,5 +143,10 @@
opacity: 0.9;
}
}

// images
img {
max-width: 100%;
}
}
}

0 comments on commit 8ba10bb

Please sign in to comment.