2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

脱Excel!管理しやすいドキュメントツールを作ってみた件

Posted at

ドキュメント管理、だるくないっすか?😅

おはこんばんちは。
ドキュメント管理、非常に面倒ですよね。
何が面倒かってお客様との打ち合わせが重なった際に変更履歴が追えないのはきついっすよね。

例えば…
image.png

ChatGPTくんも詳しくて例が出てきましたよ。闇深いっすね

こんな内容を解決していくためのツールを作りました。

深淵を覗かなければ、深淵はこちらを覗かない👿

以下のような形で構成してみました。

  • Server
    • Vercel
  • 言語
    • Next.js (React.js)
  • ドキュメント記載ツール
    • Markdown
  • 共通仕様
    • プロジェクト毎に簡単にサイトを作成できるようにしたい
    • 外出先でもレスポンシブ対応にしたい
    • ページの作成はMarkdownで作成できるようにしたい
    • Basic認証を使って簡易的にでも閲覧者を限定したい
    • Markdownファイルは/docsに格納していく

こうすれば深淵はこちらを覗かないように出来るか?と検討

Let's Coding!💻

こんな感じでやっていきます

[slug].tsx
import { getFileContent, getFileName } from "@/lib/fileUtil";
import { toHtml } from "@/lib/markdownUtil";
import { GetServerSidePropsContext } from "next";
import { NextRequest, NextResponse } from "next/server";
import { useEffect } from "react";

export default function Content({ content }: { content: string }) {

  return (
    <div className="content" dangerouslySetInnerHTML={{
      __html: content,
    }}></div>
  )
}


export const getServerSideProps = async (context: GetServerSidePropsContext) => {
  const query = decodeURI(context.query.slug as string);
  const fileNames = getFileName();

  if (query.match(/.md/)) {
    return {
      props: {
        fileNames,
        isError: true,
        title: query,
        content: null,
      }
    }
  }
  const markdownFile = getFileContent(query);
  if (markdownFile == null) {
    return {
      props: {
        fileNames,
        isError: true,
        title: query,
      }
    }
  }
  const content = await toHtml(markdownFile);
  return {
    props: {
      fileNames,
      isError: false,
      title: query,
      content: content,
    },
  }
};

MarkdownファイルをMarked.jsで変換して、HTML側に描画していきます。
例えば/要件定義書にアクセスされた場合は/docs/要件定義書.mdへアクセスするようにします。

_app.tsx

interface PageProps {
  fileNames: string[];
  isError?: boolean,
  title?: string,
  content?: string,
}


const App = ({ Component, pageProps: _PageProps }: AppProps) => {
  const {
    fileNames,
    isError,
    title,
  } = _PageProps as PageProps;

  const titleTag = `${process.env.NEXT_PUBLIC_PROJECT_NAME} | ${title || 'Home'}` ;

  return (
    <>
      <Head>
        <title>{titleTag}</title>
      </Head>
      <Navbar
        pageTitle="Home"
        navbarStart={(fileNames || []).map(x => ({
          show: true,
          type: 'LinkItem',
          label: x,
          pathname: x,
        }))}
      />
      <div className="container">
        <section className="section">
          <div className="hero">
            {
              !isError ? (
                <Component {..._PageProps}/>
              ) : <Error />
            }
          </div>
        </section>
      </div>
    </>
  );
}

export default App;

Navbarの項目はMarkdownのファイル名で作成されるようにしていきます。

画像について📷

Vercelは画像関連で読み込みが遅いので、API側で読み込みをしても504エラーが出てしまいます。
(ここら辺原因を調べてみても不明でした😫)
しかし画像はアップロードしたいですよね。
なので以下のようにしていきます。

package.json
  "scripts": {
    "dev": "next dev",
    "build": "yarn run cp && next build",
    "start": "next start",
    "lint": "next lint",
    "cp:pngImg": "cpx \"docs/img/*.png\" public/img/",
    "cp:jpegImg": "cpx \"docs/img/*.jpeg\" public/img/",
    "cp:png": "cpx \"docs/*.png\" public/",
    "cp:jpeg": "cpx \"docs/*.jpeg\" public/",
    "cp": "yarn run cp:pngImg && yarn run cp:jpegImg && yarn run cp:png && yarn run cp:jpeg"
  }
  

こんな感じでコピーしていきますよ。
デプロイの際に/docsに配置している画像をpublicにコピーしていくとWebアプリ上でも閲覧ができるようになります。

完成した画面をみてみよう 🥰

image.png

こんな感じのファイルが

image.png

こんな感じで表示されます。

簡単だけど便利だねー。
ぜひ作ってみてね!!


オブジェクティブグループではXの投稿も平日毎日行っています!
IT関連の小ネタや便利技から、日常のアニメ・ゲーム布教なども幅広く投稿してるので、
ご興味のある方は是非フォロー・いいねをお願いします。
https://x.com/obg_ocr

2
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?