瀏覽代碼

add Mapping dictionary [WIP]

nullpos 9 年之前
父節點
當前提交
859bc0dab6
共有 29 個文件被更改,包括 2151 次插入1 次删除
  1. 123 0
      src/main/java/org/codelibs/fess/app/pager/MappingPager.java
  2. 73 0
      src/main/java/org/codelibs/fess/app/service/MappingService.java
  3. 447 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/AdminDictMappingAction.java
  4. 46 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/CreateForm.java
  5. 23 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/DownloadForm.java
  6. 34 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/EditForm.java
  7. 28 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/SearchForm.java
  8. 33 0
      src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/UploadForm.java
  9. 38 0
      src/main/java/org/codelibs/fess/dict/mapping/MappingCreator.java
  10. 338 0
      src/main/java/org/codelibs/fess/dict/mapping/MappingFile.java
  11. 140 0
      src/main/java/org/codelibs/fess/dict/mapping/MappingItem.java
  12. 15 0
      src/main/java/org/codelibs/fess/mylasta/action/FessHtmlPath.java
  13. 45 0
      src/main/java/org/codelibs/fess/mylasta/action/FessLabels.java
  14. 52 0
      src/main/java/org/codelibs/fess/mylasta/action/FessMessages.java
  15. 15 0
      src/main/java/org/codelibs/fess/mylasta/direction/FessConfig.java
  16. 1 0
      src/main/resources/fess_config.properties
  17. 7 1
      src/main/resources/fess_dict.xml
  18. 15 0
      src/main/resources/fess_label.properties
  19. 15 0
      src/main/resources/fess_label_en.properties
  20. 17 0
      src/main/resources/fess_label_ja.properties
  21. 15 0
      src/main/resources/fess_label_ko.properties
  22. 3 0
      src/main/resources/fess_message.properties
  23. 3 0
      src/main/resources/fess_message_en.properties
  24. 3 0
      src/main/resources/fess_message_ja.properties
  25. 156 0
      src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping.jsp
  26. 127 0
      src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_details.jsp
  27. 103 0
      src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_download.jsp
  28. 129 0
      src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_edit.jsp
  29. 107 0
      src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_upload.jsp

+ 123 - 0
src/main/java/org/codelibs/fess/app/pager/MappingPager.java

@@ -0,0 +1,123 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.pager;
+
+import org.codelibs.fess.Constants;
+
+import java.io.Serializable;
+import java.util.List;
+
+public class MappingPager implements Serializable {
+
+    private static final long serialVersionUID = 1L;
+
+    private int allRecordCount;
+
+    private int allPageCount;
+
+    private boolean existPrePage;
+
+    private boolean existNextPage;
+
+    private List<Integer> pageNumberList;
+
+    private int pageSize;
+
+    private int currentPageNumber;
+
+    public String id;
+
+    public void clear() {
+        allRecordCount = 0;
+        allPageCount = 0;
+        existPrePage = false;
+        existNextPage = false;
+        pageSize = getDefaultPageSize();
+        currentPageNumber = getDefaultCurrentPageNumber();
+
+        id = null;
+    }
+
+    protected int getDefaultPageSize() {
+        return Constants.DEFAULT_ADMIN_PAGE_SIZE;
+    }
+
+    protected int getDefaultCurrentPageNumber() {
+        return 1;
+    }
+
+    public int getAllRecordCount() {
+        return allRecordCount;
+    }
+
+    public void setAllRecordCount(final int allRecordCount) {
+        this.allRecordCount = allRecordCount;
+    }
+
+    public int getAllPageCount() {
+        return allPageCount;
+    }
+
+    public void setAllPageCount(final int allPageCount) {
+        this.allPageCount = allPageCount;
+    }
+
+    public boolean isExistPrePage() {
+        return existPrePage;
+    }
+
+    public void setExistPrePage(final boolean existPrePage) {
+        this.existPrePage = existPrePage;
+    }
+
+    public boolean isExistNextPage() {
+        return existNextPage;
+    }
+
+    public void setExistNextPage(final boolean existNextPage) {
+        this.existNextPage = existNextPage;
+    }
+
+    public int getPageSize() {
+        if (pageSize <= 0) {
+            pageSize = getDefaultPageSize();
+        }
+        return pageSize;
+    }
+
+    public void setPageSize(final int pageSize) {
+        this.pageSize = pageSize;
+    }
+
+    public int getCurrentPageNumber() {
+        if (currentPageNumber <= 0) {
+            currentPageNumber = getDefaultCurrentPageNumber();
+        }
+        return currentPageNumber;
+    }
+
+    public void setCurrentPageNumber(final int currentPageNumber) {
+        this.currentPageNumber = currentPageNumber;
+    }
+
+    public List<Integer> getPageNumberList() {
+        return pageNumberList;
+    }
+
+    public void setPageNumberList(final List<Integer> pageNumberList) {
+        this.pageNumberList = pageNumberList;
+    }
+}

+ 73 - 0
src/main/java/org/codelibs/fess/app/service/MappingService.java

@@ -0,0 +1,73 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.service;
+
+import org.codelibs.core.beans.util.BeanUtil;
+import org.codelibs.fess.Constants;
+import org.codelibs.fess.app.pager.MappingPager;
+import org.codelibs.fess.dict.DictionaryFile.PagingList;
+import org.codelibs.fess.dict.DictionaryManager;
+import org.codelibs.fess.dict.mapping.MappingFile;
+import org.codelibs.fess.dict.mapping.MappingItem;
+import org.dbflute.optional.OptionalEntity;
+
+import javax.annotation.Resource;
+import java.util.Collections;
+import java.util.List;
+
+public class MappingService {
+    @Resource
+    protected DictionaryManager dictionaryManager;
+
+    public List<MappingItem> getMappingList(final String dictId, final MappingPager mappingPager) {
+        return getMappingFile(dictId).map(file -> {
+            final int pageSize = mappingPager.getPageSize();
+            final PagingList<MappingItem> mappingList = file.selectList((mappingPager.getCurrentPageNumber() - 1) * pageSize, pageSize);
+
+            // update pager
+                BeanUtil.copyBeanToBean(mappingList, mappingPager, option -> option.include(Constants.PAGER_CONVERSION_RULE));
+                mappingList.setPageRangeSize(5);
+                mappingPager.setPageNumberList(mappingList.createPageNumberList());
+
+                return (List<MappingItem>) mappingList;
+            }).orElse(Collections.emptyList());
+    }
+
+    public OptionalEntity<MappingFile> getMappingFile(final String dictId) {
+        return dictionaryManager.getDictionaryFile(dictId).filter(file -> file instanceof MappingFile)
+                .map(file -> OptionalEntity.of((MappingFile) file)).orElse(OptionalEntity.empty());
+    }
+
+    public OptionalEntity<MappingItem> getMappingItem(final String dictId, final long id) {
+        return getMappingFile(dictId).map(file -> file.get(id).get());
+    }
+
+    public void store(final String dictId, final MappingItem mappingItem) {
+        getMappingFile(dictId).ifPresent(file -> {
+            if (mappingItem.getId() == 0) {
+                file.insert(mappingItem);
+            } else {
+                file.update(mappingItem);
+            }
+        });
+    }
+
+    public void delete(final String dictId, final MappingItem mappingItem) {
+        getMappingFile(dictId).ifPresent(file -> {
+            file.delete(mappingItem);
+        });
+    }
+}

+ 447 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/AdminDictMappingAction.java

@@ -0,0 +1,447 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.codelibs.core.beans.util.BeanUtil;
+import org.codelibs.core.lang.StringUtil;
+import org.codelibs.fess.Constants;
+import org.codelibs.fess.app.pager.MappingPager;
+import org.codelibs.fess.app.service.MappingService;
+import org.codelibs.fess.app.web.CrudMode;
+import org.codelibs.fess.app.web.admin.dict.AdminDictAction;
+import org.codelibs.fess.app.web.base.FessAdminAction;
+import org.codelibs.fess.dict.mapping.MappingItem;
+import org.codelibs.fess.util.RenderDataUtil;
+import org.dbflute.optional.OptionalEntity;
+import org.dbflute.optional.OptionalThing;
+import org.lastaflute.web.Execute;
+import org.lastaflute.web.response.ActionResponse;
+import org.lastaflute.web.response.HtmlResponse;
+import org.lastaflute.web.response.render.RenderData;
+import org.lastaflute.web.ruts.process.ActionRuntime;
+import org.lastaflute.web.validation.VaErrorHook;
+
+import javax.annotation.Resource;
+import java.io.File;
+import java.io.IOException;
+import java.io.InputStream;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @author shinsuke
+ * @author Keiichi Watanabe
+ */
+public class AdminDictMappingAction extends FessAdminAction {
+
+    // ===================================================================================
+    //                                                                           Attribute
+    //                                                                           =========
+    @Resource
+    private MappingService mappingService;
+    @Resource
+    private MappingPager mappingPager;
+
+    // ===================================================================================
+    //                                                                               Hook
+    //                                                                              ======
+    @Override
+    protected void setupHtmlData(final ActionRuntime runtime) {
+        super.setupHtmlData(runtime);
+        runtime.registerData("helpLink", systemHelper.getHelpLink(fessConfig.getOnlineHelpNameDictMapping()));
+    }
+
+    // ===================================================================================
+    //                                                                      Search Execute
+    //                                                                      ==============
+    @Execute
+    public HtmlResponse index(final SearchForm form) {
+        validate(form, messages -> {}, () -> asDictIndexHtml());
+        return asHtml(path_AdminDictMapping_AdminDictMappingJsp).renderWith(data -> {
+            searchPaging(data, form);
+        });
+    }
+
+    @Execute
+    public HtmlResponse list(final OptionalThing<Integer> pageNumber, final SearchForm form) {
+        validate(form, messages -> {}, () -> asDictIndexHtml());
+        pageNumber.ifPresent(num -> {
+            mappingPager.setCurrentPageNumber(pageNumber.get());
+        }).orElse(() -> {
+            mappingPager.setCurrentPageNumber(0);
+        });
+        return asHtml(path_AdminDictMapping_AdminDictMappingJsp).renderWith(data -> {
+            searchPaging(data, form);
+        });
+    }
+
+    @Execute
+    public HtmlResponse search(final SearchForm form) {
+        validate(form, messages -> {}, () -> asDictIndexHtml());
+        copyBeanToBean(form, mappingPager, op -> op.exclude(Constants.PAGER_CONVERSION_RULE));
+        return asHtml(path_AdminDictMapping_AdminDictMappingJsp).renderWith(data -> {
+            searchPaging(data, form);
+        });
+    }
+
+    @Execute
+    public HtmlResponse reset(final SearchForm form) {
+        validate(form, messages -> {}, () -> asDictIndexHtml());
+        mappingPager.clear();
+        return asHtml(path_AdminDictMapping_AdminDictMappingJsp).renderWith(data -> {
+            searchPaging(data, form);
+        });
+    }
+
+    protected void searchPaging(final RenderData data, final SearchForm form) {
+        // page navi
+        RenderDataUtil.register(data, "mappingItemItems", mappingService.getMappingList(form.dictId, mappingPager));
+
+        // restore from pager
+        BeanUtil.copyBeanToBean(mappingPager, form, op -> {
+            op.exclude(Constants.PAGER_CONVERSION_RULE);
+        });
+    }
+
+    // ===================================================================================
+    //                                                                        Edit Execute
+    //                                                                        ============
+    // -----------------------------------------------------
+    //                                            Entry Page
+    //                                            ----------
+    @Execute
+    public HtmlResponse createnew(final String dictId) {
+        saveToken();
+        return asHtml(path_AdminDictMapping_AdminDictMappingEditJsp).useForm(CreateForm.class, op -> {
+            op.setup(form -> {
+                form.initialize();
+                form.crudMode = CrudMode.CREATE;
+                form.dictId = dictId;
+            });
+        });
+    }
+
+    @Execute
+    public HtmlResponse edit(final EditForm form) {
+        validate(form, messages -> {}, () -> asListHtml(form.dictId));
+        mappingService
+                .getMappingItem(form.dictId, form.id)
+                .ifPresent(entity -> {
+                    form.inputs = entity.getInputsValue();
+                    form.output = entity.getOutput();
+                })
+                .orElse(() -> {
+                    throwValidationError(messages -> messages.addErrorsCrudCouldNotFindCrudTable(GLOBAL, form.getDisplayId()),
+                            () -> asListHtml(form.dictId));
+                });
+        saveToken();
+        if (form.crudMode.intValue() == CrudMode.EDIT) {
+            // back
+            form.crudMode = CrudMode.DETAILS;
+            return asDetailsHtml();
+        } else {
+            form.crudMode = CrudMode.EDIT;
+            return asEditHtml();
+        }
+    }
+
+    // -----------------------------------------------------
+    //                                               Details
+    //                                               -------
+    @Execute
+    public HtmlResponse details(final String dictId, final int crudMode, final long id) {
+        verifyCrudMode(crudMode, CrudMode.DETAILS, dictId);
+        saveToken();
+        return asDetailsHtml().useForm(
+                EditForm.class,
+                op -> {
+                    op.setup(form -> {
+                        mappingService
+                                .getMappingItem(dictId, id)
+                                .ifPresent(entity -> {
+                                    form.inputs = entity.getInputsValue();
+                                    form.output = entity.getOutput();
+                                })
+                                .orElse(() -> {
+                                    throwValidationError(
+                                            messages -> messages.addErrorsCrudCouldNotFindCrudTable(GLOBAL, dictId + ":" + id),
+                                            () -> asListHtml(dictId));
+                                });
+                        form.id = id;
+                        form.crudMode = crudMode;
+                        form.dictId = dictId;
+                    });
+                });
+    }
+
+    // -----------------------------------------------------
+    //                                              Download
+    //                                               -------
+    @Execute
+    public HtmlResponse downloadpage(final String dictId) {
+        saveToken();
+        return asHtml(path_AdminDictMapping_AdminDictMappingDownloadJsp).useForm(DownloadForm.class, op -> {
+            op.setup(form -> {
+                form.dictId = dictId;
+            });
+        }).renderWith(data -> {
+            mappingService.getMappingFile(dictId).ifPresent(file -> {
+                RenderDataUtil.register(data, "path", file.getPath());
+            }).orElse(() -> {
+                throwValidationError(messages -> messages.addErrorsFailedToDownloadMappingFile(GLOBAL), () -> asDictIndexHtml());
+            });
+        });
+    }
+
+    @Execute
+    public ActionResponse download(final DownloadForm form) {
+        validate(form, messages -> {}, () -> downloadpage(form.dictId));
+        verifyTokenKeep(() -> downloadpage(form.dictId));
+        return mappingService.getMappingFile(form.dictId).map(file -> {
+            return asStream(new File(file.getPath()).getName()).contentTypeOctetStream().stream(out -> {
+                try (InputStream inputStream = file.getInputStream()) {
+                    out.write(inputStream);
+                }
+            });
+        }).orElseGet(() -> {
+            throwValidationError(messages -> messages.addErrorsFailedToDownloadMappingFile(GLOBAL), () -> downloadpage(form.dictId));
+            return null;
+        });
+    }
+
+    // -----------------------------------------------------
+    //                                                Upload
+    //                                               -------
+    @Execute
+    public HtmlResponse uploadpage(final String dictId) {
+        saveToken();
+        return asHtml(path_AdminDictMapping_AdminDictMappingUploadJsp).useForm(UploadForm.class, op -> {
+            op.setup(form -> {
+                form.dictId = dictId;
+            });
+        }).renderWith(data -> {
+            mappingService.getMappingFile(dictId).ifPresent(file -> {
+                RenderDataUtil.register(data, "path", file.getPath());
+            }).orElse(() -> {
+                throwValidationError(messages -> messages.addErrorsFailedToDownloadMappingFile(GLOBAL), () -> asDictIndexHtml());
+            });
+        });
+    }
+
+    @Execute
+    public HtmlResponse upload(final UploadForm form) {
+        validate(form, messages -> {}, () -> uploadpage(form.dictId));
+        verifyToken(() -> uploadpage(form.dictId));
+        return mappingService.getMappingFile(form.dictId).map(file -> {
+            try (InputStream inputStream = form.mappingFile.getInputStream()) {
+                file.update(inputStream);
+            } catch (final IOException e) {
+                throwValidationError(messages -> messages.addErrorsFailedToUploadMappingFile(GLOBAL), () -> {
+                    return redirectWith(getClass(), moreUrl("uploadpage/" + form.dictId));
+                });
+            }
+            saveInfo(messages -> messages.addSuccessUploadMappingFile(GLOBAL));
+            return redirectWith(getClass(), moreUrl("uploadpage/" + form.dictId));
+        }).orElseGet(() -> {
+            throwValidationError(messages -> messages.addErrorsFailedToUploadMappingFile(GLOBAL), () -> uploadpage(form.dictId));
+            return null;
+        });
+
+    }
+
+    // -----------------------------------------------------
+    //                                         Actually Crud
+    //                                         -------------
+    @Execute
+    public HtmlResponse create(final CreateForm form) {
+        verifyCrudMode(form.crudMode, CrudMode.CREATE, form.dictId);
+        validate(form, messages -> {}, () -> asEditHtml());
+        verifyToken(() -> asEditHtml());
+        createMappingItem(form, () -> asEditHtml()).ifPresent(
+                entity -> {
+                    try {
+                        mappingService.store(form.dictId, entity);
+                        saveInfo(messages -> messages.addSuccessCrudCreateCrudTable(GLOBAL));
+                    } catch (final Exception e) {
+                        throwValidationError(messages -> messages.addErrorsCrudFailedToCreateCrudTable(GLOBAL, buildThrowableMessage(e)),
+                                () -> asEditHtml());
+                    }
+                }).orElse(() -> {
+            throwValidationError(messages -> messages.addErrorsCrudFailedToCreateInstance(GLOBAL), () -> asEditHtml());
+        });
+        return redirectWith(getClass(), moreUrl("list/1").params("dictId", form.dictId));
+    }
+
+    @Execute
+    public HtmlResponse update(final EditForm form) {
+        verifyCrudMode(form.crudMode, CrudMode.EDIT, form.dictId);
+        validate(form, messages -> {}, () -> asEditHtml());
+        verifyToken(() -> asEditHtml());
+        createMappingItem(form, () -> asEditHtml()).ifPresent(
+                entity -> {
+                    try {
+                        mappingService.store(form.dictId, entity);
+                        saveInfo(messages -> messages.addSuccessCrudUpdateCrudTable(GLOBAL));
+                    } catch (final Exception e) {
+                        throwValidationError(messages -> messages.addErrorsCrudFailedToUpdateCrudTable(GLOBAL, buildThrowableMessage(e)),
+                                () -> asEditHtml());
+                    }
+                }).orElse(() -> {
+            throwValidationError(messages -> messages.addErrorsCrudCouldNotFindCrudTable(GLOBAL, form.getDisplayId()), () -> asEditHtml());
+        });
+        return redirectWith(getClass(), moreUrl("list/1").params("dictId", form.dictId));
+    }
+
+    @Execute
+    public HtmlResponse delete(final EditForm form) {
+        verifyCrudMode(form.crudMode, CrudMode.DETAILS, form.dictId);
+        verifyToken(() -> asDetailsHtml());
+        validate(form, messages -> {}, () -> asDetailsHtml());
+        mappingService
+                .getMappingItem(form.dictId, form.id)
+                .ifPresent(
+                        entity -> {
+                            try {
+                                mappingService.delete(form.dictId, entity);
+                                saveInfo(messages -> messages.addSuccessCrudDeleteCrudTable(GLOBAL));
+                            } catch (final Exception e) {
+                                throwValidationError(
+                                        messages -> messages.addErrorsCrudFailedToDeleteCrudTable(GLOBAL, buildThrowableMessage(e)),
+                                        () -> asEditHtml());
+                            }
+                        })
+                .orElse(() -> {
+                    throwValidationError(messages -> messages.addErrorsCrudCouldNotFindCrudTable(GLOBAL, form.getDisplayId()),
+                            () -> asDetailsHtml());
+                });
+        return redirectWith(getClass(), moreUrl("list/1").params("dictId", form.dictId));
+    }
+
+    //===================================================================================
+    //                                                                        Assist Logic
+    //                                                                        ============
+
+    private OptionalEntity<MappingItem> getEntity(final CreateForm form) {
+        switch (form.crudMode) {
+        case CrudMode.CREATE:
+            final MappingItem entity = new MappingItem(0, StringUtil.EMPTY_STRINGS, StringUtil.EMPTY);
+            return OptionalEntity.of(entity);
+        case CrudMode.EDIT:
+            if (form instanceof EditForm) {
+                return mappingService.getMappingItem(form.dictId, ((EditForm) form).id);
+            }
+            break;
+        default:
+            break;
+        }
+        return OptionalEntity.empty();
+    }
+
+    protected OptionalEntity<MappingItem> createMappingItem(final CreateForm form, final VaErrorHook hook) {
+        return getEntity(form).map(entity -> {
+            final String[] newInputs = splitLine(form.inputs);
+            validateMappingString(newInputs, "inputs", hook);
+            entity.setNewInputs(newInputs);
+            final String newOutput = form.output;
+            validateMappingString(newOutput, "output", hook);
+            entity.setNewOutput(newOutput);
+            return entity;
+        });
+    }
+
+    // ===================================================================================
+    //                                                                        Small Helper
+    //                                                                        ============
+    protected void verifyCrudMode(final int crudMode, final int expectedMode, final String dictId) {
+        if (crudMode != expectedMode) {
+            throwValidationError(messages -> {
+                messages.addErrorsCrudInvalidMode(GLOBAL, String.valueOf(expectedMode), String.valueOf(crudMode));
+            }, () -> asListHtml(dictId));
+        }
+    }
+
+    private void validateMappingString(final String[] values, final String propertyName, final VaErrorHook hook) {
+        if (values.length == 0) {
+            return;
+        }
+        for (final String value : values) {
+            if (value.indexOf(',') >= 0) {
+                throwValidationError(messages -> {
+                    messages.addErrorsInvalidStrIsIncluded(propertyName, value, ",");
+                }, hook);
+            }
+            if (value.indexOf("=>") >= 0) {
+                throwValidationError(messages -> {
+                    messages.addErrorsInvalidStrIsIncluded(propertyName, value, "=>");
+                }, hook);
+            }
+        }
+    }
+
+    private void validateMappingString(final String value, final String propertyName, final VaErrorHook hook) {
+        if (value.indexOf(',') >= 0) {
+            throwValidationError(messages -> {
+                messages.addErrorsInvalidStrIsIncluded(propertyName, value, ",");
+            }, hook);
+        }
+        if (value.indexOf("=>") >= 0) {
+            throwValidationError(messages -> {
+                messages.addErrorsInvalidStrIsIncluded(propertyName, value, "=>");
+            }, hook);
+        }
+    }
+
+    private String[] splitLine(final String value) {
+        if (StringUtil.isBlank(value)) {
+            return StringUtil.EMPTY_STRINGS;
+        }
+        final String[] values = value.split("[\r\n]");
+        final List<String> list = new ArrayList<>(values.length);
+        for (final String line : values) {
+            if (StringUtil.isNotBlank(line)) {
+                list.add(line.trim());
+            }
+        }
+        return list.toArray(new String[list.size()]);
+    }
+
+    // ===================================================================================
+    //                                                                              JSP
+    //                                                                           =========
+
+    protected HtmlResponse asDictIndexHtml() {
+        return redirect(AdminDictAction.class);
+    }
+
+    private HtmlResponse asListHtml(final String dictId) {
+        return asHtml(path_AdminDictMapping_AdminDictMappingJsp).renderWith(data -> {
+            RenderDataUtil.register(data, "mappingItemItems", mappingService.getMappingList(dictId, mappingPager));
+        }).useForm(SearchForm.class, setup -> {
+            setup.setup(form -> {
+                copyBeanToBean(mappingPager, form, op -> op.include("id"));
+            });
+        });
+    }
+
+    private HtmlResponse asEditHtml() {
+        return asHtml(path_AdminDictMapping_AdminDictMappingEditJsp);
+    }
+
+    private HtmlResponse asDetailsHtml() {
+        return asHtml(path_AdminDictMapping_AdminDictMappingDetailsJsp);
+    }
+
+}

+ 46 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/CreateForm.java

@@ -0,0 +1,46 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.codelibs.fess.app.web.CrudMode;
+import org.lastaflute.web.validation.Required;
+import org.lastaflute.web.validation.theme.conversion.ValidateTypeFailure;
+
+import javax.validation.constraints.Size;
+
+/**
+ * @author shinsuke
+ * @author Keiichi Watanabe
+ */
+public class CreateForm {
+
+    @Required
+    public String dictId;
+
+    @ValidateTypeFailure
+    public Integer crudMode;
+
+    @Required
+    @Size(max = 1000)
+    public String inputs;
+
+    @Required
+    public String output;
+
+    public void initialize() {
+        crudMode = CrudMode.CREATE;
+    }
+}

+ 23 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/DownloadForm.java

@@ -0,0 +1,23 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.lastaflute.web.validation.Required;
+
+public class DownloadForm {
+    @Required
+    public String dictId;
+}

+ 34 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/EditForm.java

@@ -0,0 +1,34 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.lastaflute.web.validation.Required;
+import org.lastaflute.web.validation.theme.conversion.ValidateTypeFailure;
+
+/**
+ * @author shinsuke
+ * @author Keiichi Watanabe
+ */
+public class EditForm extends CreateForm {
+
+    @Required
+    @ValidateTypeFailure
+    public Long id;
+
+    public String getDisplayId() {
+        return dictId + ":" + id;
+    }
+}

+ 28 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/SearchForm.java

@@ -0,0 +1,28 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.lastaflute.web.validation.Required;
+
+/**
+ * @author shinsuke
+ * @author Keiichi Watanabe
+ */
+public class SearchForm {
+
+    @Required
+    public String dictId;
+}

+ 33 - 0
src/main/java/org/codelibs/fess/app/web/admin/dict/mapping/UploadForm.java

@@ -0,0 +1,33 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.app.web.admin.dict.mapping;
+
+import org.lastaflute.web.ruts.multipart.MultipartFormFile;
+import org.lastaflute.web.validation.Required;
+
+/**
+ * @author shinsuke
+ * @author Keiichi Watanabe
+ */
+public class UploadForm {
+
+    @Required
+    public String dictId;
+
+    @Required
+    public MultipartFormFile mappingFile;
+
+}

+ 38 - 0
src/main/java/org/codelibs/fess/dict/mapping/MappingCreator.java

@@ -0,0 +1,38 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.dict.mapping;
+
+import java.util.Date;
+
+import org.codelibs.fess.dict.DictionaryCreator;
+import org.codelibs.fess.dict.DictionaryFile;
+import org.codelibs.fess.dict.DictionaryItem;
+
+public class MappingCreator extends DictionaryCreator {
+
+    public MappingCreator() {
+        super("mapping.*\\.txt");
+    }
+
+    public MappingCreator(final String pattern) {
+        super(pattern);
+    }
+
+    @Override
+    protected DictionaryFile<? extends DictionaryItem> newDictionaryFile(final String id, final String path, final Date timestamp) {
+        return new MappingFile(id, path, timestamp).manager(dictionaryManager);
+    }
+}

+ 338 - 0
src/main/java/org/codelibs/fess/dict/mapping/MappingFile.java

@@ -0,0 +1,338 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.dict.mapping;
+
+import java.io.BufferedInputStream;
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.Closeable;
+import java.io.File;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.io.OutputStreamWriter;
+import java.io.Writer;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Date;
+import java.util.List;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+import org.apache.commons.io.IOUtils;
+import org.codelibs.core.lang.StringUtil;
+import org.codelibs.fess.Constants;
+import org.codelibs.fess.dict.DictionaryException;
+import org.codelibs.fess.dict.DictionaryFile;
+import org.dbflute.optional.OptionalEntity;
+
+public class MappingFile extends DictionaryFile<MappingItem> {
+    private static final String MAPPING = "mapping";
+
+    List<MappingItem> mappingItemList;
+
+    public MappingFile(final String id, final String path, final Date timestamp) {
+        super(id, path, timestamp);
+    }
+
+    @Override
+    public String getType() {
+        return MAPPING;
+    }
+
+    @Override
+    public String getPath() {
+        return path;
+    }
+
+    @Override
+    public OptionalEntity<MappingItem> get(long id) {
+        if (mappingItemList == null) {
+            reload(null, null);
+        }
+
+        for (final MappingItem mappingItem : mappingItemList) {
+            if (id == mappingItem.getId()) {
+                return OptionalEntity.of(mappingItem);
+            }
+        }
+        return OptionalEntity.empty();
+    }
+
+    @Override
+    public synchronized PagingList<MappingItem> selectList(final int offset, final int size) {
+        if (mappingItemList == null) {
+            reload(null, null);
+        }
+
+        if (offset >= mappingItemList.size() || offset < 0) {
+            return new PagingList<>(Collections.<MappingItem> emptyList(), offset, size, mappingItemList.size());
+        }
+
+        int toIndex = offset + size;
+        if (toIndex > mappingItemList.size()) {
+            toIndex = mappingItemList.size();
+        }
+
+        return new PagingList<>(mappingItemList.subList(offset, toIndex), offset, size, mappingItemList.size());
+    }
+
+    @Override
+    public synchronized void insert(final MappingItem item) {
+        try (MappingUpdater updater = new MappingUpdater(item)) {
+            reload(updater, null);
+        }
+    }
+
+    @Override
+    public synchronized void update(final MappingItem item) {
+        try (MappingUpdater updater = new MappingUpdater(item)) {
+            reload(updater, null);
+        }
+    }
+
+    @Override
+    public synchronized void delete(final MappingItem item) {
+        final MappingItem mappingItem = item;
+        mappingItem.setNewInputs(StringUtil.EMPTY_STRINGS);
+        mappingItem.setNewOutput(StringUtil.EMPTY);
+        try (MappingUpdater updater = new MappingUpdater(item)) {
+            reload(updater, null);
+        }
+    }
+
+    protected void reload(final MappingUpdater updater, final InputStream in) {
+        final Pattern parsePattern = Pattern.compile("(.*)\\s*=>\\s*(.*)\\s*$");
+        final List<MappingItem> itemList = new ArrayList<>();
+        try (BufferedReader reader =
+                new BufferedReader(new InputStreamReader(in != null ? in : dictionaryManager.getContentInputStream(this), Constants.UTF_8))) {
+            long id = 0;
+            String line = null;
+            while ((line = reader.readLine()) != null) {
+                // Remove comments
+                line = line.replaceAll("#.*$", StringUtil.EMPTY);
+
+                // Skip empty lines or comment lines
+                if (line.trim().length() == 0) {
+                    if (updater != null) {
+                        updater.write(line);
+                    }
+                    continue;
+                }
+
+                String[] inputs;
+                String output;
+
+                Matcher m = parsePattern.matcher(line.trim());
+
+                if (!m.find()) {
+                    throw new DictionaryException("Failed to parse " + path);
+                }
+
+                inputs = parseString(m.group(1).trim()).split(",");
+                output = parseString(m.group(2).trim());
+
+                if (inputs == null || output == null || inputs.length == 0) {
+                    throw new DictionaryException("Failed to parse " + path);
+                }
+
+                id++;
+                final MappingItem item = new MappingItem(id, inputs, output);
+
+                if (updater != null) {
+                    final MappingItem newItem = updater.write(item);
+                    if (newItem != null) {
+                        itemList.add(newItem);
+                    } else {
+                        id--;
+                    }
+                } else {
+                    itemList.add(item);
+                }
+            }
+            if (updater != null) {
+                final MappingItem item = updater.commit();
+                if (item != null) {
+                    itemList.add(item);
+                }
+            }
+            mappingItemList = itemList;
+        } catch (final IOException e) {
+            throw new DictionaryException("Failed to parse " + path, e);
+        }
+    }
+
+    public String getSimpleName() {
+        return new File(path).getName();
+    }
+
+    public InputStream getInputStream() throws IOException {
+        return new BufferedInputStream(dictionaryManager.getContentInputStream(this));
+    }
+
+    public synchronized void update(final InputStream in) throws IOException {
+        try (MappingUpdater updater = new MappingUpdater(null)) {
+            reload(updater, in);
+        }
+    }
+
+    @Override
+    public String toString() {
+        return "MappingFile [path=" + path + ", mappingItemList=" + mappingItemList + ", id=" + id + "]";
+    }
+
+    protected class MappingUpdater implements Closeable {
+
+        protected boolean isCommit = false;
+
+        protected File newFile;
+
+        protected Writer writer;
+
+        protected MappingItem item;
+
+        protected MappingUpdater(final MappingItem newItem) {
+            try {
+                newFile = File.createTempFile(MAPPING, ".txt");
+                writer = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(newFile), Constants.UTF_8));
+            } catch (final IOException e) {
+                if (newFile != null) {
+                    newFile.delete();
+                }
+                throw new DictionaryException("Failed to write a userDict file.", e);
+            }
+            item = newItem;
+        }
+
+        public MappingItem write(final MappingItem oldItem) {
+            try {
+                if (item != null && item.getId() == oldItem.getId() && item.isUpdated()) {
+                    if (item.equals(oldItem)) {
+                        try {
+                            if (!item.isDeleted()) {
+                                // update
+                                writer.write(item.toLineString());
+                                writer.write(Constants.LINE_SEPARATOR);
+                                return new MappingItem(item.getId(), item.getNewInputs(), item.getNewOutput());
+                            } else {
+                                return null;
+                            }
+                        } finally {
+                            item.setNewInputs(null);
+                            item.setNewOutput(null);
+                        }
+                    } else {
+                        throw new DictionaryException("Mapping file was updated: old=" + oldItem + " : new=" + item);
+                    }
+                } else {
+                    writer.write(oldItem.toLineString());
+                    writer.write(Constants.LINE_SEPARATOR);
+                    return oldItem;
+                }
+            } catch (final IOException e) {
+                throw new DictionaryException("Failed to write: " + oldItem + " -> " + item, e);
+            }
+        }
+
+        public void write(final String line) {
+            try {
+                writer.write(line);
+                writer.write(Constants.LINE_SEPARATOR);
+            } catch (final IOException e) {
+                throw new DictionaryException("Failed to write: " + line, e);
+            }
+        }
+
+        public MappingItem commit() {
+            isCommit = true;
+            if (item != null && item.isUpdated()) {
+                try {
+                    writer.write(item.toLineString());
+                    writer.write(Constants.LINE_SEPARATOR);
+                    return item;
+                } catch (final IOException e) {
+                    throw new DictionaryException("Failed to write: " + item, e);
+                }
+            }
+            return null;
+        }
+
+        @Override
+        public void close() {
+            try {
+                writer.flush();
+            } catch (final IOException e) {
+                // ignore
+            }
+            IOUtils.closeQuietly(writer);
+
+            if (isCommit) {
+                try {
+                    dictionaryManager.store(MappingFile.this, newFile);
+                } finally {
+                    newFile.delete();
+                }
+            } else {
+                newFile.delete();
+            }
+        }
+    }
+
+    char[] out = new char[256];
+
+    private String parseString(String s) {
+        int readPos = 0;
+        int len = s.length();
+        int writePos = 0;
+        while (readPos < len) {
+            char c = s.charAt(readPos++);
+            if (c == '\\') {
+                if (readPos >= len)
+                    throw new DictionaryException("Invalid escaped char in [" + s + "]");
+                c = s.charAt(readPos++);
+                switch (c) {
+                case '\\':
+                    c = '\\';
+                    break;
+                case 'n':
+                    c = '\n';
+                    break;
+                case 't':
+                    c = '\t';
+                    break;
+                case 'r':
+                    c = '\r';
+                    break;
+                case 'b':
+                    c = '\b';
+                    break;
+                case 'f':
+                    c = '\f';
+                    break;
+                case 'u':
+                    if (readPos + 3 >= len)
+                        throw new DictionaryException("Invalid escaped char in [" + s + "]");
+                    c = (char) Integer.parseInt(s.substring(readPos, readPos + 4), 16);
+                    readPos += 4;
+                    break;
+                }
+            }
+            out[writePos++] = c;
+        }
+        return new String(out, 0, writePos);
+    }
+}

+ 140 - 0
src/main/java/org/codelibs/fess/dict/mapping/MappingItem.java

@@ -0,0 +1,140 @@
+/*
+ * Copyright 2012-2016 CodeLibs Project and the Others.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
+ * either express or implied. See the License for the specific language
+ * governing permissions and limitations under the License.
+ */
+package org.codelibs.fess.dict.mapping;
+
+import java.util.Arrays;
+
+import org.apache.commons.lang3.StringUtils;
+import org.codelibs.core.lang.StringUtil;
+import org.codelibs.fess.dict.DictionaryItem;
+
+public class MappingItem extends DictionaryItem {
+    private final String[] inputs;
+
+    private final String output;
+
+    private String[] newInputs;
+
+    private String newOutput;
+
+    public MappingItem(final long id, final String[] inputs, final String output) {
+        this.id = id;
+        this.inputs = inputs;
+        this.output = output;
+        Arrays.sort(inputs);
+
+        if (id == 0) {
+            // create
+            newInputs = inputs;
+            newOutput = output;
+        }
+    }
+
+    public String[] getNewInputs() {
+        return newInputs;
+    }
+
+    public void setNewInputs(final String[] newInputs) {
+        this.newInputs = newInputs;
+    }
+
+    public String getNewOutput() {
+        return newOutput;
+    }
+
+    public void setNewOutput(final String newOutput) {
+        this.newOutput = newOutput;
+    }
+
+    public String[] getInputs() {
+        return inputs;
+    }
+
+    public String getInputsValue() {
+        if (inputs == null) {
+            return StringUtil.EMPTY;
+        }
+        return String.join("\n", inputs);
+    }
+
+    public String getOutput() {
+        return output;
+    }
+
+    public boolean isUpdated() {
+        return newInputs != null && newOutput != null;
+    }
+
+    public boolean isDeleted() {
+        return isUpdated() && newInputs.length == 0;
+    }
+
+    public void sort() {
+        if (inputs != null) {
+            Arrays.sort(inputs);
+        }
+        if (newInputs != null) {
+            Arrays.sort(newInputs);
+        }
+    }
+
+    @Override
+    public int hashCode() {
+        final int prime = 31;
+        int result = 1;
+        result = prime * result + Arrays.hashCode(inputs);
+        result = prime * result + (output == null ? 0 : output.hashCode());
+        return result;
+    }
+
+    @Override
+    public boolean equals(final Object obj) {
+        if (this == obj) {
+            return true;
+        }
+        if (obj == null) {
+            return false;
+        }
+        if (getClass() != obj.getClass()) {
+            return false;
+        }
+        final MappingItem other = (MappingItem) obj;
+        sort();
+        other.sort();
+        if (!Arrays.equals(inputs, other.inputs)) {
+            return false;
+        }
+        if (!output.equals(other.getOutput())) {
+            return false;
+        }
+        return true;
+    }
+
+    @Override
+    public String toString() {
+        return "MappingItem [id=" + id + ", inputs=" + Arrays.toString(inputs) + ", output=" + output + ", newInputs="
+                + Arrays.toString(newInputs) + ", newOutput=" + newOutput + "]";
+    }
+
+    public String toLineString() {
+        if (isUpdated()) {
+            return StringUtils.join(newInputs, ",") + "=>" + newOutput;
+        } else {
+            return StringUtils.join(inputs, ",") + "=>" + output;
+        }
+    }
+
+}

+ 15 - 0
src/main/java/org/codelibs/fess/mylasta/action/FessHtmlPath.java

@@ -92,6 +92,21 @@ public interface FessHtmlPath {
     /** The path of the HTML: /admin/dict/kuromoji/admin_dict_kuromoji_upload.jsp */
     HtmlNext path_AdminDictKuromoji_AdminDictKuromojiUploadJsp = new HtmlNext("/admin/dict/kuromoji/admin_dict_kuromoji_upload.jsp");
 
+    /** The path of the HTML: /admin/dict/mapping/admin_dict_mapping.jsp */
+    HtmlNext path_AdminDictMapping_AdminDictMappingJsp = new HtmlNext("/admin/dict/mapping/admin_dict_mapping.jsp");
+
+    /** The path of the HTML: /admin/dict/mapping/admin_dict_mapping_details.jsp */
+    HtmlNext path_AdminDictMapping_AdminDictMappingDetailsJsp = new HtmlNext("/admin/dict/mapping/admin_dict_mapping_details.jsp");
+
+    /** The path of the HTML: /admin/dict/mapping/admin_dict_mapping_download.jsp */
+    HtmlNext path_AdminDictMapping_AdminDictMappingDownloadJsp = new HtmlNext("/admin/dict/mapping/admin_dict_mapping_download.jsp");
+
+    /** The path of the HTML: /admin/dict/mapping/admin_dict_mapping_edit.jsp */
+    HtmlNext path_AdminDictMapping_AdminDictMappingEditJsp = new HtmlNext("/admin/dict/mapping/admin_dict_mapping_edit.jsp");
+
+    /** The path of the HTML: /admin/dict/mapping/admin_dict_mapping_upload.jsp */
+    HtmlNext path_AdminDictMapping_AdminDictMappingUploadJsp = new HtmlNext("/admin/dict/mapping/admin_dict_mapping_upload.jsp");
+
     /** The path of the HTML: /admin/dict/seunjeon/admin_dict_seunjeon.jsp */
     HtmlNext path_AdminDictSeunjeon_AdminDictSeunjeonJsp = new HtmlNext("/admin/dict/seunjeon/admin_dict_seunjeon.jsp");
 

+ 45 - 0
src/main/java/org/codelibs/fess/mylasta/action/FessLabels.java

@@ -365,6 +365,9 @@ public class FessLabels extends ActionMessages {
     /** The key of the message: Bad Word File */
     public static final String LABELS_BAD_WORD_FILE = "{labels.badWordFile}";
 
+    /** The key of the message: Mapping File */
+    public static final String LABELS_MAPPING_FILE = "{labels.mappingFile}";
+
     /** The key of the message: Boost Expr */
     public static final String LABELS_BOOST_EXPR = "{labels.boostExpr}";
 
@@ -1764,6 +1767,48 @@ public class FessLabels extends ActionMessages {
     /** The key of the message: Synonym File */
     public static final String LABELS_dict_synonym_file = "{labels.dict_synonym_file}";
 
+    /** The key of the message: Mapping List */
+    public static final String LABELS_dict_mapping_configuration = "{labels.dict_mapping_configuration}";
+
+    /** The key of the message: Mapping List */
+    public static final String LABELS_dict_mapping_title = "{labels.dict_mapping_title}";
+
+    /** The key of the message: List */
+    public static final String LABELS_dict_mapping_list_link = "{labels.dict_mapping_list_link}";
+
+    /** The key of the message: Create New */
+    public static final String LABELS_dict_mapping_link_create = "{labels.dict_mapping_link_create}";
+
+    /** The key of the message: Edit */
+    public static final String LABELS_dict_mapping_link_edit = "{labels.dict_mapping_link_edit}";
+
+    /** The key of the message: Delete */
+    public static final String LABELS_dict_mapping_link_delete = "{labels.dict_mapping_link_delete}";
+
+    /** The key of the message: Details */
+    public static final String LABELS_dict_mapping_link_details = "{labels.dict_mapping_link_details}";
+
+    /** The key of the message: Download */
+    public static final String LABELS_dict_mapping_link_download = "{labels.dict_mapping_link_download}";
+
+    /** The key of the message: Upload */
+    public static final String LABELS_dict_mapping_link_upload = "{labels.dict_mapping_link_upload}";
+
+    /** The key of the message: Source */
+    public static final String LABELS_dict_mapping_source = "{labels.dict_mapping_source}";
+
+    /** The key of the message: Target */
+    public static final String LABELS_dict_mapping_target = "{labels.dict_mapping_target}";
+
+    /** The key of the message: Download */
+    public static final String LABELS_dict_mapping_button_download = "{labels.dict_mapping_button_download}";
+
+    /** The key of the message: Upload */
+    public static final String LABELS_dict_mapping_button_upload = "{labels.dict_mapping_button_upload}";
+
+    /** The key of the message: Mapping File */
+    public static final String LABELS_dict_mapping_file = "{labels.dict_mapping_file}";
+
     /** The key of the message: Seunjeon List */
     public static final String LABELS_dict_seunjeon_configuration = "{labels.dict_seunjeon_configuration}";
 

+ 52 - 0
src/main/java/org/codelibs/fess/mylasta/action/FessMessages.java

@@ -15,6 +15,7 @@
  */
 package org.codelibs.fess.mylasta.action;
 
+import org.codelibs.fess.mylasta.action.FessLabels;
 import org.lastaflute.web.ruts.message.ActionMessage;
 
 /**
@@ -254,6 +255,12 @@ public class FessMessages extends FessLabels {
     /** The key of the message: Failed to upload the Badword file. */
     public static final String ERRORS_failed_to_upload_badword_file = "{errors.failed_to_upload_badword_file}";
 
+    /** The key of the message: Failed to download the Mapping file. */
+    public static final String ERRORS_failed_to_download_mapping_file = "{errors.failed_to_download_mapping_file}";
+
+    /** The key of the message: Failed to upload the Mapping file. */
+    public static final String ERRORS_failed_to_upload_mapping_file = "{errors.failed_to_upload_mapping_file}";
+
     /** The key of the message: "{1}" in "{0}" is invalid. */
     public static final String ERRORS_invalid_str_is_included = "{errors.invalid_str_is_included}";
 
@@ -371,6 +378,9 @@ public class FessMessages extends FessLabels {
     /** The key of the message: Uploaded Bad Word file. */
     public static final String SUCCESS_upload_bad_word = "{success.upload_bad_word}";
 
+    /** The key of the message: Uploaded Mapping file. */
+    public static final String SUCCESS_upload_mapping_file = "{success.upload_mapping_file}";
+
     /** The key of the message: Sent the test mail. */
     public static final String SUCCESS_send_testmail = "{success.send_testmail}";
 
@@ -1525,6 +1535,34 @@ public class FessMessages extends FessLabels {
         return this;
     }
 
+    /**
+     * Add the created action message for the key 'errors.failed_to_download_mapping_file' with parameters.
+     * <pre>
+     * message: Failed to download the Mapping file.
+     * </pre>
+     * @param property The property name for the message. (NotNull)
+     * @return this. (NotNull)
+     */
+    public FessMessages addErrorsFailedToDownloadMappingFile(String property) {
+        assertPropertyNotNull(property);
+        add(property, new ActionMessage(ERRORS_failed_to_download_mapping_file));
+        return this;
+    }
+
+    /**
+     * Add the created action message for the key 'errors.failed_to_upload_mapping_file' with parameters.
+     * <pre>
+     * message: Failed to upload the Mapping file.
+     * </pre>
+     * @param property The property name for the message. (NotNull)
+     * @return this. (NotNull)
+     */
+    public FessMessages addErrorsFailedToUploadMappingFile(String property) {
+        assertPropertyNotNull(property);
+        add(property, new ActionMessage(ERRORS_failed_to_upload_mapping_file));
+        return this;
+    }
+
     /**
      * Add the created action message for the key 'errors.invalid_str_is_included' with parameters.
      * <pre>
@@ -2092,6 +2130,20 @@ public class FessMessages extends FessLabels {
         return this;
     }
 
+    /**
+     * Add the created action message for the key 'success.upload_mapping_file' with parameters.
+     * <pre>
+     * message: Uploaded Mapping file.
+     * </pre>
+     * @param property The property name for the message. (NotNull)
+     * @return this. (NotNull)
+     */
+    public FessMessages addSuccessUploadMappingFile(String property) {
+        assertPropertyNotNull(property);
+        add(property, new ActionMessage(SUCCESS_upload_mapping_file));
+        return this;
+    }
+
     /**
      * Add the created action message for the key 'success.send_testmail' with parameters.
      * <pre>

+ 15 - 0
src/main/java/org/codelibs/fess/mylasta/direction/FessConfig.java

@@ -15,6 +15,7 @@
  */
 package org.codelibs.fess.mylasta.direction;
 
+import org.codelibs.fess.mylasta.direction.FessEnv;
 import org.lastaflute.core.direction.exception.ConfigPropertyNotFoundException;
 
 /**
@@ -616,6 +617,9 @@ public interface FessConfig extends FessEnv, org.codelibs.fess.mylasta.direction
     /** The key of the configuration. e.g. seunjeon */
     String ONLINE_HELP_NAME_DICT_SEUNJEON = "online.help.name.dict.seunjeon";
 
+    /** The key of the configuration. e.g. mapping */
+    String ONLINE_HELP_NAME_DICT_MAPPING = "online.help.name.dict.mapping";
+
     /** The key of the configuration. e.g. webconfig */
     String ONLINE_HELP_NAME_WEBCONFIG = "online.help.name.webconfig";
 
@@ -2841,6 +2845,13 @@ public interface FessConfig extends FessEnv, org.codelibs.fess.mylasta.direction
      */
     String getOnlineHelpNameDictSeunjeon();
 
+    /**
+     * Get the value for the key 'online.help.name.dict.mapping'. <br>
+     * The value is, e.g. mapping <br>
+     * @return The value of found property. (NotNull: if not found, exception but basically no way)
+     */
+    String getOnlineHelpNameDictMapping();
+
     /**
      * Get the value for the key 'online.help.name.webconfig'. <br>
      * The value is, e.g. webconfig <br>
@@ -4677,6 +4688,10 @@ public interface FessConfig extends FessEnv, org.codelibs.fess.mylasta.direction
             return get(FessConfig.ONLINE_HELP_NAME_DICT_SEUNJEON);
         }
 
+        public String getOnlineHelpNameDictMapping() {
+            return get(FessConfig.ONLINE_HELP_NAME_DICT_MAPPING);
+        }
+
         public String getOnlineHelpNameWebconfig() {
             return get(FessConfig.ONLINE_HELP_NAME_WEBCONFIG);
         }

+ 1 - 0
src/main/resources/fess_config.properties

@@ -347,6 +347,7 @@ online.help.name.dict.synonym=synonym
 online.help.name.dict=dict
 online.help.name.dict.kuromoji=kuromoji
 online.help.name.dict.seunjeon=seunjeon
+online.help.name.dict.mapping=mapping
 online.help.name.webconfig=webconfig
 online.help.name.searchlist=searchlist
 online.help.name.log=log

+ 7 - 1
src/main/resources/fess_dict.xml

@@ -14,6 +14,9 @@
 		<postConstruct name="addCreator">
 			<arg>synonymCreator</arg>
 		</postConstruct>
+		<postConstruct name="addCreator">
+			<arg>mappingCreator</arg>
+		</postConstruct>
 	</component>
 
 	<component name="kuromojiDictCreator"
@@ -23,6 +26,9 @@
 		class="org.codelibs.fess.dict.seunjeon.SeunjeonCreator">
 	</component>
 	<component name="synonymCreator"
-		class="org.codelibs.fess.dict.synonym.SynonymCreator">
+	   class="org.codelibs.fess.dict.synonym.SynonymCreator">
+	</component>
+	<component name="mappingCreator"
+	   class="org.codelibs.fess.dict.mapping.MappingCreator">
 	</component>
 </components>

+ 15 - 0
src/main/resources/fess_label.properties

@@ -111,6 +111,7 @@ labels.token=Token
 labels.synonymFile=Synonym File
 labels.elevateWordFile=Additional Word File
 labels.badWordFile=Bad Word File
+labels.mappingFile=Mapping File
 labels.boostExpr=Boost Expr
 labels.confirmPassword=Confirm
 labels.crawler=Crawler
@@ -578,6 +579,20 @@ labels.dict_synonym_target=Target
 labels.dict_synonym_button_download=Download
 labels.dict_synonym_button_upload=Upload
 labels.dict_synonym_file=Synonym File
+labels.dict_mapping_configuration=Mapping List
+labels.dict_mapping_title=Mapping List
+labels.dict_mapping_list_link=List
+labels.dict_mapping_link_create=Create New
+labels.dict_mapping_link_edit=Edit
+labels.dict_mapping_link_delete=Delete
+labels.dict_mapping_link_details=Details
+labels.dict_mapping_link_download=Download
+labels.dict_mapping_link_upload=Upload
+labels.dict_mapping_source=Source
+labels.dict_mapping_target=Target
+labels.dict_mapping_button_download=Download
+labels.dict_mapping_button_upload=Upload
+labels.dict_mapping_file=Mapping File
 labels.dict_seunjeon_configuration = Seunjeon List
 labels.dict_seunjeon_title = Seunjeon List
 labels.dict_seunjeon_list_link = List

+ 15 - 0
src/main/resources/fess_label_en.properties

@@ -111,6 +111,7 @@ labels.token=Token
 labels.synonymFile=Synonym File
 labels.elevateWordFile=Additional Word File
 labels.badWordFile=Bad Word File
+labels.mappingFile=Mapping File
 labels.boostExpr=Boost Expr
 labels.confirmPassword=Confirm
 labels.crawler=Crawler
@@ -578,6 +579,20 @@ labels.dict_synonym_target=Target
 labels.dict_synonym_button_download=Download
 labels.dict_synonym_button_upload=Upload
 labels.dict_synonym_file=Synonym File
+labels.dict_mapping_configuration=Mapping List
+labels.dict_mapping_title=Mapping List
+labels.dict_mapping_list_link=List
+labels.dict_mapping_link_create=Create New
+labels.dict_mapping_link_edit=Edit
+labels.dict_mapping_link_delete=Delete
+labels.dict_mapping_link_details=Details
+labels.dict_mapping_link_download=Download
+labels.dict_mapping_link_upload=Upload
+labels.dict_mapping_source=Source
+labels.dict_mapping_target=Target
+labels.dict_mapping_button_download=Download
+labels.dict_mapping_button_upload=Upload
+labels.dict_mapping_file=Mapping File
 labels.dict_seunjeon_configuration = Seunjeon List
 labels.dict_seunjeon_title = Seunjeon List
 labels.dict_seunjeon_list_link = List

+ 17 - 0
src/main/resources/fess_label_ja.properties

@@ -111,6 +111,7 @@ labels.token=\u30c8\u30fc\u30af\u30f3
 labels.synonymFile=\u540c\u7fa9\u8a9e\u30d5\u30a1\u30a4\u30eb
 labels.elevateWordFile=\u8ffd\u52a0\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb
 labels.badWordFile=\u9664\u5916\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb
+labels.mappingFile=\u30de\u30c3\u30d4\u30f3\u30b0\u30d5\u30a1\u30a4\u30eb
 labels.boostExpr=\u30d6\u30fc\u30b9\u30c8\u5024\u5f0f
 labels.confirmPassword=\u78ba\u8a8d
 labels.crawler=\u30af\u30ed\u30fc\u30e9
@@ -574,6 +575,22 @@ labels.dict_synonym_target=\u5909\u63db\u5f8c
 labels.dict_synonym_button_download=\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9
 labels.dict_synonym_button_upload=\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9
 labels.dict_synonym_file=\u540c\u7fa9\u8a9e\u30d5\u30a1\u30a4\u30eb
+
+labels.dict_mapping_configuration=\u30de\u30c3\u30d4\u30f3\u30b0\u8f9e\u66f8
+labels.dict_mapping_title=\u30de\u30c3\u30d4\u30f3\u30b0\u8f9e\u66f8
+labels.dict_mapping_list_link=\u4e00\u89a7
+labels.dict_mapping_link_create=\u65b0\u898f\u4f5c\u6210
+labels.dict_mapping_link_edit=\u7de8\u96c6
+labels.dict_mapping_link_delete=\u524a\u9664
+labels.dict_mapping_link_details=\u8a73\u7d30
+labels.dict_mapping_link_download=\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9
+labels.dict_mapping_link_upload=\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9
+labels.dict_mapping_source=\u5909\u63db\u5143
+labels.dict_mapping_target=\u5909\u63db\u5f8c
+labels.dict_mapping_button_download=\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9
+labels.dict_mapping_button_upload=\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9
+labels.dict_mapping_file=\u30de\u30c3\u30d4\u30f3\u30b0\u30d5\u30a1\u30a4\u30eb
+
 labels.dict_seunjeon_configuration = Seunjeon\u5358\u8a9e\u4e00\u89a7
 labels.dict_seunjeon_title = Seunjeon\u5358\u8a9e\u4e00\u89a7
 labels.dict_seunjeon_list_link = \u4e00\u89a7

+ 15 - 0
src/main/resources/fess_label_ko.properties

@@ -109,6 +109,7 @@ labels.token = \ud1a0\ud070
 labels.synonymFile = \ub3d9\uc758\uc5b4 \ud30c\uc77c
 labels.elevateWordFile = \ucd94\uac00 \uc6cc\ub4dc \ud30c\uc77c
 labels.badWordFile = \uc81c\uc678 \uc6cc\ub4dc \ud30c\uc77c
+labels.mappingFile = Mapping \ud30c\uc77c
 labels.boostExpr = \ubd80\uc2a4\ud2b8 \uac12 \uc2dd
 labels.confirmPassword = \ud655\uc778
 labels.crawler = \ud06c\ub864\ub7ec
@@ -569,6 +570,20 @@ labels.dict_synonym_target = \ubcc0\ud658 \ud6c4
 labels.dict_synonym_button_download = \ub2e4\uc6b4\ub85c\ub4dc
 labels.dict_synonym_button_upload = \uc5c5\ub85c\ub4dc
 labels.dict_synonym_file = \ub3d9\uc758\uc5b4 \ud30c\uc77c
+labels.dict_mapping_configuration=Mapping \ubaa9\ub85d
+labels.dict_mapping_title=Mapping \ubaa9\ub85d
+labels.dict_mapping_list_link=\ubaa9\ub85d
+labels.dict_mapping_link_create=\uc0c8\ub85c \ub9cc\ub4e4\uae30
+labels.dict_mapping_link_edit=\ud3b8\uc9d1
+labels.dict_mapping_link_delete=\uc0ad\uc81c
+labels.dict_mapping_link_details=\uc0c1\uc138
+labels.dict_mapping_link_download=\ub2e4\uc6b4\ub85c\ub4dc
+labels.dict_mapping_link_upload=\uc5c5\ub85c\ub4dc
+labels.dict_mapping_source=\uc6d0\ubcf8
+labels.dict_mapping_target=\ubcc0\ud658 \ud6c4
+labels.dict_mapping_button_download=\ub2e4\uc6b4\ub85c\ub4dc
+labels.dict_mapping_button_upload=\uc5c5\ub85c\ub4dc
+labels.dict_mapping_file=Mapping \ud30c\uc77c
 labels.dict_seunjeon_configuration = Seunjeon \ubaa9\ub85d
 labels.dict_seunjeon_title = Seunjeon \ubaa9\ub85d
 labels.dict_seunjeon_list_link = \ubaa9\ub85d

+ 3 - 0
src/main/resources/fess_message.properties

@@ -106,6 +106,8 @@ errors.failed_to_download_elevate_file=Failed to download the Elevate file.
 errors.failed_to_upload_elevate_file=Failed to upload the Elevate file.
 errors.failed_to_download_badword_file=Failed to download the Badword file.
 errors.failed_to_upload_badword_file=Failed to upload the Badword file.
+errors.failed_to_download_mapping_file=Failed to download the Mapping file.
+errors.failed_to_upload_mapping_file=Failed to upload the Mapping file.
 errors.invalid_str_is_included="{1}" in "{0}" is invalid.
 errors.blank_password=Password is required.
 errors.invalid_confirm_password=Confirm Password does not match.
@@ -148,6 +150,7 @@ success.upload_synonym_file=Uploaded Synonym file.
 success.upload_kuromoji_file=Uploaded Kuromoji file.
 success.upload_elevate_word=Uploaded Additional Word file.
 success.upload_bad_word=Uploaded Bad Word file.
+success.upload_mapping_file=Uploaded Mapping file.
 success.send_testmail=Sent the test mail.
 success.job_log_delete_all=Deleted job logs.
 success.changed_password=Changed your password.

+ 3 - 0
src/main/resources/fess_message_en.properties

@@ -106,6 +106,8 @@ errors.failed_to_download_elevate_file=Failed to download the Elevate file.
 errors.failed_to_upload_elevate_file=Failed to upload the Elevate file.
 errors.failed_to_download_badword_file=Failed to download the Badword file.
 errors.failed_to_upload_badword_file=Failed to upload the Badword file.
+errors.failed_to_download_mapping_file=Failed to download the Mapping file.
+errors.failed_to_upload_mapping_file=Failed to upload the Mapping file.
 errors.invalid_str_is_included="{1}" in "{0}" is invalid.
 errors.blank_password=Password is required.
 errors.invalid_confirm_password=Confirm Password does not match.
@@ -148,6 +150,7 @@ success.upload_synonym_file=Uploaded Synonym file.
 success.upload_kuromoji_file=Uploaded Kuromoji file.
 success.upload_elevate_word=Uploaded Additional Word file.
 success.upload_bad_word=Uploaded Bad Word file.
+success.upload_mapping_file=Uploaded Mapping file.
 success.send_testmail=Sent the test mail.
 success.job_log_delete_all=Deleted job logs.
 success.changed_password=Changed your password.

+ 3 - 0
src/main/resources/fess_message_ja.properties

@@ -102,6 +102,8 @@ errors.failed_to_download_elevate_file = \u8ffd\u52a0\u30ef\u30fc\u30c9\u30d5\u3
 errors.failed_to_upload_elevate_file = \u8ffd\u52a0\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002
 errors.failed_to_download_badword_file = \u9664\u5916\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb\u306e\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002
 errors.failed_to_upload_badword_file = \u9664\u5916\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3067\u304d\u307e\u305b\u3093\u3067\u3057\u305f\u3002
+errors.failed_to_download_mapping_file = \u30de\u30c3\u30d4\u30f3\u30b0\u30d5\u30a1\u30a4\u30eb\u306e\u30c0\u30a6\u30f3\u30ed\u30fc\u30c9\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002
+errors.failed_to_upload_mapping_file = \u30de\u30c3\u30d4\u30f3\u30b0\u30d5\u30a1\u30a4\u30eb\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002
 errors.invalid_str_is_included = {0}\u3067\u306f{1}\u306f\u7121\u52b9\u3067\u3059\u3002
 errors.blank_password = \u30d1\u30b9\u30ef\u30fc\u30c9\u304c\u5fc5\u8981\u306b\u306a\u308a\u307e\u3059\u3002
 errors.invalid_confirm_password = \u30d1\u30b9\u30ef\u30fc\u30c9\u306e\u78ba\u8a8d\u3068\u4e00\u81f4\u3057\u307e\u305b\u3093\u3002
@@ -142,6 +144,7 @@ success.upload_synonym_file = \u540c\u7fa9\u8a9e\u30d5\u30a1\u30a4\u30eb\u3092\u
 success.upload_kuromoji_file = Kuromoji\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3057\u307e\u3057\u305f\u3002
 success.upload_elevate_word = \u8ffd\u52a0\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3057\u307e\u3057\u305f\u3002
 success.upload_bad_word = \u9664\u5916\u30ef\u30fc\u30c9\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3057\u307e\u3057\u305f\u3002
+success.upload_mapping_file = \u30de\u30c3\u30d4\u30f3\u30b0\u30d5\u30a1\u30a4\u30eb\u3092\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u3057\u307e\u3057\u305f\u3002
 success.send_testmail=\u30c6\u30b9\u30c8\u30e1\u30fc\u30eb\u3092\u9001\u4fe1\u3057\u307e\u3057\u305f\u3002
 success.job_log_delete_all=\u30b8\u30e7\u30d6\u30ed\u30b0\u3092\u524a\u9664\u3057\u307e\u3057\u305f\u3002
 success.changed_password=\u30d1\u30b9\u30ef\u30fc\u30c9\u3092\u5909\u66f4\u3057\u307e\u3057\u305f\u3002

+ 156 - 0
src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping.jsp

@@ -0,0 +1,156 @@
+<%@page pageEncoding="UTF-8" contentType="text/html; charset=UTF-8"%><!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title><la:message key="labels.admin_brand_title" /> | <la:message
+		key="labels.dict_mapping_configuration" /></title>
+<jsp:include page="/WEB-INF/view/common/admin/head.jsp"></jsp:include>
+</head>
+<body class="skin-blue sidebar-mini">
+	<div class="wrapper">
+		<jsp:include page="/WEB-INF/view/common/admin/header.jsp"></jsp:include>
+		<jsp:include page="/WEB-INF/view/common/admin/sidebar.jsp">
+			<jsp:param name="menuCategoryType" value="system" />
+			<jsp:param name="menuType" value="dict" />
+		</jsp:include>
+		<div class="content-wrapper">
+			<section class="content-header">
+				<h1>
+					<la:message key="labels.dict_mapping_title" />
+				</h1>
+				<ol class="breadcrumb">
+					<li><la:link href="list">
+							<la:message key="labels.dict_list_link" />
+						</la:link></li>
+					<li><la:message key="labels.dict_mapping_list_link" /></li>
+				</ol>
+			</section>
+			<section class="content">
+				<div class="row">
+					<div class="col-md-12">
+						<div class="box box-primary">
+							<div class="box-header with-border">
+								<h3 class="box-title">
+									<la:message key="labels.dict_mapping_list_link" />
+								</h3>
+								<div class="btn-group pull-right">
+									<la:link href="/admin/dict" styleClass="btn btn-default btn-xs">
+										<i class="fa fa-book"></i>
+										<la:message key="labels.dict_list_link" />
+									</la:link>
+									<la:link href="list/1?dictId=${f:u(dictId)}"
+										styleClass="btn btn-primary btn-xs">
+										<i class="fa fa-th-list"></i>
+										<la:message key="labels.dict_mapping_list_link" />
+									</la:link>
+									<la:link href="createnew/${f:u(dictId)}"
+										styleClass="btn btn-success btn-xs">
+										<i class="fa fa-plus"></i>
+										<la:message key="labels.dict_mapping_link_create" />
+									</la:link>
+									<la:link href="downloadpage/${f:u(dictId)}"
+										styleClass="btn btn-primary btn-xs">
+										<i class="fa fa-download"></i>
+										<la:message key="labels.dict_mapping_link_download" />
+									</la:link>
+									<la:link href="uploadpage/${f:u(dictId)}"
+										styleClass="btn btn-success btn-xs">
+										<i class="fa fa-upload"></i>
+										<la:message key="labels.dict_mapping_link_upload" />
+									</la:link>
+								</div>
+							</div>
+							<!-- /.box-header -->
+							<div class="box-body">
+								<%-- Message --%>
+								<div>
+									<la:info id="msg" message="true">
+										<div class="alert alert-info">${msg}</div>
+									</la:info>
+									<la:errors />
+								</div>
+								<%-- List --%>
+								<c:if test="${mappingPager.allRecordCount == 0}">
+									<div class="row top10">
+										<div class="col-sm-12">
+											<i class="fa fa-info-circle text-light-blue"></i>
+											<la:message key="labels.list_could_not_find_crud_table" />
+										</div>
+									</div>
+								</c:if>
+								<c:if test="${mappingPager.allRecordCount > 0}">
+									<div class="row">
+										<div class="col-sm-12">
+											<table class="table table-bordered table-striped">
+												<thead>
+													<tr>
+														<th><la:message key="labels.dict_mapping_source" /></th>
+														<th><la:message key="labels.dict_mapping_target" /></th>
+													</tr>
+												</thead>
+												<tbody>
+													<c:forEach var="data" varStatus="s"
+														items="${mappingItemItems}">
+														<tr
+															data-href="${contextPath}/admin/dict/mapping/details/${f:u(dictId)}/4/${f:u(data.id)}">
+															<td>${f:h(data.inputs)}</td>
+															<td>${f:h(data.outputs)}</td>
+														</tr>
+													</c:forEach>
+												</tbody>
+											</table>
+										</div>
+									</div>
+									<c:set var="pager" value="${mappingPager}" scope="request" />
+									<div class="row">
+										<div class="col-sm-2">
+											<la:message key="labels.pagination_page_guide_msg"
+												arg0="${f:h(pager.currentPageNumber)}"
+												arg1="${f:h(pager.allPageCount)}"
+												arg2="${f:h(pager.allRecordCount)}" />
+										</div>
+										<div class="col-sm-10">
+											<ul class="pagination pagination-sm no-margin pull-right">
+												<c:if test="${pager.existPrePage}">
+													<li class="prev"><la:link
+															href="list/${pager.currentPageNumber - 1}?dictId=${f:u(dictId)}">
+															<la:message key="labels.prev_page" />
+														</la:link></li>
+												</c:if>
+												<c:if test="${!pager.existPrePage}">
+													<li class="prev disabled"><a href="#"><la:message
+																key="labels.prev_page" /></a></li>
+												</c:if>
+												<c:forEach var="p" varStatus="s"
+													items="${pager.pageNumberList}">
+													<li
+														<c:if test="${p == pager.currentPageNumber}">class="active"</c:if>><la:link
+															href="list/${p}?dictId=${f:u(dictId)}">${p}</la:link></li>
+												</c:forEach>
+												<c:if test="${pager.existNextPage}">
+													<li class="next"><la:link
+															href="list/${pager.currentPageNumber + 1}?dictId=${f:u(dictId)}">
+															<la:message key="labels.next_page" />
+														</la:link></li>
+												</c:if>
+												<c:if test="${!pager.existNextPage}">
+													<li class="next disabled"><a href="#"><la:message
+																key="labels.next_page" /></a></li>
+												</c:if>
+											</ul>
+										</div>
+									</div>
+								</c:if>
+							</div>
+							<!-- /.box-body -->
+						</div>
+						<!-- /.box -->
+					</div>
+				</div>
+			</section>
+		</div>
+		<jsp:include page="/WEB-INF/view/common/admin/footer.jsp"></jsp:include>
+	</div>
+	<jsp:include page="/WEB-INF/view/common/admin/foot.jsp"></jsp:include>
+</body>
+</html>

+ 127 - 0
src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_details.jsp

@@ -0,0 +1,127 @@
+<%@page pageEncoding="UTF-8" contentType="text/html; charset=UTF-8"%><!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title><la:message key="labels.admin_brand_title" /> | <la:message
+		key="labels.dict_mapping_configuration" /></title>
+<jsp:include page="/WEB-INF/view/common/admin/head.jsp"></jsp:include>
+</head>
+<body class="skin-blue sidebar-mini">
+	<div class="wrapper">
+		<jsp:include page="/WEB-INF/view/common/admin/header.jsp"></jsp:include>
+		<jsp:include page="/WEB-INF/view/common/admin/sidebar.jsp">
+			<jsp:param name="menuCategoryType" value="system" />
+			<jsp:param name="menuType" value="dict" />
+		</jsp:include>
+		<div class="content-wrapper">
+			<section class="content-header">
+				<h1>
+					<la:message key="labels.dict_mapping_title" />
+				</h1>
+				<ol class="breadcrumb">
+					<li><la:link href="list">
+							<la:message key="labels.dict_list_link" />
+						</la:link></li>
+					<li><la:link href="list/0/?dictId=${f:u(dictId)}">
+							<la:message key="labels.dict_mapping_list_link" />
+						</la:link></li>
+					<li class="active"><la:message
+							key="labels.dict_mapping_link_details" /></li>
+				</ol>
+			</section>
+			<section class="content">
+				<la:form action="/admin/dict/mapping/">
+					<la:hidden property="crudMode" />
+					<la:hidden property="dictId" />
+					<c:if test="${crudMode==2 || crudMode==3 || crudMode==4}">
+						<la:hidden property="id" />
+					</c:if>
+					<div class="row">
+						<div class="col-md-12">
+							<div
+								class="box <c:if test="${crudMode == 1}">box-success</c:if><c:if test="${crudMode == 2}">box-warning</c:if><c:if test="${crudMode == 3}">box-danger</c:if><c:if test="${crudMode == 4}">box-primary</c:if>">
+								<%-- Box Header --%>
+								<div class="box-header with-border">
+									<h3 class="box-title">
+										<c:if test="${crudMode == 1}">
+											<la:message key="labels.dict_mapping_link_create" />
+										</c:if>
+										<c:if test="${crudMode == 2}">
+											<la:message key="labels.dict_mapping_link_edit" />
+										</c:if>
+										<c:if test="${crudMode == 3}">
+											<la:message key="labels.dict_mapping_link_delete" />
+										</c:if>
+										<c:if test="${crudMode == 4}">
+											<la:message key="labels.dict_mapping_link_details" />
+										</c:if>
+									</h3>
+									<div class="btn-group pull-right">
+										<la:link href="/admin/dict"
+											styleClass="btn btn-default btn-xs">
+											<i class="fa fa-book"></i>
+											<la:message key="labels.dict_list_link" />
+										</la:link>
+										<la:link href="../list/1?dictId=${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-th-list"></i>
+											<la:message key="labels.dict_mapping_list_link" />
+										</la:link>
+										<la:link href="../createnew/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-plus"></i>
+											<la:message key="labels.dict_mapping_link_create" />
+										</la:link>
+										<la:link href="../downloadpage/${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-download"></i>
+											<la:message key="labels.dict_mapping_link_download" />
+										</la:link>
+										<la:link href="../uploadpage/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-upload"></i>
+											<la:message key="labels.dict_mapping_link_upload" />
+										</la:link>
+									</div>
+								</div>
+								<%-- Box Body --%>
+								<div class="box-body">
+									<%-- Message --%>
+									<div>
+										<la:info id="msg" message="true">
+											<div class="alert alert-info">${msg}</div>
+										</la:info>
+										<la:errors />
+									</div>
+									<%-- Form Fields --%>
+									<table class="table table-bordered">
+										<tbody>
+											<tr>
+												<th><la:message
+														key="labels.dict_mapping_source" /></th>
+												<td>${f:br(f:h(inputs))}<la:hidden property="inputs" /></td>
+											</tr>
+											<tr>
+												<th><la:message key="labels.dict_mapping_target" /></th>
+												<td>${f:br(f:h(outputs))}<la:hidden property="outputs" /></td>
+											</tr>
+										</tbody>
+									</table>
+								</div>
+								<!-- /.box-body -->
+								<div class="box-footer">
+									<jsp:include page="/WEB-INF/view/common/admin/crud/buttons.jsp"></jsp:include>
+								</div>
+								<!-- /.box-footer -->
+							</div>
+							<!-- /.box -->
+						</div>
+					</div>
+				</la:form>
+			</section>
+		</div>
+		<jsp:include page="/WEB-INF/view/common/admin/footer.jsp"></jsp:include>
+	</div>
+	<jsp:include page="/WEB-INF/view/common/admin/foot.jsp"></jsp:include>
+</body>
+</html>

+ 103 - 0
src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_download.jsp

@@ -0,0 +1,103 @@
+<%@page pageEncoding="UTF-8" contentType="text/html; charset=UTF-8"%><!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title><la:message key="labels.admin_brand_title" /> | <la:message
+		key="labels.dict_mapping_configuration" /></title>
+<jsp:include page="/WEB-INF/view/common/admin/head.jsp"></jsp:include>
+</head>
+<body class="skin-blue sidebar-mini">
+	<div class="wrapper">
+		<jsp:include page="/WEB-INF/view/common/admin/header.jsp"></jsp:include>
+		<jsp:include page="/WEB-INF/view/common/admin/sidebar.jsp">
+			<jsp:param name="menuCategoryType" value="system" />
+			<jsp:param name="menuType" value="dict" />
+		</jsp:include>
+		<div class="content-wrapper">
+			<section class="content-header">
+				<h1>
+					<la:message key="labels.dict_mapping_title" />
+				</h1>
+				<ol class="breadcrumb">
+					<li><la:link href="list">
+							<la:message key="labels.dict_list_link" />
+						</la:link></li>
+					<li><la:link href="list/0/?dictId=${f:u(dictId)}">
+							<la:message key="labels.dict_mapping_list_link" />
+						</la:link></li>
+					<li class="active"><la:message
+							key="labels.dict_mapping_link_download" /></li>
+				</ol>
+			</section>
+			<section class="content">
+				<la:form action="/admin/dict/mapping/">
+					<la:hidden property="dictId" />
+					<div class="row">
+						<div class="col-md-12">
+							<div class="box box-primary">
+								<div class="box-header with-border">
+									<h3 class="box-title">
+										<la:message key="labels.dict_mapping_link_download" />
+									</h3>
+									<div class="btn-group pull-right">
+										<la:link href="/admin/dict"
+											styleClass="btn btn-default btn-xs">
+											<i class="fa fa-book"></i>
+											<la:message key="labels.dict_list_link" />
+										</la:link>
+										<la:link href="../list/0/?dictId=${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-th-list"></i>
+											<la:message key="labels.dict_mapping_list_link" />
+										</la:link>
+										<la:link href="../createnew/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-plus"></i>
+											<la:message key="labels.dict_mapping_link_create" />
+										</la:link>
+										<la:link href="../downloadpage/${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-download"></i>
+											<la:message key="labels.dict_mapping_link_download" />
+										</la:link>
+										<la:link href="../uploadpage/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-upload"></i>
+											<la:message key="labels.dict_mapping_link_upload" />
+										</la:link>
+									</div>
+								</div>
+								<!-- /.box-header -->
+								<div class="box-body">
+									<%-- Message --%>
+									<div>
+										<la:info id="msg" message="true">
+											<div class="alert alert-info">${msg}</div>
+										</la:info>
+										<la:errors />
+									</div>
+									<div class="form-group">
+										<label for="name" class="col-sm-12 control-label">${f:h(path)}</label>
+									</div>
+								</div>
+								<!-- /.box-body -->
+								<div class="box-footer">
+									<button type="submit" class="btn btn-primary" name="download"
+										value="<la:message key="labels.dict_mapping_button_download" />">
+										<i class="fa fa-download"></i>
+										<la:message key="labels.dict_mapping_button_download" />
+									</button>
+								</div>
+								<!-- /.box-footer -->
+							</div>
+							<!-- /.box -->
+						</div>
+					</div>
+				</la:form>
+			</section>
+		</div>
+		<jsp:include page="/WEB-INF/view/common/admin/footer.jsp"></jsp:include>
+	</div>
+	<jsp:include page="/WEB-INF/view/common/admin/foot.jsp"></jsp:include>
+</body>
+</html>

+ 129 - 0
src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_edit.jsp

@@ -0,0 +1,129 @@
+<%@page pageEncoding="UTF-8" contentType="text/html; charset=UTF-8"%><!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title><la:message key="labels.admin_brand_title" /> | <la:message
+		key="labels.dict_mapping_configuration" /></title>
+<jsp:include page="/WEB-INF/view/common/admin/head.jsp"></jsp:include>
+</head>
+<body class="skin-blue sidebar-mini">
+	<div class="wrapper">
+		<jsp:include page="/WEB-INF/view/common/admin/header.jsp"></jsp:include>
+		<jsp:include page="/WEB-INF/view/common/admin/sidebar.jsp">
+			<jsp:param name="menuCategoryType" value="system" />
+			<jsp:param name="menuType" value="dict" />
+		</jsp:include>
+		<div class="content-wrapper">
+			<section class="content-header">
+				<h1>
+					<la:message key="labels.dict_mapping_title" />
+				</h1>
+				<ol class="breadcrumb">
+					<li><la:link href="list">
+							<la:message key="labels.dict_list_link" />
+						</la:link></li>
+					<li><la:link href="list/0/?dictId=${f:u(dictId)}">
+							<la:message key="labels.dict_mapping_list_link" />
+						</la:link></li>
+					<c:if test="${crudMode == 1}">
+						<li class="active"><la:message
+								key="labels.dict_mapping_link_create" /></li>
+					</c:if>
+					<c:if test="${crudMode == 2}">
+						<li class="active"><la:message
+								key="labels.dict_mapping_link_edit" /></li>
+					</c:if>
+				</ol>
+			</section>
+			<section class="content">
+				<la:form action="/admin/dict/mapping/" styleClass="form-horizontal">
+					<la:hidden property="crudMode" />
+					<la:hidden property="dictId" />
+					<c:if test="${crudMode==2}">
+						<la:hidden property="id" />
+					</c:if>
+					<div class="row">
+						<div class="col-md-12">
+							<div
+								class="box <c:if test="${crudMode == 1}">box-success</c:if><c:if test="${crudMode == 2}">box-warning</c:if>">
+								<div class="box-header with-border">
+									<h3 class="box-title">
+										<c:if test="${crudMode == 1}">
+											<la:message key="labels.dict_mapping_link_create" />
+										</c:if>
+										<c:if test="${crudMode == 2}">
+											<la:message key="labels.dict_mapping_link_edit" />
+										</c:if>
+									</h3>
+									<div class="btn-group pull-right">
+										<la:link href="/admin/dict"
+											styleClass="btn btn-default btn-xs">
+											<i class="fa fa-book"></i>
+											<la:message key="labels.dict_list_link" />
+										</la:link>
+										<la:link href="../list/1?dictId=${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-th-list"></i>
+											<la:message key="labels.dict_mapping_list_link" />
+										</la:link>
+										<la:link href="../createnew/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-plus"></i>
+											<la:message key="labels.dict_mapping_link_create" />
+										</la:link>
+										<la:link href="../downloadpage/${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-download"></i>
+											<la:message key="labels.dict_mapping_link_download" />
+										</la:link>
+										<la:link href="../uploadpage/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-upload"></i>
+											<la:message key="labels.dict_mapping_link_upload" />
+										</la:link>
+									</div>
+								</div>
+								<!-- /.box-header -->
+								<div class="box-body">
+									<div>
+										<la:info id="msg" message="true">
+											<div class="alert alert-info">${msg}</div>
+										</la:info>
+										<la:errors property="_global" />
+									</div>
+									<div class="form-group">
+										<label for="term" class="col-sm-3 control-label"><la:message
+												key="labels.dict_mapping_source" /></label>
+										<div class="col-sm-9">
+											<la:errors property="inputs" />
+											<la:textarea property="inputs" rows="5"
+												styleClass="form-control" />
+										</div>
+									</div>
+									<div class="form-group">
+										<label for="outputs" class="col-sm-3 control-label"><la:message
+												key="labels.dict_mapping_target" /></label>
+										<div class="col-sm-9">
+											<la:errors property="outputs" />
+											<la:textarea property="outputs" rows="5"
+												styleClass="form-control" />
+										</div>
+									</div>
+								</div>
+								<!-- /.box-body -->
+								<div class="box-footer">
+									<jsp:include page="/WEB-INF/view/common/admin/crud/buttons.jsp"></jsp:include>
+								</div>
+								<!-- /.box-footer -->
+							</div>
+							<!-- /.box -->
+						</div>
+					</div>
+				</la:form>
+			</section>
+		</div>
+		<jsp:include page="/WEB-INF/view/common/admin/footer.jsp"></jsp:include>
+	</div>
+	<jsp:include page="/WEB-INF/view/common/admin/foot.jsp"></jsp:include>
+</body>
+</html>

+ 107 - 0
src/main/webapp/WEB-INF/view/admin/dict/mapping/admin_dict_mapping_upload.jsp

@@ -0,0 +1,107 @@
+<%@page pageEncoding="UTF-8" contentType="text/html; charset=UTF-8"%><!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<title><la:message key="labels.admin_brand_title" /> | <la:message
+		key="labels.dict_mapping_configuration" /></title>
+<jsp:include page="/WEB-INF/view/common/admin/head.jsp"></jsp:include>
+</head>
+<body class="skin-blue sidebar-mini">
+	<div class="wrapper">
+		<jsp:include page="/WEB-INF/view/common/admin/header.jsp"></jsp:include>
+		<jsp:include page="/WEB-INF/view/common/admin/sidebar.jsp">
+			<jsp:param name="menuCategoryType" value="system" />
+			<jsp:param name="menuType" value="dict" />
+		</jsp:include>
+		<div class="content-wrapper">
+			<section class="content-header">
+				<h1>
+					<la:message key="labels.dict_mapping_title" />
+				</h1>
+				<ol class="breadcrumb">
+					<li><la:link href="list">
+							<la:message key="labels.dict_list_link" />
+						</la:link></li>
+					<li><la:link href="list/0/?dictId=${f:u(dictId)}">
+							<la:message key="labels.dict_mapping_list_link" />
+						</la:link></li>
+					<li class="active"><la:message
+							key="labels.dict_mapping_link_upload" /></li>
+				</ol>
+			</section>
+			<section class="content">
+				<la:form action="/admin/dict/mapping/upload" enctype="multipart/form-data">
+					<la:hidden property="dictId" />
+					<div class="row">
+						<div class="col-md-12">
+							<div class="box box-primary">
+								<div class="box-header with-border">
+									<h3 class="box-title">
+										<la:message key="labels.dict_mapping_link_upload" />
+									</h3>
+									<div class="btn-group pull-right">
+										<la:link href="/admin/dict"
+											styleClass="btn btn-default btn-xs">
+											<i class="fa fa-book"></i>
+											<la:message key="labels.dict_list_link" />
+										</la:link>
+										<la:link href="../list/0/?dictId=${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-th-list"></i>
+											<la:message key="labels.dict_mapping_list_link" />
+										</la:link>
+										<la:link href="../createnew/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-plus"></i>
+											<la:message key="labels.dict_mapping_link_create" />
+										</la:link>
+										<la:link href="../downloadpage/${f:u(dictId)}"
+											styleClass="btn btn-primary btn-xs">
+											<i class="fa fa-download"></i>
+											<la:message key="labels.dict_mapping_link_download" />
+										</la:link>
+										<la:link href="../uploadpage/${f:u(dictId)}"
+											styleClass="btn btn-success btn-xs">
+											<i class="fa fa-upload"></i>
+											<la:message key="labels.dict_mapping_link_upload" />
+										</la:link>
+									</div>
+								</div>
+								<!-- /.box-header -->
+								<div class="box-body">
+									<%-- Message --%>
+									<div>
+										<la:info id="msg" message="true">
+											<div class="alert alert-info">${msg}</div>
+										</la:info>
+										<la:errors />
+									</div>
+									<div class="form-group">
+										<label for="name" class="col-sm-3 control-label"><la:message
+												key="labels.dict_mapping_file" /></label>
+										<div class="col-sm-9">
+											<input type="file" name="mappingFile" />
+										</div>
+									</div>
+								</div>
+								<!-- /.box-body -->
+								<div class="box-footer">
+									<button type="submit" class="btn btn-success"
+										value="<la:message key="labels.dict_mapping_button_upload" />">
+										<i class="fa fa-upload"></i>
+										<la:message key="labels.dict_mapping_button_upload" />
+									</button>
+								</div>
+								<!-- /.box-footer -->
+							</div>
+							<!-- /.box -->
+						</div>
+					</div>
+				</la:form>
+			</section>
+		</div>
+		<jsp:include page="/WEB-INF/view/common/admin/footer.jsp"></jsp:include>
+	</div>
+	<jsp:include page="/WEB-INF/view/common/admin/foot.jsp"></jsp:include>
+</body>
+</html>