neil
6 years ago
12 changed files with 1238 additions and 247 deletions
@ -0,0 +1,825 @@
|
||||
package com.fr.design.onlineupdate.ui.dialog; |
||||
|
||||
import com.fr.design.RestartHelper; |
||||
import com.fr.design.constants.LayoutConstants; |
||||
import com.fr.design.dialog.UIDialog; |
||||
import com.fr.design.gui.ibutton.UIButton; |
||||
import com.fr.design.gui.icontainer.UIScrollPane; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.gui.itextfield.UITextField; |
||||
import com.fr.design.layout.TableLayout; |
||||
import com.fr.design.layout.TableLayoutHelper; |
||||
import com.fr.design.mainframe.DesignerContext; |
||||
import com.fr.design.onlineupdate.actions.FileDownloader; |
||||
import com.fr.design.onlineupdate.domain.DownloadItem; |
||||
import com.fr.design.onlineupdate.domain.UpdateConstants; |
||||
import com.fr.design.onlineupdate.domain.UpdateInfoCachePropertyManager; |
||||
import com.fr.design.onlineupdate.factory.DirectoryOperationFactory; |
||||
import com.fr.design.onlineupdate.ui.widget.LoadingLabel; |
||||
import com.fr.design.onlineupdate.ui.widget.UpdateActionLabel; |
||||
import com.fr.design.onlineupdate.ui.widget.UpdateInfoTable; |
||||
import com.fr.design.onlineupdate.ui.widget.UpdateInfoTableCellRender; |
||||
import com.fr.design.onlineupdate.ui.widget.UpdateInfoTableModel; |
||||
import com.fr.design.onlineupdate.ui.widget.UpdateInfoTextAreaCellRender; |
||||
import com.fr.design.utils.gui.GUICoreUtils; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.DateUtils; |
||||
import com.fr.general.GeneralContext; |
||||
import com.fr.general.GeneralUtils; |
||||
import com.fr.general.IOUtils; |
||||
import com.fr.general.SiteCenter; |
||||
import com.fr.general.http.HttpClient; |
||||
import com.fr.json.JSONArray; |
||||
import com.fr.json.JSONObject; |
||||
import com.fr.log.FineLoggerFactory; |
||||
import com.fr.stable.ArrayUtils; |
||||
import com.fr.stable.ProductConstants; |
||||
import com.fr.stable.StableUtils; |
||||
import com.fr.stable.StringUtils; |
||||
import com.fr.stable.project.ProjectConstants; |
||||
import com.fr.workspace.WorkContext; |
||||
import com.sun.java.swing.plaf.motif.MotifProgressBarUI; |
||||
|
||||
import javax.swing.BorderFactory; |
||||
import javax.swing.JPanel; |
||||
import javax.swing.JProgressBar; |
||||
import javax.swing.RowSorter; |
||||
import javax.swing.SortOrder; |
||||
import javax.swing.SwingConstants; |
||||
import javax.swing.SwingWorker; |
||||
import javax.swing.event.DocumentEvent; |
||||
import javax.swing.event.DocumentListener; |
||||
import javax.swing.table.TableRowSorter; |
||||
import java.awt.BorderLayout; |
||||
import java.awt.Color; |
||||
import java.awt.Component; |
||||
import java.awt.Dialog; |
||||
import java.awt.Dimension; |
||||
import java.awt.Frame; |
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.ActionListener; |
||||
import java.io.*; |
||||
import java.text.ParsePosition; |
||||
import java.text.SimpleDateFormat; |
||||
import java.util.ArrayList; |
||||
import java.util.Date; |
||||
import java.util.HashMap; |
||||
import java.util.HashSet; |
||||
import java.util.List; |
||||
import java.util.Locale; |
||||
import java.util.Map; |
||||
import java.util.Set; |
||||
import java.util.concurrent.ExecutionException; |
||||
|
||||
/** |
||||
* Created by XINZAI on 2018/8/21. |
||||
*/ |
||||
public class UpdateMainDialog extends UIDialog { |
||||
public static final Dimension DEFAULT = new Dimension(660, 620); |
||||
|
||||
private static final Dimension PROGRESSBAR = new Dimension(120, 15); |
||||
private static final Dimension UPDATE_BUTTON = new Dimension(80, 24); |
||||
private static final int UPDATE_PANE_ROW_SIZE = 30; |
||||
private static final int UPDATE_CONTENT_PANE_ROW_SIZE = 10; |
||||
private static final int UPDATE_CONTENT_PANE_COLUMN_SIZE = 10; |
||||
private static final int UPDATE_CONTENT_PANE_LABEL_COLUMN_SIZE = 100; |
||||
private static final int SEARCH_PANE_ROW_SIZE = 50; |
||||
private static final int SEARCH_PANE_TEXT_COLUMN = 130; |
||||
private static final int SEARCH_PANE_COLUMN_GAP = 3; |
||||
private static final int UPDATE_INFO_TABLE_HEADER_TIME_WIDTH = 120; |
||||
private static final int UPDATE_CONTENT_PANE_BORDER_COLOR = 0xCCCCCC; |
||||
private static final int RESTORE_LABEL_COLOR = 0x3384F0; |
||||
|
||||
private static final String UPDATE_CACHE_STATE_FAIL = "fail"; |
||||
private static final String UPDATE_CACHE_STATE_SUCCESS = "success"; |
||||
|
||||
private static final SimpleDateFormat CHANGELOG_FORMAT = new SimpleDateFormat("M/d/y, h:m:s a", Locale.ENGLISH); |
||||
private static final SimpleDateFormat UPDATE_INFO_TABLE_FORMAT = new SimpleDateFormat("yyyy.MM.dd"); |
||||
|
||||
private Set<DownloadItem> downloadItems = new HashSet<DownloadItem>(); |
||||
private JSONObject downloadFileConfig; |
||||
//最新版本标签
|
||||
private LoadingLabel loadingLabel; |
||||
//更新按钮
|
||||
private UIButton updateButton; |
||||
//有新版本提示标签
|
||||
private UILabel updateLabel; |
||||
|
||||
//jar包版本信息面板,包括当前版本和最新版本
|
||||
private JPanel jarVersionInfoPane; |
||||
//jar包更新信息面板,包括每个版本更新的信息
|
||||
private JPanel jarUpdateInfoPane; |
||||
//jar包更新操作面板,包括更新重启按钮和进度条
|
||||
private JPanel updateActionPane; |
||||
//进度条
|
||||
private JProgressBar progressBar; |
||||
//更新版本提示面板
|
||||
private JPanel updateVersionReminderPane; |
||||
//jar包版本标签
|
||||
private UILabel jarCurrentLabel; |
||||
//jar包还原标签
|
||||
private UILabel jarRestoreLabel; |
||||
//更新信息搜索按钮
|
||||
private UIButton searchUpdateInfoBtn; |
||||
//搜索更新信息关键词文本框
|
||||
private UITextField searchUpdateInfoKeyword; |
||||
|
||||
private boolean updateSuccessful; |
||||
|
||||
private UpdateInfoTable updateInfoTable; |
||||
|
||||
private ArrayList<Object[]> updateInfoList; |
||||
|
||||
private boolean getUpdateInfoSuccess; |
||||
|
||||
private UpdateInfoCachePropertyManager cacheProperty; |
||||
private String lastUpdateCacheTime; |
||||
private String lastUpdateCacheState = UPDATE_CACHE_STATE_FAIL; |
||||
|
||||
public UpdateMainDialog(Dialog parent) { |
||||
super(parent); |
||||
initComponents(); |
||||
} |
||||
|
||||
public UpdateMainDialog(Frame parent) { |
||||
super(parent); |
||||
setModal(true); |
||||
initComponents(); |
||||
} |
||||
|
||||
private void initUpdateActionPane() { |
||||
double[] rowUpdateSubContentPaneSize = {UPDATE_CONTENT_PANE_ROW_SIZE, TableLayout.PREFERRED, UPDATE_CONTENT_PANE_ROW_SIZE}; |
||||
double[] rowUpdateContentPaneSize = {TableLayout.PREFERRED}; |
||||
double[] columnUpdateSubContentPaneProgressSize = {TableLayout.FILL, TableLayout.PREFERRED}; |
||||
double[] columnUpdateSubContentPaneSize = {UPDATE_CONTENT_PANE_COLUMN_SIZE, TableLayout.FILL, TableLayout.PREFERRED}; |
||||
JPanel progressBarPane = new JPanel(new BorderLayout()); |
||||
|
||||
progressBar = new JProgressBar(); |
||||
progressBar.setUI(new MotifProgressBarUI()); |
||||
progressBar.setForeground(UpdateConstants.BAR_COLOR); |
||||
progressBar.setVisible(false); |
||||
progressBar.setStringPainted(true); |
||||
progressBar.setPreferredSize(PROGRESSBAR); |
||||
|
||||
updateLabel = new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_New_Version_Available")); |
||||
updateLabel.setHorizontalAlignment(SwingConstants.RIGHT); |
||||
updateLabel.setVisible(false); |
||||
|
||||
progressBarPane.add(GUICoreUtils.createBorderLayoutPane( |
||||
progressBar, BorderLayout.CENTER, |
||||
updateLabel, BorderLayout.EAST |
||||
), BorderLayout.CENTER); |
||||
|
||||
updateActionPane = TableLayoutHelper.createCommonTableLayoutPane(new Component[][]{ |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()}, |
||||
new Component[]{new UILabel(), initPaneContent(getBackground(), rowUpdateContentPaneSize, columnUpdateSubContentPaneProgressSize, progressBarPane, updateButton), new UILabel()}, |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()} |
||||
}, rowUpdateSubContentPaneSize, columnUpdateSubContentPaneSize, LayoutConstants.VGAP_LARGE); |
||||
} |
||||
|
||||
private JPanel initPaneContent(Color color, double[] row, double[] column, Component... var) { |
||||
JPanel paneContent = TableLayoutHelper.createTableLayoutPane(new Component[][]{var}, row, column); |
||||
paneContent.setBackground(color); |
||||
return paneContent; |
||||
} |
||||
|
||||
private void initJarVersionInfoPane() { |
||||
double[] rowUpdatePaneSize = {UPDATE_PANE_ROW_SIZE, TableLayout.PREFERRED, TableLayout.PREFERRED}; |
||||
double[] columnUpdatePaneSize = {TableLayout.PREFERRED, TableLayout.FILL, TableLayout.PREFERRED}; |
||||
double[] rowUpdateContentPaneSize = {TableLayout.PREFERRED}; |
||||
double[] columnUpdateContentPaneSize = {TableLayout.PREFERRED, TableLayout.FILL, TableLayout.PREFERRED}; |
||||
double[] rowUpdateSubContentPaneSize = {UPDATE_CONTENT_PANE_ROW_SIZE, TableLayout.PREFERRED, TableLayout.PREFERRED, TableLayout.PREFERRED, UPDATE_CONTENT_PANE_ROW_SIZE}; |
||||
double[] columnUpdateSubContentPaneSize = {UPDATE_CONTENT_PANE_COLUMN_SIZE, TableLayout.FILL, TableLayout.PREFERRED}; |
||||
double[] columnUpdateSubContentPaneLabelSize = {UPDATE_CONTENT_PANE_LABEL_COLUMN_SIZE, TableLayout.PREFERRED}; |
||||
|
||||
JPanel jarUpdateContentPane = new JPanel(); |
||||
jarUpdateContentPane.setLayout(new BorderLayout()); |
||||
jarUpdateContentPane.setBorder(BorderFactory.createLineBorder(new Color(UPDATE_CONTENT_PANE_BORDER_COLOR))); |
||||
|
||||
JPanel jarUpdateContentPane2 = TableLayoutHelper.createCommonTableLayoutPane(new Component[][]{ |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()}, |
||||
new Component[]{new UILabel(), updateVersionReminderPane, new UILabel()}, |
||||
new Component[]{new UILabel(), initPaneContent(Color.WHITE, rowUpdateContentPaneSize, columnUpdateSubContentPaneLabelSize, new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_JAR_Version")), jarCurrentLabel), new UILabel()}, |
||||
new Component[]{new UILabel(), initPaneContent(Color.WHITE, rowUpdateContentPaneSize, columnUpdateSubContentPaneLabelSize, new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Latest_JAR")), loadingLabel), new UILabel()}, |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()} |
||||
}, rowUpdateSubContentPaneSize, columnUpdateSubContentPaneSize, LayoutConstants.VGAP_LARGE); |
||||
jarUpdateContentPane2.setBackground(Color.WHITE); |
||||
jarUpdateContentPane.add(jarUpdateContentPane2); |
||||
jarVersionInfoPane = TableLayoutHelper.createCommonTableLayoutPane(new Component[][]{ |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()}, |
||||
new Component[]{new UILabel(), initPaneContent(getBackground(), rowUpdateContentPaneSize, columnUpdateContentPaneSize, new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_JarUpdate")), new UILabel(), jarRestoreLabel), new UILabel()}, |
||||
new Component[]{new UILabel(), jarUpdateContentPane, new UILabel()} |
||||
}, rowUpdatePaneSize, columnUpdatePaneSize, LayoutConstants.VGAP_LARGE); |
||||
} |
||||
|
||||
private void initJarUpdateInfoPane() { |
||||
double[] rowUpdatePaneSize = {SEARCH_PANE_ROW_SIZE, TableLayout.FILL}; |
||||
double[] columnUpdatePaneSize = {TableLayout.PREFERRED, TableLayout.FILL, TableLayout.PREFERRED}; |
||||
|
||||
double[] searchRow = {UPDATE_CONTENT_PANE_ROW_SIZE, TableLayout.PREFERRED, SEARCH_PANE_COLUMN_GAP * 2}; |
||||
double[] searchColumn = {TableLayout.FILL, SEARCH_PANE_TEXT_COLUMN, TableLayout.PREFERRED}; |
||||
initUpdateInfoSearchPane(); |
||||
JPanel searchPane = TableLayoutHelper.createCommonTableLayoutPane(new Component[][]{ |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()}, |
||||
new Component[]{new UILabel(), searchUpdateInfoKeyword, searchUpdateInfoBtn}, |
||||
new Component[]{new UILabel(), new UILabel(), new UILabel()} |
||||
}, searchRow, searchColumn, LayoutConstants.VGAP_LARGE); |
||||
|
||||
String[] columnNames = {com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Date"), com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Content"), com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_SignHeader")}; |
||||
initUpdateInfoTable(columnNames); |
||||
|
||||
UIScrollPane uiScrollPane = new UIScrollPane(updateInfoTable); |
||||
jarUpdateInfoPane = TableLayoutHelper.createCommonTableLayoutPane(new Component[][]{ |
||||
new Component[]{new UILabel(), searchPane, new UILabel()}, |
||||
new Component[]{new UILabel(), uiScrollPane, new UILabel()} |
||||
}, rowUpdatePaneSize, columnUpdatePaneSize, LayoutConstants.VGAP_LARGE); |
||||
} |
||||
|
||||
private void initUpdateInfoTable(String[] columnNames) { |
||||
int updateTimeColIndex = 0; |
||||
int updateTitleColIndex = 1; |
||||
int updateSignColIndex = 2; |
||||
|
||||
updateInfoTable = new UpdateInfoTable(columnNames); |
||||
|
||||
updateInfoTable.setShowGrid(false); |
||||
updateInfoTable.setCellSelectionEnabled(false); |
||||
TableRowSorter<UpdateInfoTableModel> sorter = new TableRowSorter<UpdateInfoTableModel>(updateInfoTable.getDataModel()); |
||||
sorter.setSortable(updateTimeColIndex, true); |
||||
sorter.setSortable(updateTitleColIndex, false); |
||||
sorter.setSortable(updateSignColIndex, false); |
||||
updateInfoTable.setRowSorter(sorter); |
||||
List<RowSorter.SortKey> sortKeys = new ArrayList<RowSorter.SortKey>(); |
||||
sortKeys.add(new RowSorter.SortKey(updateTimeColIndex, SortOrder.DESCENDING)); |
||||
sorter.setSortKeys(sortKeys); |
||||
|
||||
updateInfoTable.getTableHeader().setReorderingAllowed(false); |
||||
updateInfoTable.getColumnModel().getColumn(updateTimeColIndex).setMaxWidth(UPDATE_INFO_TABLE_HEADER_TIME_WIDTH); |
||||
updateInfoTable.getColumnModel().getColumn(updateTimeColIndex).setMinWidth(UPDATE_INFO_TABLE_HEADER_TIME_WIDTH); |
||||
updateInfoTable.getColumnModel().getColumn(updateSignColIndex).setMaxWidth(0); |
||||
updateInfoTable.getColumnModel().getColumn(updateSignColIndex).setMinWidth(0); |
||||
updateInfoTable.getTableHeader().getColumnModel().getColumn(updateSignColIndex).setMaxWidth(0); |
||||
updateInfoTable.getTableHeader().getColumnModel().getColumn(updateSignColIndex).setMinWidth(0); |
||||
updateInfoTable.getColumn(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Date")).setCellRenderer(new UpdateInfoTableCellRender()); |
||||
updateInfoTable.getColumn(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Content")).setCellRenderer(new UpdateInfoTextAreaCellRender()); |
||||
} |
||||
|
||||
private void initUpdateInfoSearchPane() { |
||||
searchUpdateInfoKeyword = new UITextField(); |
||||
searchUpdateInfoKeyword.getDocument().addDocumentListener(new DocumentListener() { |
||||
@Override |
||||
public void insertUpdate(DocumentEvent e) { |
||||
} |
||||
|
||||
@Override |
||||
public void removeUpdate(DocumentEvent e) { |
||||
String keyword = searchUpdateInfoKeyword.getText(); |
||||
if (ComparatorUtils.equals(keyword, StringUtils.EMPTY) && getUpdateInfoSuccess) { |
||||
updateInfoList.clear(); |
||||
getUpdateInfo(keyword).execute(); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void changedUpdate(DocumentEvent e) { |
||||
} |
||||
}); |
||||
searchUpdateInfoBtn = new UIButton(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Search")); |
||||
searchUpdateInfoBtn.addActionListener(new ActionListener() { |
||||
@Override |
||||
public void actionPerformed(ActionEvent e) { |
||||
if (getUpdateInfoSuccess) { |
||||
updateInfoList.clear(); |
||||
getUpdateInfo(searchUpdateInfoKeyword.getText()).execute(); |
||||
} |
||||
} |
||||
}); |
||||
} |
||||
|
||||
private void initButtonAndLabel() { |
||||
loadingLabel = new LoadingLabel(); |
||||
loadingLabel.setText(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Checking_Jar_Update")); |
||||
updateButton = new UIButton(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Update")); |
||||
updateButton.setPreferredSize(UPDATE_BUTTON); |
||||
updateButton.setEnabled(false); |
||||
|
||||
double[] rowSize = {TableLayout.PREFERRED}; |
||||
|
||||
double[] colSize = {UPDATE_CONTENT_PANE_LABEL_COLUMN_SIZE, TableLayout.PREFERRED}; |
||||
updateVersionReminderPane = initPaneContent( |
||||
Color.WHITE, rowSize, colSize, |
||||
new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Designer_Version")), |
||||
new UILabel(UpdateConstants.DEFAULT_APP_NAME + StringUtils.BLANK + ProductConstants.VERSION) |
||||
); |
||||
|
||||
|
||||
jarCurrentLabel = new UILabel(StringUtils.isEmpty(GeneralUtils.readBuildNO()) ? com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Not_Install_Version") : GeneralUtils.readBuildNO(), SwingConstants.CENTER); |
||||
UILabel noJarPreviousRevision = new UILabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_No_Previous_Version")); |
||||
UpdateActionLabel jarRestorePreviousRevision = new UpdateActionLabel(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Restore"), false); |
||||
jarRestorePreviousRevision.setForeground(new Color(RESTORE_LABEL_COLOR)); |
||||
jarRestorePreviousRevision.addActionListener(new ActionListener() { |
||||
@Override |
||||
public void actionPerformed(ActionEvent e) { |
||||
RestoreDialog dialog = new RestoreDialog(DesignerContext.getDesignerFrame(), true); |
||||
dialog.showDialog(); |
||||
} |
||||
}); |
||||
//choose RestoreLabel to show
|
||||
boolean isNeedRestore = ArrayUtils.isNotEmpty(DirectoryOperationFactory.listFilteredFiles(StableUtils.getInstallHome(), getBackupDirectory())); |
||||
jarRestoreLabel = isNeedRestore ? jarRestorePreviousRevision : noJarPreviousRevision; |
||||
} |
||||
|
||||
private void initComponents() { |
||||
JPanel contentPane = (JPanel) getContentPane(); |
||||
contentPane.setLayout(new BorderLayout()); |
||||
|
||||
|
||||
initButtonAndLabel(); |
||||
|
||||
initJarVersionInfoPane(); |
||||
initJarUpdateInfoPane(); |
||||
initUpdateActionPane(); |
||||
|
||||
add(jarVersionInfoPane, BorderLayout.NORTH); |
||||
|
||||
//海外版本不显示更新信息
|
||||
if (GeneralContext.getLocale().equals(Locale.CHINA) || GeneralContext.getLocale().equals(Locale.TAIWAN)){ |
||||
add(jarUpdateInfoPane, BorderLayout.CENTER); |
||||
add(updateActionPane, BorderLayout.SOUTH); |
||||
}else { |
||||
add(updateActionPane, BorderLayout.CENTER); |
||||
} |
||||
|
||||
addActionListenerForUpdateBtn(); |
||||
|
||||
new SwingWorker<JSONObject, Void>() { |
||||
@Override |
||||
protected JSONObject doInBackground() throws Exception { |
||||
HttpClient hc = new HttpClient(SiteCenter.getInstance().acquireUrlByKind("jar10.update")); |
||||
hc.setTimeout(UpdateConstants.CONNECTION_TIMEOUT); |
||||
return new JSONObject(hc.getResponseText()); |
||||
} |
||||
|
||||
@Override |
||||
protected void done() { |
||||
try { |
||||
downloadFileConfig = get(); |
||||
showDownLoadInfo(); |
||||
} catch (InterruptedException e) { |
||||
stopLoading(); |
||||
} catch (ExecutionException e) { |
||||
stopLoading(); |
||||
} finally { |
||||
getUpdateInfo(StringUtils.EMPTY).execute(); |
||||
} |
||||
} |
||||
}.execute(); |
||||
} |
||||
|
||||
private SwingWorker<JSONArray, Void> getUpdateInfo(final String keyword) { |
||||
updateInfoList = new ArrayList<Object[]>(); |
||||
lastUpdateCacheTime = UpdateConstants.CHANGELOG_X_START; |
||||
String cacheConfigPath = getUpdateCacheConfig(); |
||||
cacheProperty = new UpdateInfoCachePropertyManager(StableUtils.pathJoin(WorkContext.getCurrent().getPath(), "resources", "offlineres", cacheConfigPath)); |
||||
String recordUpdateTime = cacheProperty.readProperty("updateTime"); |
||||
if (StringUtils.isNotEmpty(recordUpdateTime)) { |
||||
lastUpdateCacheTime = recordUpdateTime; |
||||
} |
||||
String recordUpdateState = cacheProperty.readProperty("updateState"); |
||||
if (StringUtils.isNotEmpty(recordUpdateState)) { |
||||
lastUpdateCacheState = recordUpdateState; |
||||
} |
||||
return new SwingWorker<JSONArray, Void>() { |
||||
@Override |
||||
protected JSONArray doInBackground() { |
||||
try { |
||||
getUpdateInfoSuccess = false; |
||||
//step1:read from cache file
|
||||
getCachedUpdateInfo(keyword); |
||||
//step2:read from website,start from cacheRecordTime
|
||||
if (downloadFileConfig == null) { |
||||
throw new Exception("network error."); |
||||
} |
||||
HttpClient hc = new HttpClient(SiteCenter.getInstance().acquireUrlByKind("changelog10") + "&start=" + lastUpdateCacheTime + "&end=" + getLatestJARTimeStr()); |
||||
hc.asGet(); |
||||
hc.setTimeout(UpdateConstants.CONNECTION_TIMEOUT * 2); |
||||
String responseText = hc.getResponseText(); |
||||
JSONArray array = JSONArray.create(); |
||||
//假如返回"-1",说明socket出错了
|
||||
if (!ComparatorUtils.equals(responseText, "-1")) { |
||||
array = new JSONArray(responseText); |
||||
} |
||||
hc.release(); |
||||
return array; |
||||
} catch (Exception e) { |
||||
FineLoggerFactory.getLogger().error(e.getMessage()); |
||||
} |
||||
return JSONArray.create(); |
||||
} |
||||
|
||||
@Override |
||||
protected void done() { |
||||
try { |
||||
JSONArray jsonArray = get(); |
||||
//step3:generateInfoTableList
|
||||
updateInfoTable.getDataModel().populateBean(generateUpdateInfoList(jsonArray, keyword)); |
||||
|
||||
getUpdateInfoSuccess = true; |
||||
//step4:update cache file,start from cacheRecordTime,end latest server jartime
|
||||
updateCachedInfoFile(jsonArray); |
||||
} catch (Exception e) { |
||||
getUpdateInfoSuccess = true; |
||||
FineLoggerFactory.getLogger().error(e.getMessage()); |
||||
} |
||||
} |
||||
}; |
||||
} |
||||
|
||||
//从文件中读取缓存的更新信息
|
||||
private void getCachedUpdateInfo(String keyword) throws Exception { |
||||
String cacheInfoPath = getUpdateCacheInfo(); |
||||
File cacheFile = new File(StableUtils.pathJoin(WorkContext.getCurrent().getPath(), "resources", "offlineres", cacheInfoPath)); |
||||
if (!ComparatorUtils.equals(lastUpdateCacheState, "success")) { |
||||
cacheFile.delete(); |
||||
return; |
||||
} |
||||
if (cacheFile.exists()) { |
||||
InputStreamReader streamReader = new InputStreamReader(new FileInputStream(cacheFile), "UTF-8"); |
||||
BufferedReader br = new BufferedReader(streamReader); |
||||
String readStr, updateTimeStr; |
||||
|
||||
while ((readStr = br.readLine()) != null) { |
||||
String[] updateInfo = readStr.split("\\t"); |
||||
if (updateInfo.length == 2) { |
||||
updateTimeStr = updateInfo[0]; |
||||
Date updateTime = CHANGELOG_FORMAT.parse(updateTimeStr); |
||||
//形如 Build#release-2018.07.31.03.03.52.80
|
||||
String currentNO = GeneralUtils.readBuildNO(); |
||||
Date curJarDate = UPDATE_INFO_TABLE_FORMAT.parse(currentNO, new ParsePosition(currentNO.indexOf("-") + 1)); |
||||
if (!ComparatorUtils.equals(keyword, StringUtils.EMPTY)) { |
||||
if (!containsKeyword(UPDATE_INFO_TABLE_FORMAT.format(updateTime), keyword) && !containsKeyword(updateInfo[1], keyword)) { |
||||
continue; |
||||
} |
||||
} |
||||
if (isValidLogInfo(updateInfo[1])) { |
||||
updateInfoList.add(new Object[]{UPDATE_INFO_TABLE_FORMAT.format(updateTime), updateInfo[1], updateTime.after(curJarDate)}); |
||||
} |
||||
} |
||||
} |
||||
br.close(); |
||||
streamReader.close(); |
||||
} |
||||
} |
||||
|
||||
private void updateCachedInfoFile(JSONArray jsonArray) throws Exception { |
||||
String cacheDirPath = StableUtils.pathJoin(WorkContext.getCurrent().getPath(), "resources", "offlineres"); |
||||
File cacheFileDir = new File(cacheDirPath); |
||||
if (!StableUtils.mkdirs(cacheFileDir)) { |
||||
FineLoggerFactory.getLogger().error("make dir error."); |
||||
return; |
||||
} |
||||
final File cacheFile = new File(StableUtils.pathJoin(cacheDirPath, getUpdateCacheInfo())); |
||||
if (!cacheFile.exists()) { |
||||
cacheFile.createNewFile(); |
||||
lastUpdateCacheTime = UpdateConstants.CHANGELOG_X_START; |
||||
lastUpdateCacheState = UPDATE_CACHE_STATE_FAIL; |
||||
} |
||||
if (downloadFileConfig == null) { |
||||
return; |
||||
} |
||||
String endTime = getLatestJARTimeStr(); |
||||
if (endTime.equals(lastUpdateCacheTime) || jsonArray.length() == 0 || ComparatorUtils.compare(endTime, lastUpdateCacheTime) <= 0) { |
||||
return; |
||||
} |
||||
OutputStreamWriter writerStream = new OutputStreamWriter(new FileOutputStream(cacheFile), "UTF-8"); |
||||
BufferedWriter bufferWriter = new BufferedWriter(writerStream); |
||||
for (int i = 0; i < jsonArray.length(); i++) { |
||||
JSONObject jo = (JSONObject) jsonArray.get(i); |
||||
bufferWriter.write((String) jo.get("update") + '\t' + jo.get("title")); |
||||
bufferWriter.newLine(); |
||||
bufferWriter.flush(); |
||||
} |
||||
bufferWriter.close(); |
||||
writerStream.close(); |
||||
lastUpdateCacheState = UPDATE_CACHE_STATE_SUCCESS; |
||||
lastUpdateCacheTime = endTime; |
||||
cacheProperty.updateProperty("updateTime", lastUpdateCacheTime); |
||||
cacheProperty.updateProperty("updateState", lastUpdateCacheState); |
||||
} |
||||
|
||||
private ArrayList<Object[]> generateUpdateInfoList(JSONArray jsonArray, String keyword) throws Exception { |
||||
for (int i = 0; i < jsonArray.length(); i++) { |
||||
JSONObject jo = (JSONObject) jsonArray.get(i); |
||||
String updateTitle = (String) jo.get("title"); |
||||
String updateTimeStr = (String) jo.get("update"); |
||||
Date updateTime = CHANGELOG_FORMAT.parse(updateTimeStr); |
||||
//形如 Build#release-2018.07.31.03.03.52.80
|
||||
String currentNO = GeneralUtils.readBuildNO(); |
||||
Date curJarDate = UPDATE_INFO_TABLE_FORMAT.parse(currentNO, new ParsePosition(currentNO.indexOf("-") + 1)); |
||||
if (!ComparatorUtils.equals(keyword, StringUtils.EMPTY)) { |
||||
if (!containsKeyword(UPDATE_INFO_TABLE_FORMAT.format(updateTime), keyword) && !containsKeyword(updateTitle, keyword)) { |
||||
continue; |
||||
} |
||||
} |
||||
if (isValidLogInfo(updateTitle)) { |
||||
updateInfoList.add(new Object[]{UPDATE_INFO_TABLE_FORMAT.format(updateTime), updateTitle, updateTime.after(curJarDate)}); |
||||
} |
||||
} |
||||
return new ArrayList<Object[]>(updateInfoList); |
||||
} |
||||
|
||||
private boolean containsKeyword(String str, String keyword) { |
||||
return str.toUpperCase().contains(keyword.toUpperCase()); |
||||
} |
||||
|
||||
private void stopLoading() { |
||||
loadingLabel.stopLoading(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Connect_VersionUpdateServer_Failed")); |
||||
} |
||||
|
||||
|
||||
private void showDownLoadInfo() { |
||||
//形如 Build#release-2018.07.31.03.03.52.80
|
||||
String buildNO = downloadFileConfig.optString("buildNO"); |
||||
Date jarDate = (new SimpleDateFormat("yyyy.MM.dd")).parse(buildNO, new ParsePosition(buildNO.indexOf("-") + 1)); |
||||
String serverVersionNO = downloadFileConfig.optString("versionNO"); |
||||
String currentVersionNO = ProductConstants.RELEASE_VERSION; |
||||
String[] serverVersionSplitStr = serverVersionNO.split("\\."); |
||||
String[] currentVersionSplitStr = currentVersionNO.split("\\."); |
||||
int index = 0; |
||||
int compareResult; |
||||
int versionLength = Math.min(serverVersionSplitStr.length, currentVersionSplitStr.length); |
||||
|
||||
//形如 Build#release-2018.07.31.03.03.52.80
|
||||
String currentNO = GeneralUtils.readBuildNO(); |
||||
if (!".".equals(StableUtils.getInstallHome())) { |
||||
Date currentDate = (new SimpleDateFormat("yyyy.MM.dd")).parse(currentNO, new ParsePosition(currentNO.indexOf("-") + 1)); |
||||
if (DateUtils.subtractDate(jarDate, currentDate, DateUtils.DAY) > 0) { |
||||
updateButton.setEnabled(true); |
||||
updateLabel.setVisible(true); |
||||
loadingLabel.stopLoading(buildNO.contains("-") ? buildNO.substring(buildNO.lastIndexOf("-") + 1) : buildNO); |
||||
} else { |
||||
loadingLabel.stopLoading(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Already_Latest_Version")); |
||||
} |
||||
} else { |
||||
updateButton.setEnabled(true); |
||||
updateLabel.setVisible(true); |
||||
loadingLabel.stopLoading(buildNO.contains("-") ? buildNO.substring(buildNO.lastIndexOf("-") + 1) : buildNO); |
||||
} |
||||
|
||||
while (index < versionLength) { |
||||
compareResult = serverVersionSplitStr[index].length() - currentVersionSplitStr[index].length(); |
||||
if (0 == compareResult) { |
||||
compareResult = serverVersionSplitStr[index].compareTo(currentVersionSplitStr[index]); |
||||
if (0 == compareResult) { |
||||
++index; |
||||
continue; |
||||
} |
||||
break; |
||||
} |
||||
break; |
||||
} |
||||
|
||||
initMapWithInfo(downloadFileConfig); |
||||
} |
||||
|
||||
public void initMapWithInfo(JSONObject result) { |
||||
addJarNameToMap(result, "designer"); |
||||
addJarNameToMap(result, "server"); |
||||
} |
||||
|
||||
private void addJarNameToMap(JSONObject result, String category) { |
||||
JSONArray jsonArray = result.optJSONArray(category); |
||||
if (jsonArray != null) { |
||||
for (int i = 0, len = jsonArray.length(); i < len; i++) { |
||||
JSONObject jo = jsonArray.optJSONObject(i); |
||||
String downloadName = jo.optString("name"); |
||||
String downloadUrl = jo.optString("url"); |
||||
long downloadSize = jo.optLong("size"); |
||||
if (ComparatorUtils.equals(category, "server")) { |
||||
File currentJAR = new File(StableUtils.pathJoin(WorkContext.getCurrent().getPath(), ProjectConstants.LIB_NAME, downloadName)); |
||||
if (currentJAR.exists() && ComparatorUtils.equals(currentJAR.length(), downloadSize)) { |
||||
//假如大小一样的jar包就不要下载了
|
||||
continue; |
||||
} |
||||
} |
||||
downloadItems.add(new DownloadItem(downloadName, downloadUrl, downloadSize)); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* jar包更新按钮监听器 |
||||
*/ |
||||
private void addActionListenerForUpdateBtn() { |
||||
updateButton.addActionListener(new ActionListener() { |
||||
@Override |
||||
public void actionPerformed(ActionEvent e) { |
||||
if (updateSuccessful) { |
||||
RestartHelper.restart(); |
||||
} else { |
||||
deletePreviousPropertyFile(); |
||||
updateButton.setEnabled(false); |
||||
progressBar.setVisible(true); |
||||
updateLabel.setVisible(false); |
||||
|
||||
new FileDownloader( |
||||
downloadItems.toArray(new DownloadItem[downloadItems.size()]), |
||||
StableUtils.pathJoin(StableUtils.getInstallHome(), UpdateConstants.DOWNLOAD_DIR)) { |
||||
@Override |
||||
protected void process(java.util.List<DownloadItem> chunks) { |
||||
DownloadItem fileInfo = chunks.get(chunks.size() - 1); |
||||
progressBar.setString(fileInfo.getName() + " " + fileInfo.getProgressString()); |
||||
progressBar.setValue(fileInfo.getProgressValue()); |
||||
} |
||||
|
||||
@Override |
||||
public void onDownloadSuccess() { |
||||
updateButton.setEnabled(true); |
||||
progressBar.setVisible(false); |
||||
backup(); |
||||
putNewFiles(); |
||||
updateSuccessful = true; |
||||
updateButton.setText(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_Restart_Designer")); |
||||
} |
||||
|
||||
@Override |
||||
public void onDownloadFailed() { |
||||
progressBar.setVisible(false); |
||||
} |
||||
}.execute(); |
||||
} |
||||
} |
||||
}); |
||||
} |
||||
|
||||
/** |
||||
* 确保升级更新之前删除以前的配置文件 |
||||
*/ |
||||
public static void deletePreviousPropertyFile() { |
||||
//在进行更新升级之前确保move和delete.properties删除
|
||||
File moveFile = new File(RestartHelper.MOVE_FILE); |
||||
File delFile = new File(RestartHelper.RECORD_FILE); |
||||
if ((moveFile.exists()) && (!moveFile.delete())) { |
||||
FineLoggerFactory.getLogger().error(RestartHelper.MOVE_FILE + "delete failed!"); |
||||
} |
||||
if ((delFile.exists()) && (!delFile.delete())) { |
||||
FineLoggerFactory.getLogger().error(RestartHelper.RECORD_FILE + "delete failed!"); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* JAR包更新的时候备份老的jar包,包括设计器相关的和服务器相关的几个 |
||||
*/ |
||||
private void backup() { |
||||
String installHome = StableUtils.getInstallHome(); |
||||
//jar包备份文件的目录为"backup/"+jar包当前版本号
|
||||
String todayBackupDir = StableUtils.pathJoin(installHome, getBackupDirectory(), (GeneralUtils.readBuildNO())); |
||||
backupFilesFromInstallEnv(installHome, todayBackupDir, getJARList4Server()); |
||||
backupFilesFromInstallLib(installHome, todayBackupDir, getJARList4Designer()); |
||||
jarCurrentLabel.setText(downloadFileConfig.optString("buildNO")); |
||||
} |
||||
|
||||
private void backupFilesFromInstallEnv(String installHome, String todayBackupDir, List<String> files) { |
||||
for (String file : files) { |
||||
try { |
||||
IOUtils.copy( |
||||
new File(StableUtils.pathJoin(installHome, UpdateConstants.APPS_FOLDER_NAME, ProductConstants.getAppFolderName(), ProjectConstants.WEBINF_NAME, ProjectConstants.LIB_NAME, file)), |
||||
new File(StableUtils.pathJoin(todayBackupDir))); |
||||
} catch (IOException e) { |
||||
FineLoggerFactory.getLogger().error(e.getMessage()); |
||||
} |
||||
} |
||||
} |
||||
|
||||
private void backupFilesFromInstallLib(String installHome, String todayBackupDir, List<String> files) { |
||||
for (String file : files) { |
||||
try { |
||||
IOUtils.copy( |
||||
new File(StableUtils.pathJoin(installHome, ProjectConstants.LIB_NAME, file)), |
||||
new File(StableUtils.pathJoin(todayBackupDir))); |
||||
} catch (IOException e) { |
||||
FineLoggerFactory.getLogger().error(e.getMessage()); |
||||
} |
||||
} |
||||
} |
||||
|
||||
private void putNewFiles() { |
||||
Map<String, String> map = new HashMap<String, String>(); |
||||
java.util.List<String> list = new ArrayList<String>(); |
||||
String installHome = StableUtils.getInstallHome(); |
||||
putNewFilesToInstallLib(installHome, getDownLoadJAR4Designer(), map, list); |
||||
putNewFilesToInstallEnv(installHome, getDownLoadJAR4Server(), map, list); |
||||
RestartHelper.saveFilesWhichToMove(map); |
||||
RestartHelper.saveFilesWhichToDelete(list.toArray(new String[list.size()])); |
||||
} |
||||
|
||||
private void putNewFilesToInstallLib(String installHome, String[] files, Map<String, String> map, java.util.List<String> list) { |
||||
for (String file : files) { |
||||
map.put(StableUtils.pathJoin(installHome, UpdateConstants.DOWNLOAD_DIR, file), |
||||
StableUtils.pathJoin(installHome, ProjectConstants.LIB_NAME, file)); |
||||
list.add(StableUtils.pathJoin(installHome, ProjectConstants.LIB_NAME, file)); |
||||
} |
||||
} |
||||
|
||||
private void putNewFilesToInstallEnv(String installHome, String[] files, Map<String, String> map, java.util.List<String> list) { |
||||
for (String file : files) { |
||||
map.put(StableUtils.pathJoin(installHome, UpdateConstants.DOWNLOAD_DIR, file), |
||||
StableUtils.pathJoin(installHome, UpdateConstants.APPS_FOLDER_NAME, ProductConstants.getAppFolderName(), ProjectConstants.WEBINF_NAME, ProjectConstants.LIB_NAME, file)); |
||||
list.add(StableUtils.pathJoin(installHome, UpdateConstants.APPS_FOLDER_NAME, ProductConstants.getAppFolderName(), ProjectConstants.WEBINF_NAME, ProjectConstants.LIB_NAME, file)); |
||||
} |
||||
} |
||||
|
||||
//获取备份目录
|
||||
private String getBackupDirectory() { |
||||
return UpdateConstants.DESIGNER_BACKUP_DIR; |
||||
} |
||||
|
||||
//获取服务器jar包列表
|
||||
private List<String> getJARList4Server() { |
||||
return UpdateConstants.JARS_FOR_SERVER_X; |
||||
} |
||||
|
||||
//获取设计器jar包列表
|
||||
private List<String> getJARList4Designer() { |
||||
return UpdateConstants.JARS_FOR_DESIGNER_X; |
||||
} |
||||
|
||||
//获取服务器jar包下载列表
|
||||
private String[] getDownLoadJAR4Server() { |
||||
ArrayList<String> jarList = new ArrayList<String>(); |
||||
List<String> serverItems = getJARList4Server(); |
||||
for (DownloadItem downloadItem : downloadItems) { |
||||
String downloadItemName = downloadItem.getName(); |
||||
if (serverItems.contains(downloadItemName)) { |
||||
jarList.add(downloadItemName); |
||||
} |
||||
} |
||||
return jarList.toArray(new String[jarList.size()]); |
||||
} |
||||
|
||||
//获取设计器jar包下载列表
|
||||
private String[] getDownLoadJAR4Designer() { |
||||
ArrayList<String> jarList = new ArrayList<String>(); |
||||
List<String> designerJarItems = getJARList4Designer(); |
||||
for (DownloadItem downloadItem : downloadItems) { |
||||
String downloadItemName = downloadItem.getName(); |
||||
if (designerJarItems.contains(downloadItemName)) { |
||||
jarList.add(downloadItemName); |
||||
} |
||||
} |
||||
return jarList.toArray(new String[jarList.size()]); |
||||
} |
||||
|
||||
//获取更新日志缓存配置文件名
|
||||
private String getUpdateCacheConfig() { |
||||
return UpdateConstants.UPDATE_CACHE_CONFIG_X; |
||||
} |
||||
|
||||
//获取更新日志缓存内容文件名
|
||||
private String getUpdateCacheInfo() { |
||||
return UpdateConstants.UPDATE_CACHE_INFO_X; |
||||
} |
||||
|
||||
//获取最新的jar包时间字符串
|
||||
private String getLatestJARTimeStr() { |
||||
if (downloadFileConfig == null) { |
||||
return StringUtils.EMPTY; |
||||
} |
||||
String buildNO = downloadFileConfig.optString("buildNO"); |
||||
Date jarDate = (new SimpleDateFormat("yyyy.MM.dd")).parse(buildNO, new ParsePosition(buildNO.indexOf("-") + 1)); |
||||
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd"); |
||||
return df.format(jarDate); |
||||
} |
||||
|
||||
//判断是否是有效的日志内容
|
||||
private boolean isValidLogInfo(String logContent) { |
||||
String log = logContent.toUpperCase(); |
||||
List<String> logType = UpdateConstants.LOG_TYPE; |
||||
for (String s : logType) { |
||||
if (log.startsWith(s)) { |
||||
return true; |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
/** |
||||
* 显示窗口 |
||||
*/ |
||||
public void showDialog() { |
||||
setSize(DEFAULT); |
||||
setTitle(com.fr.design.i18n.Toolkit.i18nText("Fine-Design_Updater_UpdateAndUpgrade")); |
||||
GUICoreUtils.centerWindow(this); |
||||
setVisible(true); |
||||
} |
||||
|
||||
/** |
||||
* 检查有效性 |
||||
* |
||||
* @throws Exception |
||||
*/ |
||||
@Override |
||||
public void checkValid() throws Exception { |
||||
|
||||
} |
||||
} |
Loading…
Reference in new issue