hzzz
7 years ago
29 changed files with 1254 additions and 852 deletions
@ -0,0 +1,34 @@ |
|||||||
|
package com.fr.start; |
||||||
|
|
||||||
|
/** |
||||||
|
* 启动动画策略接口 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
* @date 2018/6/1 |
||||||
|
*/ |
||||||
|
public interface SplashStrategy { |
||||||
|
|
||||||
|
/** |
||||||
|
* 显示启动动画窗口 |
||||||
|
*/ |
||||||
|
void show(); |
||||||
|
|
||||||
|
/** |
||||||
|
* 隐藏启动动画窗口 |
||||||
|
*/ |
||||||
|
void hide(); |
||||||
|
|
||||||
|
/** |
||||||
|
* 设置模块加载信息 |
||||||
|
* |
||||||
|
* @param text 更新的文字 |
||||||
|
*/ |
||||||
|
void updateModuleLog(String text); |
||||||
|
|
||||||
|
/** |
||||||
|
* 设置感谢文字 |
||||||
|
* |
||||||
|
* @param text 更新的文字 |
||||||
|
*/ |
||||||
|
void updateThanksLog(String text); |
||||||
|
} |
@ -1,56 +0,0 @@ |
|||||||
package com.fr.start; |
|
||||||
|
|
||||||
import java.io.File; |
|
||||||
import javax.servlet.ServletException; |
|
||||||
import org.apache.catalina.Context; |
|
||||||
import org.apache.catalina.LifecycleException; |
|
||||||
import org.apache.catalina.core.AprLifecycleListener; |
|
||||||
import org.apache.catalina.core.StandardServer; |
|
||||||
import org.apache.catalina.startup.Tomcat; |
|
||||||
|
|
||||||
public class TomcatFRHost { |
|
||||||
private static Tomcat tomcat; |
|
||||||
|
|
||||||
public static Tomcat getTomcat() { |
|
||||||
return tomcat; |
|
||||||
} |
|
||||||
|
|
||||||
private static StandardServer server; |
|
||||||
|
|
||||||
private static AprLifecycleListener listener; |
|
||||||
|
|
||||||
public static void main(String[] args) throws Exception { |
|
||||||
tomcat = new Tomcat(); |
|
||||||
// 主机名,或ip
|
|
||||||
// tomcat.setHostname("localhost");
|
|
||||||
// 设置端口,80为默认端口
|
|
||||||
tomcat.setPort(8071); |
|
||||||
// tomcat用于存储自身的信息,可以随意指定,最好包含在项目目录下
|
|
||||||
tomcat.setBaseDir("."); |
|
||||||
// 建立server参照tomcat文件结构
|
|
||||||
server = (StandardServer) tomcat.getServer(); |
|
||||||
listener = new AprLifecycleListener(); |
|
||||||
server.addLifecycleListener(listener); |
|
||||||
// 将appBase设为本项目所在目录
|
|
||||||
//tomcat.getHost().setAppBase(".");
|
|
||||||
tomcat.getHost().setAppBase( |
|
||||||
System.getProperty("user.dir") + File.separator + "."); |
|
||||||
|
|
||||||
// 第二个参数对应docBase为web应用路径,目录下应有WEB-INF,WEB-INF下要有web.xml
|
|
||||||
// 启动tomcat
|
|
||||||
try { |
|
||||||
tomcat.start(); |
|
||||||
Context ct1 = tomcat.addWebapp("/WebReport", "/Users/momeak/Documents/Working/develop/others/tomcatsrc/WebReport"); |
|
||||||
} catch (LifecycleException e) { |
|
||||||
e.printStackTrace(); |
|
||||||
} catch (ServletException e) { |
|
||||||
e.printStackTrace(); |
|
||||||
} |
|
||||||
// Context ct1 = tomcat.addWebapp("/examples", "/Users/momeak/Documents/Working/develop/others/tomcatsrc/examples");
|
|
||||||
|
|
||||||
// Context ct = tomcat.addWebapp("", "/Users/momeak/Documents/Working/develop/others/tomcatsrc/webapps/ROOT");
|
|
||||||
|
|
||||||
// tomcat.getServer().await();
|
|
||||||
System.out.println("启动成功"); |
|
||||||
} |
|
||||||
} |
|
@ -0,0 +1,15 @@ |
|||||||
|
package com.fr.start.server; |
||||||
|
|
||||||
|
import com.fr.event.Event; |
||||||
|
import com.fr.event.Null; |
||||||
|
|
||||||
|
/** |
||||||
|
* Created by juhaoyu on 2018/6/5. |
||||||
|
* 内置服务器事件 |
||||||
|
*/ |
||||||
|
public enum EmbedServerEvent implements Event<Null> { |
||||||
|
BeforeStart, |
||||||
|
AfterStart, |
||||||
|
BeforeStop, |
||||||
|
AfterStop |
||||||
|
} |
@ -1,133 +0,0 @@ |
|||||||
package com.fr.start.server; |
|
||||||
import java.lang.reflect.InvocationTargetException; |
|
||||||
import java.util.HashMap; |
|
||||||
import java.util.Map; |
|
||||||
import java.util.logging.Level; |
|
||||||
import java.util.logging.Logger; |
|
||||||
import javax.servlet.ServletException; |
|
||||||
import org.apache.catalina.Context; |
|
||||||
import org.apache.catalina.Host; |
|
||||||
import org.apache.catalina.core.ContainerBase; |
|
||||||
import org.apache.catalina.core.StandardContext; |
|
||||||
import org.apache.catalina.core.StandardHost; |
|
||||||
import org.apache.catalina.startup.ContextConfig; |
|
||||||
import org.apache.catalina.startup.Tomcat; |
|
||||||
|
|
||||||
public class FRTomcat extends Tomcat{ |
|
||||||
|
|
||||||
private final Map<String, Logger> frpinnedLoggers = new HashMap<String, Logger>(); |
|
||||||
private boolean frsilent = false; |
|
||||||
|
|
||||||
|
|
||||||
public Context addWebapp(String contextPath, String docBase) throws ServletException { |
|
||||||
silence(host, contextPath); |
|
||||||
|
|
||||||
Context ctx = createContext(host, contextPath); |
|
||||||
if (ctx instanceof StandardContext) { |
|
||||||
((StandardContext)ctx).setDelegate(true); |
|
||||||
} |
|
||||||
ctx.setPath(contextPath); |
|
||||||
ctx.setDocBase(docBase); |
|
||||||
|
|
||||||
ctx.addLifecycleListener(new DefaultWebXmlListener()); |
|
||||||
ctx.setConfigFile(getWebappConfigFile(docBase, contextPath)); |
|
||||||
|
|
||||||
ContextConfig ctxCfg = new ContextConfig(); |
|
||||||
ctx.addLifecycleListener(ctxCfg); |
|
||||||
|
|
||||||
ctxCfg.setDefaultWebXml(noDefaultWebXmlPath()); |
|
||||||
|
|
||||||
if (host == null) { |
|
||||||
getHost().addChild(ctx); |
|
||||||
} else { |
|
||||||
host.addChild(ctx); |
|
||||||
} |
|
||||||
|
|
||||||
return ctx; |
|
||||||
} |
|
||||||
|
|
||||||
private void silence(Host host, String contextPath) { |
|
||||||
String loggerName = getLoggerName(host, contextPath); |
|
||||||
Logger logger = Logger.getLogger(loggerName); |
|
||||||
frpinnedLoggers.put(loggerName, logger); |
|
||||||
if (frsilent) { |
|
||||||
logger.setLevel(Level.WARNING); |
|
||||||
} else { |
|
||||||
logger.setLevel(Level.INFO); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private String getLoggerName(Host host, String contextName) { |
|
||||||
if (host == null) { |
|
||||||
host = getHost(); |
|
||||||
} |
|
||||||
StringBuilder loggerName = new StringBuilder(); |
|
||||||
loggerName.append(ContainerBase.class.getName()); |
|
||||||
loggerName.append(".["); |
|
||||||
// Engine name
|
|
||||||
loggerName.append(host.getParent().getName()); |
|
||||||
loggerName.append("].["); |
|
||||||
// Host name
|
|
||||||
loggerName.append(host.getName()); |
|
||||||
loggerName.append("].["); |
|
||||||
// Context name
|
|
||||||
if (contextName == null || contextName.equals("")) { |
|
||||||
loggerName.append("/"); |
|
||||||
} else if (contextName.startsWith("##")) { |
|
||||||
loggerName.append("/"); |
|
||||||
loggerName.append(contextName); |
|
||||||
} |
|
||||||
loggerName.append(']'); |
|
||||||
|
|
||||||
return loggerName.toString(); |
|
||||||
} |
|
||||||
|
|
||||||
private Context createContext(Host host, String url) { |
|
||||||
String contextClass = StandardContext.class.getName(); |
|
||||||
if (host == null) { |
|
||||||
host = this.getHost(); |
|
||||||
} |
|
||||||
if (host instanceof StandardHost) { |
|
||||||
contextClass = ((StandardHost) host).getContextClass(); |
|
||||||
} |
|
||||||
try { |
|
||||||
return (Context) Class.forName(contextClass).getConstructor() |
|
||||||
.newInstance(); |
|
||||||
} catch (InstantiationException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (IllegalAccessException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (IllegalArgumentException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (InvocationTargetException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (NoSuchMethodException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (SecurityException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} catch (ClassNotFoundException e) { |
|
||||||
throw new IllegalArgumentException( |
|
||||||
"Can't instantiate context-class " + contextClass |
|
||||||
+ " for host " + host + " and url " |
|
||||||
+ url, e); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
@ -0,0 +1,119 @@ |
|||||||
|
package com.fr.start.server; |
||||||
|
|
||||||
|
import com.fr.base.FRContext; |
||||||
|
import com.fr.design.DesignerEnvManager; |
||||||
|
import com.fr.event.EventDispatcher; |
||||||
|
import com.fr.log.FineLoggerFactory; |
||||||
|
import com.fr.module.ModuleRole; |
||||||
|
import com.fr.stable.lifecycle.AbstractLifecycle; |
||||||
|
import com.fr.startup.FineWebApplicationInitializer; |
||||||
|
import com.fr.third.springframework.web.SpringServletContainerInitializer; |
||||||
|
import com.fr.third.springframework.web.context.support.AnnotationConfigWebApplicationContext; |
||||||
|
import org.apache.catalina.Context; |
||||||
|
import org.apache.catalina.LifecycleException; |
||||||
|
import org.apache.catalina.loader.VirtualWebappLoader; |
||||||
|
import org.apache.catalina.startup.Tomcat; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.HashSet; |
||||||
|
import java.util.Set; |
||||||
|
|
||||||
|
/** |
||||||
|
* Created by juhaoyu on 2018/6/5. |
||||||
|
*/ |
||||||
|
public class FineEmbedServer extends AbstractLifecycle { |
||||||
|
|
||||||
|
private static final FineEmbedServer INSTANCE = new FineEmbedServer(); |
||||||
|
|
||||||
|
private Tomcat tomcat; |
||||||
|
|
||||||
|
public static FineEmbedServer getInstance() { |
||||||
|
|
||||||
|
return INSTANCE; |
||||||
|
} |
||||||
|
|
||||||
|
private FineEmbedServer() {} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected synchronized void executeStart() { |
||||||
|
|
||||||
|
EventDispatcher.fire(EmbedServerEvent.BeforeStart); |
||||||
|
try { |
||||||
|
//初始化tomcat
|
||||||
|
initTomcat(); |
||||||
|
tomcat.start(); |
||||||
|
} catch (LifecycleException e) { |
||||||
|
FineLoggerFactory.getLogger().error(e.getMessage(), e); |
||||||
|
} |
||||||
|
EventDispatcher.fire(EmbedServerEvent.AfterStart); |
||||||
|
} |
||||||
|
|
||||||
|
private void initTomcat() { |
||||||
|
|
||||||
|
tomcat = new Tomcat(); |
||||||
|
|
||||||
|
tomcat.setPort(DesignerEnvManager.getEnvManager().getEmbedServerPort()); |
||||||
|
String docBase = new File(FRContext.getCurrentEnv().getPath()).getParent(); |
||||||
|
String appName = "/" + FRContext.getCurrentEnv().getAppName(); |
||||||
|
Context context = tomcat.addContext(appName, docBase); |
||||||
|
tomcat.addServlet(appName, "default", "org.apache.catalina.servlets.DefaultServlet"); |
||||||
|
//覆盖tomcat的WebAppClassLoader
|
||||||
|
context.setLoader(new FRTomcatLoader()); |
||||||
|
|
||||||
|
//直接指定initializer,tomcat就不用再扫描一遍了
|
||||||
|
SpringServletContainerInitializer initializer = new SpringServletContainerInitializer(); |
||||||
|
Set<Class<?>> classes = new HashSet<Class<?>>(); |
||||||
|
classes.add(FineWebApplicationInitializer.class); |
||||||
|
context.addServletContainerInitializer(initializer, classes); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
protected synchronized void executeStop() { |
||||||
|
|
||||||
|
EventDispatcher.fire(EmbedServerEvent.BeforeStop); |
||||||
|
try { |
||||||
|
stopSpring(); |
||||||
|
stopServerActivator(); |
||||||
|
stopTomcat(); |
||||||
|
} catch (LifecycleException e) { |
||||||
|
FineLoggerFactory.getLogger().error(e.getMessage(), e); |
||||||
|
} |
||||||
|
EventDispatcher.fire(EmbedServerEvent.AfterStop); |
||||||
|
} |
||||||
|
|
||||||
|
private void stopServerActivator() { |
||||||
|
|
||||||
|
ModuleRole.ServerRoot.stop(); |
||||||
|
} |
||||||
|
|
||||||
|
private void stopSpring() { |
||||||
|
|
||||||
|
AnnotationConfigWebApplicationContext context = ModuleRole.ServerRoot.getSingleton(AnnotationConfigWebApplicationContext.class); |
||||||
|
if (context != null) { |
||||||
|
context.stop(); |
||||||
|
context.destroy(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private void stopTomcat() throws LifecycleException { |
||||||
|
|
||||||
|
tomcat.stop(); |
||||||
|
tomcat.destroy(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Created by juhaoyu on 2018/6/5. |
||||||
|
* 自定义的tomcat loader,主要用于防止内置服务器再加载一遍class |
||||||
|
*/ |
||||||
|
private static class FRTomcatLoader extends VirtualWebappLoader { |
||||||
|
|
||||||
|
@Override |
||||||
|
public ClassLoader getClassLoader() { |
||||||
|
|
||||||
|
return this.getClass().getClassLoader(); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -1,62 +0,0 @@ |
|||||||
package com.fr.start.server; |
|
||||||
|
|
||||||
import java.io.IOException; |
|
||||||
import java.io.OutputStream; |
|
||||||
import java.util.ArrayList; |
|
||||||
import java.util.List; |
|
||||||
|
|
||||||
/** |
|
||||||
* p: 这是为了将一个OutputStream输出多个OutputStream |
|
||||||
*/ |
|
||||||
public class MultiOutputStream extends OutputStream { |
|
||||||
private List outList = new ArrayList(); |
|
||||||
|
|
||||||
public MultiOutputStream() { |
|
||||||
} |
|
||||||
|
|
||||||
public void addOutputStream(OutputStream output) { |
|
||||||
this.outList.add(output); |
|
||||||
} |
|
||||||
|
|
||||||
public void removeOutputStream(OutputStream output) { |
|
||||||
this.outList.remove(output); |
|
||||||
} |
|
||||||
|
|
||||||
public int getOutputStreamCount() { |
|
||||||
return this.outList.size(); |
|
||||||
} |
|
||||||
|
|
||||||
public OutputStream getOutputStream(int index) { |
|
||||||
return (OutputStream) this.outList.get(index); |
|
||||||
} |
|
||||||
|
|
||||||
public void write(int b) throws IOException { |
|
||||||
for(int i = 0; i < outList.size(); i++) { |
|
||||||
((OutputStream)outList.get(i)).write(b); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public void write(byte buff[]) throws IOException { |
|
||||||
for(int i = 0; i < outList.size(); i++) { |
|
||||||
((OutputStream)outList.get(i)).write(buff); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public void write(byte buff[], int off, int len) throws IOException { |
|
||||||
for(int i = 0; i < outList.size(); i++) { |
|
||||||
((OutputStream)outList.get(i)).write(buff, off, len); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public void flush() throws IOException { |
|
||||||
for(int i = 0; i < outList.size(); i++) { |
|
||||||
((OutputStream)outList.get(i)).flush(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public void close() throws IOException { |
|
||||||
for(int i = 0; i < outList.size(); i++) { |
|
||||||
((OutputStream)outList.get(i)).close(); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
@ -1,280 +0,0 @@ |
|||||||
package com.fr.start.server; |
|
||||||
|
|
||||||
import com.fr.module.ModuleContext; |
|
||||||
import java.awt.SystemTray; |
|
||||||
import java.io.File; |
|
||||||
import java.io.FileOutputStream; |
|
||||||
import java.io.IOException; |
|
||||||
import java.io.PrintStream; |
|
||||||
import java.lang.reflect.Field; |
|
||||||
import java.text.DateFormat; |
|
||||||
import java.text.SimpleDateFormat; |
|
||||||
import java.util.ArrayList; |
|
||||||
import java.util.Calendar; |
|
||||||
import java.util.HashMap; |
|
||||||
import java.util.List; |
|
||||||
import java.util.Map; |
|
||||||
|
|
||||||
import com.fr.general.GeneralContext; |
|
||||||
import com.fr.stable.ProductConstants; |
|
||||||
import org.apache.catalina.Context; |
|
||||||
import org.apache.catalina.LifecycleException; |
|
||||||
import org.apache.catalina.Server; |
|
||||||
import org.apache.catalina.core.AprLifecycleListener; |
|
||||||
import org.apache.catalina.core.StandardServer; |
|
||||||
|
|
||||||
import com.fr.base.Env; |
|
||||||
import com.fr.base.FRContext; |
|
||||||
import com.fr.dav.LocalEnv; |
|
||||||
import com.fr.design.DesignerEnvManager; |
|
||||||
import com.fr.general.Inter; |
|
||||||
import com.fr.stable.StableUtils; |
|
||||||
import com.fr.stable.project.ProjectConstants; |
|
||||||
import com.fr.start.StartServer; |
|
||||||
|
|
||||||
public class TomcatHost { |
|
||||||
|
|
||||||
private static FRTomcat tomcat; |
|
||||||
private StandardServer server; |
|
||||||
private AprLifecycleListener listener; |
|
||||||
|
|
||||||
// private Server server;
|
|
||||||
private MultiOutputStream multiOutputStream = null; |
|
||||||
private File outLogFile = null; |
|
||||||
private int currentPort = -1; |
|
||||||
// 内置服务器一个端口下面可以有多个应用,但是content不能重名
|
|
||||||
private Map<String, Context> webAppsMap = new HashMap<String, Context>(); |
|
||||||
private List<TomcatServerListener> listenerList = new ArrayList<TomcatServerListener>(); |
|
||||||
private boolean isDemoAppLoaded = false; |
|
||||||
|
|
||||||
public TomcatHost(int port) { |
|
||||||
this.currentPort = port; |
|
||||||
initServer(); |
|
||||||
|
|
||||||
initLogFileAndOutputStream(); |
|
||||||
|
|
||||||
// TODO: 将HostTomcatServer放到ServerTray中去
|
|
||||||
tryStartServerTray(); |
|
||||||
} |
|
||||||
|
|
||||||
public static FRTomcat getTomcat() { |
|
||||||
return tomcat; |
|
||||||
} |
|
||||||
|
|
||||||
private void initServer() { |
|
||||||
try { |
|
||||||
//直接用自定义的,不用server.xml
|
|
||||||
this.tomcat = new FRTomcat(); |
|
||||||
this.tomcat.setPort(this.currentPort); |
|
||||||
this.tomcat.setBaseDir(StableUtils.getInstallHome()); |
|
||||||
this.server = (StandardServer) tomcat.getServer(); |
|
||||||
this.listener = new AprLifecycleListener(); |
|
||||||
this.server.addLifecycleListener(listener); |
|
||||||
this.tomcat.getHost().setAppBase(StableUtils.getInstallHome() + File.separator + "."); |
|
||||||
this.tomcat.getConnector().setURIEncoding("UTF-8"); |
|
||||||
} catch (Exception e) { |
|
||||||
//todo 最好加一个用server.xml
|
|
||||||
FRContext.getLogger().error(e.getMessage(), e); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private void initLogFileAndOutputStream() { |
|
||||||
// log文件放置的位置
|
|
||||||
File logDir = null; |
|
||||||
String installHome = StableUtils.getInstallHome(); |
|
||||||
if (installHome == null) {// 没有installHome的时候,就放到user.home下面喽
|
|
||||||
logDir = new File(ProductConstants.getEnvHome() + File.separator + ProjectConstants.LOGS_NAME); |
|
||||||
} else { |
|
||||||
// james:logs放在安装目录下面
|
|
||||||
logDir = new File(installHome + File.separator + ProjectConstants.LOGS_NAME + File.separator + "tomcat"); |
|
||||||
} |
|
||||||
StableUtils.mkdirs(logDir); |
|
||||||
DateFormat fateFormat = new SimpleDateFormat("yyyy-MM-dd"); |
|
||||||
Calendar curCalendar = Calendar.getInstance(); |
|
||||||
outLogFile = new File(logDir, "tomcat_" + fateFormat.format(curCalendar.getTime()) + ".log"); |
|
||||||
|
|
||||||
try { |
|
||||||
multiOutputStream = new MultiOutputStream(); |
|
||||||
multiOutputStream.addOutputStream(new FileOutputStream(outLogFile, true)); |
|
||||||
multiOutputStream.addOutputStream(System.out); |
|
||||||
System.setErr(new PrintStream(multiOutputStream)); |
|
||||||
System.setOut(new PrintStream(multiOutputStream)); |
|
||||||
} catch (IOException ioe) { |
|
||||||
FRContext.getLogger().error(ioe.getMessage(), ioe); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private synchronized void addWebApplication(String context, String webappsPath) { |
|
||||||
FRContext.getLogger().info("The new Application Path is: \n" + webappsPath + ", it will be added."); |
|
||||||
if (webAppsMap.get(context) != null) { |
|
||||||
Context webapp = webAppsMap.remove(context); |
|
||||||
} |
|
||||||
try { |
|
||||||
if (!isStarted()) { |
|
||||||
start(); |
|
||||||
} |
|
||||||
Context webapp = tomcat.addWebapp(context, webappsPath); |
|
||||||
webAppsMap.put(context, webapp); |
|
||||||
} catch (Exception e) { |
|
||||||
FRContext.getLogger().error(e.getMessage(), e); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private void addAndStartWebApplication(String context, String webAppPath) { |
|
||||||
addWebApplication(context, webAppPath); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Get MultiOutputStream. |
|
||||||
*/ |
|
||||||
public MultiOutputStream getMultiOutputStream() { |
|
||||||
return this.multiOutputStream; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Get out log file |
|
||||||
*/ |
|
||||||
public File getOutLogFile() { |
|
||||||
return this.outLogFile; |
|
||||||
} |
|
||||||
|
|
||||||
private Server getServer() { |
|
||||||
if (server == null) { |
|
||||||
initServer(); |
|
||||||
} |
|
||||||
|
|
||||||
return server; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Start |
|
||||||
* |
|
||||||
* @throws Exception |
|
||||||
*/ |
|
||||||
public void start() throws Exception { |
|
||||||
tomcat.start(); |
|
||||||
for (int i = 0; i < listenerList.size(); i++) { |
|
||||||
TomcatServerListener listener = TomcatHost.this.getLinstener(i); |
|
||||||
listener.started(this); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Stop |
|
||||||
* |
|
||||||
* @throws Exception |
|
||||||
*/ |
|
||||||
public void stop() throws Exception { |
|
||||||
|
|
||||||
tomcat.stop(); |
|
||||||
|
|
||||||
for (int i = 0; i < listenerList.size(); i++) { |
|
||||||
TomcatServerListener listener = this.getLinstener(i); |
|
||||||
listener.stopped(this); |
|
||||||
} |
|
||||||
|
|
||||||
StartServer.currentEnvChanged(); |
|
||||||
server = null;//重置server
|
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Is started |
|
||||||
* |
|
||||||
* @throws Exception |
|
||||||
*/ |
|
||||||
public boolean isStarted() throws Exception { |
|
||||||
return getServer().getState().isAvailable(); |
|
||||||
} |
|
||||||
|
|
||||||
public void addListener(TomcatServerListener listener) { |
|
||||||
this.listenerList.add(listener); |
|
||||||
} |
|
||||||
|
|
||||||
public int getLinstenerCount() { |
|
||||||
return this.listenerList.size(); |
|
||||||
} |
|
||||||
|
|
||||||
public TomcatServerListener getLinstener(int index) { |
|
||||||
if (index < 0 || index >= this.getLinstenerCount()) { |
|
||||||
return null; |
|
||||||
} |
|
||||||
|
|
||||||
return this.listenerList.get(index); |
|
||||||
} |
|
||||||
|
|
||||||
public void clearLinsteners() { |
|
||||||
this.listenerList.clear(); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* 尝试启动系统托盘 |
|
||||||
*/ |
|
||||||
private void tryStartServerTray() { |
|
||||||
if (SystemTray.isSupported()) { |
|
||||||
new ServerTray(this); |
|
||||||
} else { |
|
||||||
FRContext.getLogger().error("Do not support the SystemTray!"); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public void exit() { |
|
||||||
try { |
|
||||||
getServer().stop(); |
|
||||||
} catch (LifecycleException e) { |
|
||||||
FRContext.getLogger().error(e.getMessage(), e); |
|
||||||
} |
|
||||||
|
|
||||||
for (int i = 0; i < listenerList.size(); i++) { |
|
||||||
TomcatServerListener listener = this.getLinstener(i); |
|
||||||
listener.exited(this); |
|
||||||
} |
|
||||||
|
|
||||||
try { |
|
||||||
getServer().destroy(); |
|
||||||
} catch (LifecycleException e) { |
|
||||||
FRContext.getLogger().error(e.getMessage(), e); |
|
||||||
} |
|
||||||
StartServer.currentEnvChanged(); |
|
||||||
|
|
||||||
} |
|
||||||
|
|
||||||
public int getCurrentPort() { |
|
||||||
return currentPort; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* 安装目录下的默认的WebReport,这个只执行一次,除了预览demo,其他的不要调用这个方法 |
|
||||||
*/ |
|
||||||
public void addAndStartInstallHomeWebApp() { |
|
||||||
if (!isDemoAppLoaded) { |
|
||||||
String installHome = StableUtils.getInstallHome(); |
|
||||||
String webApplication = StableUtils.pathJoin(new String[]{installHome, ProjectConstants.WEBAPP_NAME}); |
|
||||||
|
|
||||||
if (new File(webApplication).isDirectory()) { |
|
||||||
addAndStartWebApplication("/" + ProjectConstants.WEBAPP_NAME, webApplication); |
|
||||||
} |
|
||||||
} |
|
||||||
isDemoAppLoaded = true; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* 加载Env下的报表运行环境 |
|
||||||
*/ |
|
||||||
public void addAndStartLocalEnvHomeWebApp() { |
|
||||||
String name = DesignerEnvManager.getEnvManager().getCurEnvName(); |
|
||||||
if (name.equals(Inter.getLocText("Default"))) { |
|
||||||
isDemoAppLoaded = true; |
|
||||||
} |
|
||||||
Env env = FRContext.getCurrentEnv(); |
|
||||||
if (env instanceof LocalEnv) { |
|
||||||
String webApplication = new File(env.getPath()).getParent(); |
|
||||||
FRContext.getLogger().info(Inter.getLocText("INFO-Reset_Webapp") + ":" + webApplication); |
|
||||||
addAndStartWebApplication("/" + GeneralContext.getCurrentAppNameOfEnv(), webApplication); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public boolean isDemoAppLoaded() { |
|
||||||
return isDemoAppLoaded; |
|
||||||
} |
|
||||||
|
|
||||||
} |
|
@ -1,18 +0,0 @@ |
|||||||
package com.fr.start.server; |
|
||||||
|
|
||||||
public interface TomcatServerListener { |
|
||||||
/** |
|
||||||
* Started |
|
||||||
*/ |
|
||||||
public void started(TomcatHost tomcatServer); |
|
||||||
|
|
||||||
/** |
|
||||||
* Stopped |
|
||||||
*/ |
|
||||||
public void stopped(TomcatHost tomcatServer); |
|
||||||
/** |
|
||||||
* Exited |
|
||||||
*/ |
|
||||||
public void exited(TomcatHost tomcatServer); |
|
||||||
|
|
||||||
} |
|
@ -0,0 +1,142 @@ |
|||||||
|
package com.fr.start; |
||||||
|
|
||||||
|
import com.fr.base.FRContext; |
||||||
|
import com.fr.design.mainframe.bbs.BBSConstants; |
||||||
|
import com.fr.general.Inter; |
||||||
|
import com.fr.stable.StringUtils; |
||||||
|
import com.fr.stable.module.ModuleAdapter; |
||||||
|
import com.fr.stable.module.ModuleListener; |
||||||
|
|
||||||
|
import java.util.Locale; |
||||||
|
import java.util.Random; |
||||||
|
import java.util.concurrent.Executors; |
||||||
|
import java.util.concurrent.ScheduledExecutorService; |
||||||
|
import java.util.concurrent.TimeUnit; |
||||||
|
|
||||||
|
/** |
||||||
|
* 启动动画策略 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
* @date 2018/6/5 |
||||||
|
*/ |
||||||
|
public class SplashContext { |
||||||
|
|
||||||
|
private static final SplashContext SPLASH_CONTEXT = new SplashContext(); |
||||||
|
|
||||||
|
private SplashStrategy splashStrategy; |
||||||
|
|
||||||
|
private String moduleID = ""; |
||||||
|
private int loadingIndex = 0; |
||||||
|
private String[] loading = new String[]{"..", "....", "......"}; |
||||||
|
|
||||||
|
private static final String GUEST = getRandomUser(); |
||||||
|
|
||||||
|
private ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1); |
||||||
|
|
||||||
|
private ModuleListener listener; |
||||||
|
|
||||||
|
|
||||||
|
public static SplashContext getInstance() { |
||||||
|
return SPLASH_CONTEXT; |
||||||
|
} |
||||||
|
|
||||||
|
private SplashContext() { |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 注册具体的启动动画 |
||||||
|
*/ |
||||||
|
public void registerSplash(SplashStrategy splashStrategy) { |
||||||
|
this.splashStrategy = splashStrategy; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 注册监听 |
||||||
|
*/ |
||||||
|
public ModuleListener getModuleListener() { |
||||||
|
initListener(); |
||||||
|
return listener; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 展示启动动画 |
||||||
|
*/ |
||||||
|
public void show() { |
||||||
|
splashStrategy.show(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 隐藏启动动画 |
||||||
|
*/ |
||||||
|
public void hide() { |
||||||
|
splashStrategy.hide(); |
||||||
|
// 窗口关闭后取消定时获取模块信息的timer
|
||||||
|
scheduler.shutdown(); |
||||||
|
// 一次性
|
||||||
|
splashStrategy = null; |
||||||
|
} |
||||||
|
|
||||||
|
private void initListener() { |
||||||
|
scheduler.scheduleAtFixedRate(new Runnable() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
loadingIndex++; |
||||||
|
updateModuleLog(moduleID.isEmpty() ? StringUtils.EMPTY : moduleID + loading[loadingIndex % 3]); |
||||||
|
} |
||||||
|
}, 0, 300, TimeUnit.MILLISECONDS); |
||||||
|
|
||||||
|
listener = new ModuleAdapter() { |
||||||
|
@Override |
||||||
|
public void onStartBefore(String moduleName, String moduleI18nName) { |
||||||
|
moduleID = moduleI18nName; |
||||||
|
loadingIndex++; |
||||||
|
updateModuleLog(moduleID.isEmpty() ? StringUtils.EMPTY : moduleID + loading[loadingIndex % 3]); |
||||||
|
|
||||||
|
} |
||||||
|
}; |
||||||
|
showThanks(); |
||||||
|
} |
||||||
|
|
||||||
|
private void updateModuleLog(String text) { |
||||||
|
splashStrategy.updateModuleLog(text); |
||||||
|
} |
||||||
|
|
||||||
|
private void updateThanksLog(String text) { |
||||||
|
splashStrategy.updateThanksLog(text); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取随机感谢人员 |
||||||
|
*/ |
||||||
|
private static String getRandomUser() { |
||||||
|
String[] allGuest = BBSConstants.getAllGuest(); |
||||||
|
if (allGuest.length == 0) { |
||||||
|
return StringUtils.EMPTY; |
||||||
|
} |
||||||
|
int num = new Random().nextInt(allGuest.length); |
||||||
|
return StringUtils.BLANK + allGuest[num]; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 展示感谢信息 |
||||||
|
*/ |
||||||
|
private void showThanks() { |
||||||
|
if (shouldShowThanks()) { |
||||||
|
updateThanksLog(Inter.getLocText("FR-Designer_Thanks-To") + GUEST); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 是否显示鸣谢面板 |
||||||
|
*/ |
||||||
|
private boolean shouldShowThanks() { |
||||||
|
Locale[] hideLocales = {Locale.CHINA, Locale.TAIWAN}; |
||||||
|
for (Locale loc : hideLocales) { |
||||||
|
if (FRContext.getLocale().equals(loc)) { |
||||||
|
return true; |
||||||
|
} |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,210 @@ |
|||||||
|
package com.fr.start.fx; |
||||||
|
|
||||||
|
import com.sun.imageio.plugins.gif.GIFImageReader; |
||||||
|
import com.sun.imageio.plugins.gif.GIFImageReaderSpi; |
||||||
|
import com.sun.javafx.tk.ImageLoader; |
||||||
|
import com.sun.javafx.tk.PlatformImage; |
||||||
|
import javafx.animation.KeyFrame; |
||||||
|
import javafx.animation.Timeline; |
||||||
|
import javafx.event.Event; |
||||||
|
import javafx.event.EventHandler; |
||||||
|
import javafx.scene.image.WritableImage; |
||||||
|
import javafx.util.Duration; |
||||||
|
|
||||||
|
import javax.imageio.stream.FileImageInputStream; |
||||||
|
import java.io.File; |
||||||
|
import java.lang.ref.WeakReference; |
||||||
|
import java.lang.reflect.InvocationTargetException; |
||||||
|
import java.lang.reflect.Method; |
||||||
|
import java.net.MalformedURLException; |
||||||
|
import java.net.URI; |
||||||
|
import java.net.URL; |
||||||
|
import java.util.regex.Pattern; |
||||||
|
|
||||||
|
/** |
||||||
|
* 边加载边播放的gif加载器 |
||||||
|
* |
||||||
|
* @author daniel |
||||||
|
*/ |
||||||
|
public class FastGifImage extends WritableImage { |
||||||
|
private String url; |
||||||
|
private int gifCount; |
||||||
|
|
||||||
|
public FastGifImage(String url, int w, int h) { |
||||||
|
super(w, h); |
||||||
|
this.url = validateUrl(url); |
||||||
|
seekCount(); |
||||||
|
initialize(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 给出gif帧数,加快加载速度 |
||||||
|
* |
||||||
|
* @param url gif url |
||||||
|
* @param gifCount gif帧数 |
||||||
|
* @param w 宽 |
||||||
|
* @param h 高 |
||||||
|
*/ |
||||||
|
public FastGifImage(String url, int gifCount, int w, int h) { |
||||||
|
super(w, h); |
||||||
|
this.url = validateUrl(url); |
||||||
|
this.gifCount = gifCount; |
||||||
|
initialize(); |
||||||
|
} |
||||||
|
|
||||||
|
private void seekCount() { |
||||||
|
try { |
||||||
|
GIFImageReaderSpi spi = new GIFImageReaderSpi(); |
||||||
|
GIFImageReader gifReader = (GIFImageReader) spi.createReaderInstance(); |
||||||
|
gifReader.setInput(new FileImageInputStream(new File(new URI(url)))); |
||||||
|
gifCount = gifReader.getNumImages(true); |
||||||
|
} catch (Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private static final Pattern URL_QUICKMATCH = Pattern.compile("^\\p{Alpha}[\\p{Alnum}+.-]*:.*$"); |
||||||
|
|
||||||
|
private static String validateUrl(final String url) { |
||||||
|
if (url == null) { |
||||||
|
throw new NullPointerException("URL must not be null"); |
||||||
|
} |
||||||
|
|
||||||
|
if (url.trim().isEmpty()) { |
||||||
|
throw new IllegalArgumentException("URL must not be empty"); |
||||||
|
} |
||||||
|
|
||||||
|
try { |
||||||
|
if (!URL_QUICKMATCH.matcher(url).matches()) { |
||||||
|
final ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader(); |
||||||
|
URL resource; |
||||||
|
if (url.charAt(0) == '/') { |
||||||
|
resource = contextClassLoader.getResource(url.substring(1)); |
||||||
|
} else { |
||||||
|
resource = contextClassLoader.getResource(url); |
||||||
|
} |
||||||
|
if (resource == null) { |
||||||
|
throw new IllegalArgumentException("Invalid URL or resource not found"); |
||||||
|
} |
||||||
|
return resource.toString(); |
||||||
|
} |
||||||
|
// Use URL constructor for validation
|
||||||
|
return new URL(url).toString(); |
||||||
|
} catch (final IllegalArgumentException e) { |
||||||
|
throw new IllegalArgumentException("Invalid URL" + e.getMessage()); |
||||||
|
} catch (final MalformedURLException e) { |
||||||
|
throw new IllegalArgumentException("Invalid URL" + e.getMessage()); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void finishImage(Exception e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
|
||||||
|
private void finishImage(ImageLoader loader) { |
||||||
|
final Exception loadingException = loader.getException(); |
||||||
|
if (loadingException != null) { |
||||||
|
finishImage(loadingException); |
||||||
|
return; |
||||||
|
} |
||||||
|
initializeAnimatedImage(loader); |
||||||
|
} |
||||||
|
|
||||||
|
// Generates the animation Timeline for multiframe images.
|
||||||
|
private void initializeAnimatedImage(ImageLoader loader) { |
||||||
|
|
||||||
|
animation = new Animation(this, loader); |
||||||
|
animation.start(); |
||||||
|
} |
||||||
|
|
||||||
|
// Support for animated images.
|
||||||
|
private Animation animation; |
||||||
|
|
||||||
|
private static final class Animation { |
||||||
|
final WeakReference<FastGifImage> imageRef; |
||||||
|
final Timeline timeline; |
||||||
|
final ImageLoader loader; |
||||||
|
|
||||||
|
public Animation(final FastGifImage image, final ImageLoader loader) { |
||||||
|
this.loader = loader; |
||||||
|
imageRef = new WeakReference<FastGifImage>(image); |
||||||
|
timeline = new Timeline(); |
||||||
|
timeline.setCycleCount(Timeline.INDEFINITE); |
||||||
|
|
||||||
|
final int frameCount = loader.getFrameCount(); |
||||||
|
int duration = 0; |
||||||
|
|
||||||
|
for (int i = 0; i < frameCount; ++i) { |
||||||
|
addKeyFrame(i, duration); |
||||||
|
duration = duration + loader.getFrameDelay(i); |
||||||
|
} |
||||||
|
|
||||||
|
// Note: we need one extra frame in the timeline to define how long
|
||||||
|
// the last frame is shown, the wrap around is "instantaneous"
|
||||||
|
addKeyFrame(0, duration); |
||||||
|
} |
||||||
|
|
||||||
|
public void start() { |
||||||
|
timeline.play(); |
||||||
|
} |
||||||
|
|
||||||
|
public void stop() { |
||||||
|
timeline.stop(); |
||||||
|
} |
||||||
|
|
||||||
|
private void updateImage(final int frameIndex) { |
||||||
|
final FastGifImage image = imageRef.get(); |
||||||
|
if (image != null) { |
||||||
|
image.setPlatformImagePropertyImpl( |
||||||
|
loader.getFrame(frameIndex)); |
||||||
|
} else { |
||||||
|
timeline.stop(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private void addKeyFrame(final int index, final double duration) { |
||||||
|
timeline.getKeyFrames().add( |
||||||
|
new KeyFrame(Duration.millis(duration), |
||||||
|
new EventHandler() { |
||||||
|
@Override |
||||||
|
public void handle(Event event) { |
||||||
|
updateImage(index); |
||||||
|
} |
||||||
|
} |
||||||
|
)); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private static Method method; |
||||||
|
|
||||||
|
static { |
||||||
|
try { |
||||||
|
method = FastGifImage.class.getSuperclass().getSuperclass().getDeclaredMethod("platformImagePropertyImpl"); |
||||||
|
method.setAccessible(true); |
||||||
|
} catch (Exception e) { |
||||||
|
|
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private void setPlatformImagePropertyImpl(PlatformImage image) { |
||||||
|
try { |
||||||
|
Object o = method.invoke(this); |
||||||
|
Method method = o.getClass().getDeclaredMethod("set", Object.class); |
||||||
|
method.setAccessible(true); |
||||||
|
method.invoke(o, image); |
||||||
|
} catch (IllegalAccessException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} catch (InvocationTargetException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} catch (NoSuchMethodException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void initialize() { |
||||||
|
finishImage(new PrismImageLoader2(url, gifCount, (int) getRequestedWidth(), (int) getRequestedHeight(), isPreserveRatio(), isSmooth())); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,211 @@ |
|||||||
|
package com.fr.start.fx; |
||||||
|
|
||||||
|
import com.sun.javafx.iio.ImageFrame; |
||||||
|
import com.sun.javafx.iio.ImageLoadListener; |
||||||
|
import com.sun.javafx.iio.ImageLoader; |
||||||
|
import com.sun.javafx.iio.ImageMetadata; |
||||||
|
import com.sun.javafx.iio.ImageStorageException; |
||||||
|
import com.sun.javafx.iio.common.ImageTools; |
||||||
|
import com.sun.javafx.iio.gif.GIFImageLoaderFactory; |
||||||
|
import com.sun.javafx.tk.PlatformImage; |
||||||
|
import com.sun.prism.Image; |
||||||
|
import com.sun.prism.impl.PrismSettings; |
||||||
|
import sun.util.logging.PlatformLogger; |
||||||
|
|
||||||
|
import java.io.IOException; |
||||||
|
import java.io.InputStream; |
||||||
|
|
||||||
|
/** |
||||||
|
* 边加载边播放的gif加载器 |
||||||
|
* |
||||||
|
* @author daniel |
||||||
|
*/ |
||||||
|
class PrismImageLoader2 implements com.sun.javafx.tk.ImageLoader { |
||||||
|
|
||||||
|
private static PlatformLogger imageioLogger = null; |
||||||
|
|
||||||
|
private Image[] images; |
||||||
|
private int[] delayTimes; |
||||||
|
private int width; |
||||||
|
private int height; |
||||||
|
private int gifCount = 1; |
||||||
|
private Exception exception; |
||||||
|
|
||||||
|
public PrismImageLoader2(final String url, int gifCount, final int width, final int height, |
||||||
|
final boolean preserveRatio, final boolean smooth) { |
||||||
|
this.gifCount = gifCount; |
||||||
|
images = new Image[gifCount]; |
||||||
|
delayTimes = new int[gifCount]; |
||||||
|
this.width = width; |
||||||
|
this.height = height; |
||||||
|
new Thread() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
InputStream inputStream = null; |
||||||
|
try { |
||||||
|
inputStream = ImageTools.createInputStream(url); |
||||||
|
loadAll(inputStream, width, height, preserveRatio, smooth); |
||||||
|
} catch (IOException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} finally { |
||||||
|
try { |
||||||
|
inputStream.close(); |
||||||
|
} catch (IOException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
}.start(); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int getWidth() { |
||||||
|
return width; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int getHeight() { |
||||||
|
return height; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int getFrameCount() { |
||||||
|
return gifCount; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public PlatformImage getFrame(int index) { |
||||||
|
while (images[index] == null) { |
||||||
|
synchronized (this) { |
||||||
|
if (images[index] == null) { |
||||||
|
try { |
||||||
|
this.wait(); |
||||||
|
} catch (InterruptedException e) { |
||||||
|
e.printStackTrace(); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return images[index]; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public int getFrameDelay(int index) { |
||||||
|
// while (images[0] == null) {
|
||||||
|
// synchronized (this) {
|
||||||
|
// if(images[0] == null) {
|
||||||
|
// try {
|
||||||
|
// this.wait();
|
||||||
|
// } catch (InterruptedException e) {
|
||||||
|
// e.printStackTrace();
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// return 0;
|
||||||
|
// }
|
||||||
|
// return delayTimes[0];
|
||||||
|
// 直接使用第一帧的时间
|
||||||
|
return 40; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int getLoopCount() { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public Exception getException() { |
||||||
|
return exception; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void loadAll(InputStream stream, int w, int h, |
||||||
|
boolean preserveRatio, boolean smooth) { |
||||||
|
ImageLoadListener listener = new PrismLoadListener(); |
||||||
|
|
||||||
|
try { |
||||||
|
ImageLoader loader = null; |
||||||
|
loader = GIFImageLoaderFactory.getInstance().createImageLoader(stream); |
||||||
|
loader.addListener(listener); |
||||||
|
|
||||||
|
for (int i = 0; i < gifCount; i++) { |
||||||
|
ImageFrame imageFrame = loader.load(i, w, h, preserveRatio, smooth); |
||||||
|
images[i] = convert(imageFrame); |
||||||
|
synchronized (this) { |
||||||
|
this.notify(); |
||||||
|
} |
||||||
|
} |
||||||
|
} catch (ImageStorageException e) { |
||||||
|
handleException(e); |
||||||
|
} catch (Exception e) { |
||||||
|
handleException(e); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private void handleException(final ImageStorageException isException) { |
||||||
|
// unwrap ImageStorageException if possible
|
||||||
|
final Throwable exceptionCause = isException.getCause(); |
||||||
|
if (exceptionCause instanceof Exception) { |
||||||
|
handleException((Exception) exceptionCause); |
||||||
|
} else { |
||||||
|
handleException((Exception) isException); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private void handleException(final Exception exception) { |
||||||
|
if (PrismSettings.verbose) { |
||||||
|
exception.printStackTrace(System.err); |
||||||
|
} |
||||||
|
this.exception = exception; |
||||||
|
} |
||||||
|
|
||||||
|
private Image convert(ImageFrame imgFrames) { |
||||||
|
ImageFrame frame = imgFrames; |
||||||
|
Image image = Image.convertImageFrame(frame); |
||||||
|
ImageMetadata metadata = frame.getMetadata(); |
||||||
|
if (metadata != null) { |
||||||
|
Integer delay = metadata.delayTime; |
||||||
|
if (delay != null) { |
||||||
|
delayTimes[0] = delay.intValue(); |
||||||
|
} |
||||||
|
} |
||||||
|
return image; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the PlatformLogger for logging imageio-related activities. |
||||||
|
*/ |
||||||
|
private static synchronized PlatformLogger getImageioLogger() { |
||||||
|
if (imageioLogger == null) { |
||||||
|
imageioLogger = PlatformLogger.getLogger("imageio"); |
||||||
|
} |
||||||
|
|
||||||
|
return imageioLogger; |
||||||
|
} |
||||||
|
|
||||||
|
private class PrismLoadListener implements ImageLoadListener { |
||||||
|
@Override |
||||||
|
public void imageLoadWarning(ImageLoader loader, String message) { |
||||||
|
getImageioLogger().warning(message); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void imageLoadProgress(ImageLoader loader, |
||||||
|
float percentageComplete) { |
||||||
|
// progress only matters when backgroundLoading=true, but
|
||||||
|
// currently we are relying on AbstractRemoteResource for tracking
|
||||||
|
// progress of the InputStream, so there's no need to implement
|
||||||
|
// this for now; eventually though we might want to consider
|
||||||
|
// moving away from AbstractRemoteResource and instead use
|
||||||
|
// the built-in support for progress in the javafx-iio library...
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void imageLoadMetaData(ImageLoader loader, ImageMetadata metadata) { |
||||||
|
// We currently have no need to listen for ImageMetadata ready.
|
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,49 @@ |
|||||||
|
package com.fr.start.fx; |
||||||
|
|
||||||
|
import com.fr.start.SplashStrategy; |
||||||
|
import javafx.application.Application; |
||||||
|
import javafx.application.Platform; |
||||||
|
|
||||||
|
import java.util.concurrent.ExecutorService; |
||||||
|
import java.util.concurrent.Executors; |
||||||
|
|
||||||
|
/** |
||||||
|
* JavaFx方式启动启动动画。这种方式在mac下与 |
||||||
|
* swing一起启动会会出现线程死锁,jvm等问题, |
||||||
|
* 所以这个方式仅用于windows上。 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
* @date 2018/6/4 |
||||||
|
* @see com.fr.start.jni.SplashMac |
||||||
|
*/ |
||||||
|
public class SplashFx implements SplashStrategy { |
||||||
|
|
||||||
|
private SplashFxWindow test; |
||||||
|
private static final ExecutorService SERVICE = Executors.newSingleThreadExecutor(); |
||||||
|
|
||||||
|
@Override |
||||||
|
public void show() { |
||||||
|
SERVICE.execute(new Runnable() { |
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
Application.launch(SplashFxWindow.class); |
||||||
|
} |
||||||
|
}); |
||||||
|
test = SplashFxWindow.waitForStartUpTest(); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void hide() { |
||||||
|
Platform.exit(); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void updateModuleLog(String text) { |
||||||
|
test.updateModuleInfo(text); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void updateThanksLog(String text) { |
||||||
|
test.updateThanks(text); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,133 @@ |
|||||||
|
package com.fr.start.fx; |
||||||
|
|
||||||
|
import com.bulenkov.iconloader.util.JBUI; |
||||||
|
import com.fr.base.FRContext; |
||||||
|
import com.fr.stable.OperatingSystem; |
||||||
|
import javafx.application.Application; |
||||||
|
import javafx.scene.Scene; |
||||||
|
import javafx.scene.image.Image; |
||||||
|
import javafx.scene.image.ImageView; |
||||||
|
import javafx.scene.layout.AnchorPane; |
||||||
|
import javafx.scene.paint.Color; |
||||||
|
import javafx.scene.text.Font; |
||||||
|
import javafx.scene.text.Text; |
||||||
|
import javafx.stage.Stage; |
||||||
|
import javafx.stage.StageStyle; |
||||||
|
|
||||||
|
import java.util.concurrent.CountDownLatch; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* JavaFx启动动画窗口 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
*/ |
||||||
|
public class SplashFxWindow extends Application { |
||||||
|
|
||||||
|
private static float JBUI_INIT_SCALE = JBUI.scale(1f); |
||||||
|
|
||||||
|
private static final String ARIAL_FONT_NAME = "Arial"; |
||||||
|
private static final String PF_FONT_NAME = "PingFang"; |
||||||
|
private static final String YAHEI_FONT_NAME = "Microsoft YaHei"; |
||||||
|
private static final int MODULE_INFO_LEFT_MARGIN = 36; |
||||||
|
private static final int MODULE_INFO_BOTTOM_MARGIN = 28; |
||||||
|
private static final int THINKS_BOTTOM_RIGHT = 35; |
||||||
|
private static final int THINKS_BOTTOM_MARGIN = 27; |
||||||
|
private static final int WINDOW_WIDTH = 640; |
||||||
|
private static final int WINDOW_HEIGHT = 360; |
||||||
|
private static final int FONT = 12; |
||||||
|
private static final String THINKS_COLOR = "#82b1ce"; |
||||||
|
|
||||||
|
private static final CountDownLatch LATCH = new CountDownLatch(1); |
||||||
|
private static SplashFxWindow app = null; |
||||||
|
|
||||||
|
private Text moduleInfo; |
||||||
|
private Text thanks; |
||||||
|
|
||||||
|
private static int uiScale(int i) { |
||||||
|
return (int) (i * JBUI_INIT_SCALE); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取当前运行实例。黑科技 |
||||||
|
* |
||||||
|
* @return 运行实例 |
||||||
|
*/ |
||||||
|
public static SplashFxWindow waitForStartUpTest() { |
||||||
|
try { |
||||||
|
LATCH.await(); |
||||||
|
} catch (InterruptedException e) { |
||||||
|
FRContext.getLogger().error(e.getMessage(), e); |
||||||
|
} |
||||||
|
return app; |
||||||
|
} |
||||||
|
|
||||||
|
private static void setApp(SplashFxWindow window) { |
||||||
|
app = window; |
||||||
|
LATCH.countDown(); |
||||||
|
} |
||||||
|
|
||||||
|
public SplashFxWindow() { |
||||||
|
setApp(this); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void start(Stage primaryStage) { |
||||||
|
AnchorPane root = new AnchorPane(); |
||||||
|
primaryStage.initStyle(StageStyle.TRANSPARENT); |
||||||
|
long t = System.currentTimeMillis(); |
||||||
|
Image image = new FastGifImage("com/fr/base/images/oem/splash_10.gif", 254, WINDOW_WIDTH, WINDOW_HEIGHT); |
||||||
|
|
||||||
|
ImageView gif = new ImageView(image); |
||||||
|
|
||||||
|
AnchorPane.setBottomAnchor(gif, 0d); |
||||||
|
AnchorPane.setTopAnchor(gif, 0d); |
||||||
|
AnchorPane.setLeftAnchor(gif, 0d); |
||||||
|
AnchorPane.setRightAnchor(gif, 0d); |
||||||
|
Font font; |
||||||
|
if (OperatingSystem.isWindows()) { |
||||||
|
font = new Font(YAHEI_FONT_NAME, uiScale(FONT)); |
||||||
|
} else if (OperatingSystem.isMacOS()) { |
||||||
|
font = new Font(PF_FONT_NAME, uiScale(FONT)); |
||||||
|
} else { |
||||||
|
font = new Font(ARIAL_FONT_NAME, uiScale(FONT)); |
||||||
|
} |
||||||
|
|
||||||
|
moduleInfo = new Text(); |
||||||
|
moduleInfo.setFont(font); |
||||||
|
moduleInfo.setFill(Color.WHITE); |
||||||
|
AnchorPane.setLeftAnchor(moduleInfo, (double) uiScale(MODULE_INFO_LEFT_MARGIN)); |
||||||
|
AnchorPane.setBottomAnchor(moduleInfo, (double) uiScale(MODULE_INFO_BOTTOM_MARGIN)); |
||||||
|
thanks = new Text(); |
||||||
|
thanks.setFont(font); |
||||||
|
thanks.setFill(Color.valueOf(THINKS_COLOR)); |
||||||
|
AnchorPane.setRightAnchor(thanks, (double) uiScale(THINKS_BOTTOM_RIGHT)); |
||||||
|
AnchorPane.setBottomAnchor(thanks, (double) uiScale(THINKS_BOTTOM_MARGIN)); |
||||||
|
|
||||||
|
root.getChildren().add(gif); |
||||||
|
root.getChildren().add(moduleInfo); |
||||||
|
root.getChildren().add(thanks); |
||||||
|
|
||||||
|
Scene scene = new Scene(root, WINDOW_WIDTH, WINDOW_HEIGHT, null); |
||||||
|
primaryStage.setScene(scene); |
||||||
|
primaryStage.show(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 更新模块信息 |
||||||
|
* |
||||||
|
* @param s 文字 |
||||||
|
*/ |
||||||
|
public void updateModuleInfo(String s) { |
||||||
|
moduleInfo.setText(s); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 更新欢迎信息 |
||||||
|
* |
||||||
|
* @param s 文字 |
||||||
|
*/ |
||||||
|
public void updateThanks(String s) { |
||||||
|
thanks.setText(s); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,69 @@ |
|||||||
|
package com.fr.start.jni; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.FileOutputStream; |
||||||
|
import java.io.InputStream; |
||||||
|
|
||||||
|
/** |
||||||
|
* Splash JNI调用。jni类改名或者移包之后 |
||||||
|
* 必须重新编译动态库 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
* @date 2018/6/4 |
||||||
|
*/ |
||||||
|
public class SplashJNI { |
||||||
|
|
||||||
|
static { |
||||||
|
try { |
||||||
|
System.setProperty("java.library.path", "."); |
||||||
|
System.loadLibrary("splash"); |
||||||
|
} catch (UnsatisfiedLinkError e) { |
||||||
|
loadLibraryFromJar("/com/fr/start/jni/splash.dylib"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 显示启动动画窗口 |
||||||
|
*/ |
||||||
|
public native void show(String path); |
||||||
|
|
||||||
|
/** |
||||||
|
* 隐藏启动动画窗口 |
||||||
|
*/ |
||||||
|
public native void hide(); |
||||||
|
|
||||||
|
/** |
||||||
|
* 设置模块加载信息 |
||||||
|
*/ |
||||||
|
public native void updateModuleLog(String text); |
||||||
|
|
||||||
|
/** |
||||||
|
* 设置感谢文字 |
||||||
|
*/ |
||||||
|
public native void updateThanksLog(String text); |
||||||
|
|
||||||
|
/** |
||||||
|
* 从jar中加载动态库 |
||||||
|
* |
||||||
|
* @param path 路径,如/com/a/b |
||||||
|
* @throws UnsatisfiedLinkError 没有找到合适的动态库 |
||||||
|
*/ |
||||||
|
private static void loadLibraryFromJar(String path) throws UnsatisfiedLinkError { |
||||||
|
try (InputStream inputStream = SplashJNI.class.getResourceAsStream(path)) { |
||||||
|
File tempLib = File.createTempFile(path, ""); |
||||||
|
|
||||||
|
byte[] buffer = new byte[1024]; |
||||||
|
int read = -1; |
||||||
|
|
||||||
|
try (FileOutputStream fileOutputStream = new FileOutputStream(tempLib)) { |
||||||
|
while ((read = inputStream.read(buffer)) != -1) { |
||||||
|
fileOutputStream.write(buffer, 0, read); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
System.load(tempLib.getAbsolutePath()); |
||||||
|
} catch (Exception e) { |
||||||
|
throw new UnsatisfiedLinkError("Unable to open " + path + " from jar file."); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,87 @@ |
|||||||
|
package com.fr.start.jni; |
||||||
|
|
||||||
|
import com.fr.stable.ProductConstants; |
||||||
|
import com.fr.stable.StableUtils; |
||||||
|
import com.fr.start.SplashContext; |
||||||
|
import com.fr.start.SplashStrategy; |
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.FileOutputStream; |
||||||
|
import java.io.IOException; |
||||||
|
import java.io.InputStream; |
||||||
|
|
||||||
|
/** |
||||||
|
* mac上使用jni方式绘制gif。不使用javafx有两个原因: |
||||||
|
* 1.mac上javafx和swing同时启动会导致卡死; |
||||||
|
* 2.platform.exit会导致设计器崩溃 |
||||||
|
* |
||||||
|
* @author vito |
||||||
|
* @see com.fr.start.fx.SplashFx |
||||||
|
*/ |
||||||
|
public class SplashMac implements SplashStrategy { |
||||||
|
|
||||||
|
private static final String SPLASH_CACHE_NAME = "splash_10.gif"; |
||||||
|
private static final String SPLASH_PATH = "/com/fr/base/images/oem/splash_10.gif"; |
||||||
|
|
||||||
|
private SplashJNI jni; |
||||||
|
|
||||||
|
public SplashMac() { |
||||||
|
jni = new SplashJNI(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 将jar中的资源拷贝到缓存文件夹 |
||||||
|
* |
||||||
|
* @return 路径 |
||||||
|
* @throws IOException 拷贝失败 |
||||||
|
*/ |
||||||
|
private static String loadResFromJar() throws UnsatisfiedLinkError { |
||||||
|
File tempLib = null; |
||||||
|
try (InputStream inputStream = SplashContext.class.getResourceAsStream(SplashMac.SPLASH_PATH)) { |
||||||
|
tempLib = new File(StableUtils.pathJoin(ProductConstants.getEnvHome(), SPLASH_CACHE_NAME)); |
||||||
|
byte[] buffer = new byte[1024]; |
||||||
|
int read = -1; |
||||||
|
try (FileOutputStream fileOutputStream = new FileOutputStream(tempLib)) { |
||||||
|
while ((read = inputStream.read(buffer)) != -1) { |
||||||
|
fileOutputStream.write(buffer, 0, read); |
||||||
|
} |
||||||
|
} |
||||||
|
return tempLib.getAbsolutePath(); |
||||||
|
} catch (IOException e) { |
||||||
|
tempLib.deleteOnExit(); |
||||||
|
throw new UnsatisfiedLinkError("Unable to open " + SplashMac.SPLASH_PATH + " from jar file."); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void show() { |
||||||
|
if (jni != null) { |
||||||
|
File splash = new File(StableUtils.pathJoin(ProductConstants.getEnvHome(), SPLASH_CACHE_NAME)); |
||||||
|
String path = splash.exists() ? splash.getAbsolutePath() : loadResFromJar(); |
||||||
|
jni.show(path); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void hide() { |
||||||
|
if (jni != null) { |
||||||
|
jni.hide(); |
||||||
|
jni = null; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void updateModuleLog(String text) { |
||||||
|
if (jni != null) { |
||||||
|
jni.updateModuleLog(text); |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void updateThanksLog(String text) { |
||||||
|
if (jni != null) { |
||||||
|
jni.updateThanksLog(text); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Binary file not shown.
Loading…
Reference in new issue