美文网首页开发组件收藏
组件分享之前端组件——甘特图时间表时间线日历组件

组件分享之前端组件——甘特图时间表时间线日历组件

作者: cn華少 | 来源:发表于2022-02-02 19:37 被阅读0次

    组件分享之前端组件——甘特图时间表时间线日历组件

    背景

    近期正在探索前端、后端、系统端各类常用组件与工具,对其一些常见的组件进行再次整理一下,形成标准化组件专题,后续该专题将包含各类语言中的一些常用组件。欢迎大家进行持续关注。

    组件基本信息

    内容

    本次分享的组件是甘特图时间表时间线日历组件

    Gantt-elastic -Javascript 甘特图(可编辑,响应式)
    vue、jquery、vanilla js等框架的Javascript甘特图

    其后续延伸的组件 gantt-schedule-timeline-calendar是收费版本的,可以根据实际需要进行选用。

    当前组件样子如下:


    image.png

    测试案例如下:

    <!DOCTYPE html>
    <html charset="utf-8">
      <head>
        <meta charset="utf-8" />
        <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0" />
        <title>GanttElastic editor demo</title>
        <script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
        <script src="https://cdn.jsdelivr.net/npm/dayjs"></script>
    
        <script src="https://unpkg.com/gantt-elastic/dist/GanttElastic.umd.js"></script>
        <script src="https://unpkg.com/gantt-elastic-header/dist/Header.umd.js"></script>
      </head>
    
      <body>
        <div style="width:100%;height:100%">
          <div id="app" v-if="!destroy">
            <gantt-elastic :tasks="tasks" :options="options" :dynamic-style="dynamicStyle">
              <gantt-header slot="header"></gantt-header>
              <gantt-footer slot="footer"></gantt-footer>
            </gantt-elastic>
          </div>
        </div>
    
        <script>
          // just helper to get current dates
          function getDate(hours) {
            const currentDate = new Date();
            const currentYear = currentDate.getFullYear();
            const currentMonth = currentDate.getMonth();
            const currentDay = currentDate.getDate();
            const timeStamp = new Date(currentYear, currentMonth, currentDay, 0, 0, 0).getTime();
            return new Date(timeStamp + hours * 60 * 60 * 1000).getTime();
          }
    
          let tasks = [
            {
              id: 1,
              label: 'Make some noise',
              user:
                '<a href="https://www.google.com/search?q=John+Doe" target="_blank" style="color:#0077c0;">John Doe</a>',
              start: getDate(-24 * 5),
              duration: 15 * 24 * 60 * 60 * 1000,
              progress: 85,
              type: 'project',
              //collapsed: true,
            },
            {
              id: 2,
              label: 'With great power comes great responsibility',
              user:
                '<a href="https://www.google.com/search?q=Peter+Parker" target="_blank" style="color:#0077c0;">Peter Parker</a>',
              parentId: 1,
              start: getDate(-24 * 4),
              duration: 4 * 24 * 60 * 60 * 1000,
              progress: 50,
              type: 'milestone',
              collapsed: true,
              style: {
                base: {
                  fill: '#1EBC61',
                  stroke: '#0EAC51',
                },
                /*'tree-row-bar': {
                  fill: '#1EBC61',
                  stroke: '#0EAC51'
                },
                'tree-row-bar-polygon': {
                  stroke: '#0EAC51'
                }*/
              },
            },
            {
              id: 3,
              label: 'Courage is being scared to death, but saddling up anyway.',
              user:
                '<a href="https://www.google.com/search?q=John+Wayne" target="_blank" style="color:#0077c0;">John Wayne</a>',
              parentId: 2,
              start: getDate(-24 * 3),
              duration: 2 * 24 * 60 * 60 * 1000,
              progress: 100,
              type: 'task',
            },
            {
              id: 4,
              label: 'Put that toy AWAY!',
              user:
                '<a href="https://www.google.com/search?q=Clark+Kent" target="_blank" style="color:#0077c0;">Clark Kent</a>',
              start: getDate(-24 * 2),
              duration: 2 * 24 * 60 * 60 * 1000,
              progress: 50,
              type: 'task',
              dependentOn: [3],
            },
          ];
    
          let options = {
            maxRows: 100,
            maxHeight: 300,
            title: {
              label: 'Your project title as html (link or whatever...)',
              html: false,
            },
            row: {
              height: 24,
            },
            calendar: {
              hour: {
                display: false,
              },
            },
            chart: {
              progress: {
                bar: false,
              },
              expander: {
                display: true,
              },
            },
            taskList: {
              expander: {
                straight: false,
              },
              columns: [
                {
                  id: 1,
                  label: 'ID',
                  value: 'id',
                  width: 40,
                },
                {
                  id: 2,
                  label: 'Description',
                  value: 'label',
                  width: 200,
                  expander: true,
                  html: true,
                  events: {
                    click({ data, column }) {
                      alert('description clicked!\n' + data.label);
                    },
                  },
                },
                {
                  id: 3,
                  label: 'Assigned to',
                  value: 'user',
                  width: 130,
                  html: true,
                },
                {
                  id: 3,
                  label: 'Start',
                  value: (task) => dayjs(task.start).format('YYYY-MM-DD'),
                  width: 78,
                },
                {
                  id: 4,
                  label: 'Type',
                  value: 'type',
                  width: 68,
                },
                {
                  id: 5,
                  label: '%',
                  value: 'progress',
                  width: 35,
                  style: {
                    'task-list-header-label': {
                      'text-align': 'center',
                      width: '100%',
                    },
                    'task-list-item-value-container': {
                      'text-align': 'center',
                      width: '100%',
                    },
                  },
                },
              ],
            },
            /*locale:{
              name: 'pl', // name String
              weekdays: 'Poniedziałek_Wtorek_Środa_Czwartek_Piątek_Sobota_Niedziela'.split('_'), // weekdays Array
              weekdaysShort: 'Pon_Wto_Śro_Czw_Pią_Sob_Nie'.split('_'), // OPTIONAL, short weekdays Array, use first three letters if not provided
              weekdaysMin: 'Pn_Wt_Śr_Cz_Pt_So_Ni'.split('_'), // OPTIONAL, min weekdays Array, use first two letters if not provided
              months: 'Styczeń_Luty_Marzec_Kwiecień_Maj_Czerwiec_Lipiec_Sierpień_Wrzesień_Październik_Listopad_Grudzień'.split('_'), // months Array
              monthsShort: 'Sty_Lut_Mar_Kwi_Maj_Cze_Lip_Sie_Wrz_Paź_Lis_Gru'.split('_'), // OPTIONAL, short months Array, use first three letters if not provided
              ordinal: n => `${n}`, // ordinal Function (number) => return number + output
              relativeTime: { // relative time format strings, keep %s %d as the same
                future: 'za %s', // e.g. in 2 hours, %s been replaced with 2hours
                past: '%s temu',
                s: 'kilka sekund',
                m: 'minutę',
                mm: '%d minut',
                h: 'godzinę',
                hh: '%d godzin', // e.g. 2 hours, %d been replaced with 2
                d: 'dzień',
                dd: '%d dni',
                M: 'miesiąc',
                MM: '%d miesięcy',
                y: 'rok',
                yy: '%d lat'
              }
           }*/
          };
    
          // create instance
          const app = new Vue({
            components: {
              'gantt-header': Header,
              'gantt-elastic': GanttElastic,
              'gantt-footer': {
                template: `<span>this is a footer</span>`,
              },
            },
            data: {
              tasks: tasks.map((task) => Object.assign({}, task)),
              options,
              dynamicStyle: {
                'task-list-header-label': {
                  'font-weight': 'bold',
                },
              },
              destroy: false,
            },
          });
    
          // gantt state which will be updated in realtime
          let ganttState, ganttInstance;
    
          // listen to 'gantt-elastic.ready' or 'gantt-elastic.mounted' event
          // to get the gantt state for real time modification
          app.$on('gantt-elastic-ready', (ganttElasticInstance) => {
            ganttInstance = ganttElasticInstance;
    
            ganttInstance.$on('tasks-changed', (tasks) => {
              app.tasks = tasks;
            });
            ganttInstance.$on('options-changed', (options) => {
              app.options = options;
            });
            ganttInstance.$on('dynamic-style-changed', (style) => {
              app.dynamicStyle = style;
            });
    
            ganttInstance.$on('chart-task-mouseenter', ({ data, event }) => {
              console.log('task mouse enter', { data, event });
            });
            ganttInstance.$on('updated', () => {
              //console.log('gantt view was updated');
            });
            ganttInstance.$on('destroyed', () => {
              //console.log('gantt was destroyed');
            });
            ganttInstance.$on('times-timeZoom-updated', () => {
              console.log('time zoom changed');
            });
            ganttInstance.$on('taskList-task-click', ({ event, data, column }) => {
              console.log('task list clicked! (task)', { data, column });
            });
          });
    
          // mount gantt to DOM
          app.$mount('#app');
        </script>
      </body>
    </html>
    
    本文声明:
    88x31.png
    知识共享许可协议
    本作品由 cn華少 采用 知识共享署名-非商业性使用 4.0 国际许可协议 进行许可。

    相关文章

      网友评论

        本文标题:组件分享之前端组件——甘特图时间表时间线日历组件

        本文链接:https://www.haomeiwen.com/subject/gdtnhrtx.html