Android和js、H5进行交互数据

Android和H5、Js进行交互调用

Android开发过程中,我们或多或少都会用到webview,使用webview来展示一些经常变动的界面更加方便简单,也已于维护。另一方面hybrid App开发现在用的也越来越多了。其中native和h5之间的交互更是必不可少的。

具体Android中是如何和h5交互的?或者说Android中是如何和js交互的。

1 Webview加载页面

我们都知道在android中是通过webview来加载html页面的。根据html文件所在的位置不同写法也不同。

例如:加载assets文件夹下的test,html页面

mWebview.loadUrl("file:///android_asset/test.html")

例如:加载网页

mWebView.loadUrl("http://www.baidu.com")

如果只是这样调用webview.loadUrl 加载的话,那么当你点击页面中的链接的时候,页面将会在你手机默认的浏览器上打开。那如果想要在页面在App内中打开的话,那么就得设置setWebViewClient:

mWebView.setWebViewClient(new WebViewClient() {

        @Override

        public boolean shouldOverrideUrlLoading(WebView view, String url) {

                //我们可以在这里拦截特定的rl请求,然后进行自己要的操作

                if (url.equals("file:///android_asset/test2.html")) {

                    Log.e(TAG, "shouldOverrideUrlLoading: " + url);

                    startActivity(new Intent(MainActivity.this,Main2Activity.class));

                    return true;

                } else {

                //这里我们自己重新加载新的url页面,防止点击链接跳转到系统浏览器

                    mWebView.loadUrl(url);

                    return true;

                }

            }

        }

    });

重写Activity的onBackPressed方法,使得返回按钮不会关闭当前页面,而是返回webview上一个历史页面。

@Override

    public void onBackPressed() {

        if (webView.canGoBack()) {

            //返回上一个页

            webView.goBack();

            return ;

        }

        super.onBackPressed();

    }

二、给webview添加加载新页面的进度条。

开启和关闭进度条

webView.setWebViewClient(new WebViewClient() {

      //重写页面打开和结束的监听。打开时弹出对话框,关闭时隐藏

      /**

      * 界面打开的回调

      */

      @Override

      public void onPageStarted(WebView view, String url, Bitmap favicon) {

          if (progressDialog != null && progressDialog.isShowing()) {

              progressDialog.dismiss();

          }

          //弹出对话框

          progressDialog = new ProgressDialog(JSActivity.this);

          progressDialog.setTitle("提示");

          progressDialog.setMessage("软软正在拼命加载……");

          progressDialog.show();

      }

      /**

      * 界面打开完毕的回调

      */

      @Override

      public void onPageFinished(WebView view, String url) {

          //隐藏对话框:不为空,正在显示。才隐藏关闭

          if (progressDialog != null && progressDialog.isShowing()) {

              progressDialog.dismiss();

          }

      }

  });

让进度条显示页面加载进度:

//设置进度条

        //WebChromeClient与webViewClient的区别

        //webViewClient处理偏界面的操作:打开新界面,界面打开,界面打开结束

        //WebChromeClient处理偏js的操作

        webView.setWebChromeClient(new WebChromeClient() {

            /**

            * 进度改变的回调

            * WebView:就是本身

            * newProgress:即将要显示的进度

            */

            @Override

            public void onProgressChanged(WebView view, int newProgress) {

                if (progressDialog != null && progressDialog.isShowing())

                    progressDialog.setMessage("软软正在拼命加载……" + newProgress + "%");

            }

三、android本地通过java调用HTML页面中的javaScript方法

想要调用js方法那么就必须让webview支持js的代码。

//首先设置Webview支持JS代码

webView.getSettings().setJavaScriptEnabled(true);

若调用的js方法没有返回值,则可以直接调用mWebView.loadUrl(“javascript:do());其中do是js中的方法;

若有返回值时我们可以调用mwebview,evaluteJavascript方法;

@TargetApi(Build.VERSION_CODES.KITKAT)

    public void onSum(View view){

        webView.evaluateJavascript("sum(1,2)", new ValueCallback<String>() {

            @Override

            public void onReceiveValue(String value) {

                Toast.makeText(getApplicationContext(),

                        "相加结果:"+value, Toast.LENGTH_SHORT).show();

            }

        });

    }

    public void onDoing(View view){

        String msg = "测试";

        webView.loadUrl("javascript:showInfoFromJava('"+msg+"')");

    }

对应的js方法

    function sum(a,b){

        return a+b;

    }

    function showInfoFromJava(){

        document.getElementById("p").innerHTML="Java成功调的JS方法";

    }

四、js调用Android本地java方法

在android 4.2以上可以直接使用@javascriptinterface注解来声明,下面是在一个本地java方法

public void addJavascriptInterface(Object object, String name);

1 object参数:在object对象里面添加我们想要在Js里面调用的Android方法,下面的代码中我们调用了showToast方法

2 name参数:这里的name就是我们可以在JS里面调用的对象名称,对应下面代码中的JSText

对应的JS代码

function jsJava(){

        //调用java的方法,顶级对象,java方法

        //可以直接访问JSTest,这是因为JSTest挂载到js的window对象下了

        JSTest.showToast("我是被JS执行的Android代码");

    }

对应的Java代码:

//java与js回调,自定义方法

        //1.java调用js

        //2.js调用java

        //首先java暴露接口,供js调用

        /**

        * obj:暴露的要调用的对象

        * interfaceName:对象的映射名称 ,object的对象名,在js中可以直接调用

        * 在html的js中:JSTest.showToast(msg)

        * 可以直接访问JSTest,这是因为JSTest挂载到js的window对象下了

        */

        webView.addJavascriptInterface(new Object() {

            //定义要调用的方法

            //msg由js调用的时候传递

            @JavascriptInterface

            public void showToast(String msg) {

                Toast.makeText(getApplicationContext(),

                        msg, Toast.LENGTH_SHORT).show();

            }

        }, "JSTest");

五、重绘alert 、confirm和prompt的弹出效果,并把用户具体操作结果回调给JS

alert弹窗:

重绘confirm弹窗:

重绘prompt弹窗:

具体代码:

/**

            * Webview加载html中有alert()执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            * JsResult:java将结果回传到js中

            */

            @TargetApi(Build.VERSION_CODES.JELLY_BEAN_MR1)

            @Override

            public boolean onJsAlert(WebView view, String url, String message, final JsResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:看到这个,说明Java成功重写了Js的Alert方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理确定按钮,且通过jsresult传递,告诉js点击的是确定按钮

                        result.confirm();

                    }

                });

                builder.show();

                //自己处理

                result.cancel();

                return true;

            }

            /**

            * Webview加载html中有confirm执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            * JsResult:java将结果回传到js中

            */

            @Override

            public boolean onJsConfirm(WebView view, String url, String message, final JsResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:看到这个,说明Java成功重写了Js的Confirm方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理确定按钮,且通过jsresult传递,告诉js点击的是确定按钮

                        result.confirm();

                    }

                });

                builder.setNegativeButton("取消", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理取消按钮,且通过jsresult传递,告诉js点击的是取消按钮

                        result.cancel();

                    }

                });

                builder.show();

                //自己处理

                result.cancel();

                return true;

            }

            /**

            * Webview加载html中有prompt()执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            *defaultValue就是prompt的第二个参数值,输入框的默认值

            * JsPromptResult:java将结果重新回传到js中

            */

            @Override

            public boolean onJsPrompt(WebView view, String url, String message, String defaultValue,

                                      final JsPromptResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:看到这个,说明Java成功重写了Js的Prompt方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                //添加一个EditText

                final EditText editText = new EditText(JSActivity.this);

                editText.setText(defaultValue);//这个就是prompt 输入框的默认值

                //添加到对话框

                builder.setView(editText);

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //获取edittext的新输入的值

                        String newValue = editText.getText().toString().trim();

                        //处理确定按钮了,且过jsresult传递,告诉js点击的是确定按钮(参数就是输入框新输入的值,我们需要回传到js中)

                        result.confirm(newValue);

                    }

                });

                builder.setNegativeButton("取消", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理取消按钮,且过jsresult传递,告诉js点击的是取消按钮

                        result.cancel();

                    }

                });

                builder.show();

                //自己处理

                result.cancel();

                return true;

            }

        });

效果图:

界面上方是两个按钮,下方是一个webview控件,开启页面自动加载url,这里为了方便学习,

我已经写了一个html文件放置在了Asset文件中,通过 file:///android_asset/test.html 来进行加载。

webview成功加载页面后,会出现四个新的按钮,点击不同的按钮,会产生不同的效果

asset文件夹下面的test.html文件

<html>

<head>

    <meta charset="UTF-8">

    <title>Document</title>

    <script>

    function jsAlert(){

        var r = alert("我是Alert的提示框");

        document.getElementById("p").innerHTML="Java成功调的JS的alert方法";

    }

    function jsConFirm(){

      var r = confirm("我是ConFirm的弹出框");

      if (r == true)

      {

        document.getElementById("p").innerHTML="用户点击了确认按钮";

      }else{

        document.getElementById("p").innerHTML="用户点击了取消按钮";

      }

    }

    function jsPrompt(){

        //第一个参数是提示

        //第二个参数是默认值

        var r = prompt("请输入姓名:","小明同学");

        if (r != null)

        {

        document.getElementById("p").innerHTML="用户输入的姓名为:"+r;

        }else{

        document.getElementById("p").innerHTML="用户点击了取消按钮";

        }

    }

    function jsJava(){

        //调用java的方法,顶级对象,java方法

        //可以直接访问JSTest,这是因为JSTest挂载到js的window对象下了

        JSTest.showToast("我是被JS执行的Android代码");

    }

    function sum(a,b){

    return a+b;

    }

    function showInfoFromJava(){

    document.getElementById("p").innerHTML="JS方法成功被Java调用";

    }

    </script>

</head>

<body>

<h3 id="p">界面成功初始化</h3>

<h5 onclick="jsAlert()">调用jsAlert方法</h5>

<input type="button" value="开启Alert提示框" onclick="jsAlert()"/>

<h5 onclick="jsAlert()">调用jsConFirm方法</h5>

<input type="button" value="开启ConFirm弹窗" onclick="jsConFirm()"/>

<h5 onclick="jsAlert()">调用jsPrompt方法</h5>

<input type="button" value="开启Prompt弹窗" onclick="jsPrompt()"/>

<h5 onclick="jsAlert()">调用jsJava方法</h5>

<input type="button" value="调用Java方法" onclick="jsJava()"/>

</body>

</html>

具体的java代码

public class JSActivity extends AppCompatActivity {

    //assets下的文件的test.html所在的绝对路径

    private static final String DEFAULT_URL = "file:///android_asset/test.html";

    private WebView webView;

    private ProgressDialog progressDialog;//加载界面的菊花

    @Override

    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_js);

        initView();

        initWebView();

    }

    /**

    * 初始化控件

    */

    private void initView() {

        webView = (WebView) findViewById(R.id.webView);

        webView.loadUrl(DEFAULT_URL);

    }

    /**

    * 初始化webview

    */

    private void initWebView() {

        //首先设置Webview支持JS代码

        webView.getSettings().setJavaScriptEnabled(true);

        //Webview自己处理超链接(Webview的监听器非常多,封装一个特殊的监听类来处理)

        webView.setWebViewClient(new WebViewClient() {

            /**

            * 当打开超链接的时候,回调的方法

            * WebView:自己本身webView

            * url:即将打开的url

            */

            @Override

            public boolean shouldOverrideUrlLoading(WebView view, String url) {

                //自己处理新的url

                webView.loadUrl(url);

                //true就是自己处理

                return true;

            }

            //重写页面打开和结束的监听。打开时弹出菊花

            /**

            * 界面打开的回调

            */

            @Override

            public void onPageStarted(WebView view, String url, Bitmap favicon) {

                if (progressDialog != null && progressDialog.isShowing()) {

                    progressDialog.dismiss();

                }

                //弹出菊花

                progressDialog = new ProgressDialog(JSActivity.this);

                progressDialog.setTitle("提示");

                progressDialog.setMessage("软软正在拼命加载……");

                progressDialog.show();

            }

            /**

            * 重写页面打开和结束的监听。打开时弹出菊花,关闭时隐藏菊花

            * 界面打开完毕的回调

            */

            @Override

            public void onPageFinished(WebView view, String url) {

                //隐藏菊花:不为空,正在显示。才隐藏

                if (progressDialog != null && progressDialog.isShowing()) {

                    progressDialog.dismiss();

                }

            }

        });

        //设置进度条

        //WebChromeClient与webViewClient的区别

        //webViewClient处理偏界面的操作:打开新界面,界面打开,界面打开结束

        //WebChromeClient处理偏js的操作

        webView.setWebChromeClient(new WebChromeClient() {

            /**

            * 进度改变的回调

            * WebView:就是本身

            * newProgress:即将要显示的进度

            */

            @Override

            public void onProgressChanged(WebView view, int newProgress) {

                if (progressDialog != null && progressDialog.isShowing())

                    progressDialog.setMessage("软软正在拼命加载……" + newProgress + "%");

            }

            /**

            * 重写alert、confirm和prompt的弹出效果,并把用户操作的结果回调给JS

            */

            /**

            * Webview加载html中有alert()执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            * JsResult:java将结果回传到js中

            */

            @Override

            public boolean onJsAlert(WebView view, String url, String message, final JsResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:看到这个,说明Java成功重写了Js的Alert方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理确定按钮了,且通过jsresult传递,告诉js点击的是确定按钮

                        result.confirm();

                    }

                });

                builder.setOnCancelListener(new DialogInterface.OnCancelListener() {

                    @Override

                    public void onCancel(DialogInterface dialog) {

                        //防止用户点击对话框外围,再次点击按钮页面无响应

                        result.cancel();

                    }

                });

                builder.show();

                //自己处理

                return true;

            }

            /**

            * Webview加载html中有confirm执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            * JsResult:java将结果回传到js中

            */

            @Override

            public boolean onJsConfirm(WebView view, String url, String message, final JsResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:" +

                        "看到这个,说明Java成功重写了Js的Confirm方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理确定按钮了,且通过jsresult传递,告诉js点击的是确定按钮

                        result.confirm();

                    }

                });

                builder.setNegativeButton("取消", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理取消按钮,且通过jsresult传递,告诉js点击的是取消按钮

                        result.cancel();

                    }

                });

                builder.setOnCancelListener(new DialogInterface.OnCancelListener() {

                    @Override

                    public void onCancel(DialogInterface dialog) {

                        //防止用户点击对话框外围,再次点击按钮页面无反应

                        result.cancel();

                    }

                });

                builder.show();

                //自己处理

                return true;

            }

            /**

            * Webview加载html中有prompt()执行的时候,会回调这个方法

            * url:当前Webview显示的url

            * message:alert的参数值

            *defaultValue就是prompt的第二个参数值,输入框的默认值

            * JsPromptResult:java将结果重新回传到js中

            */

            @Override

            public boolean onJsPrompt(WebView view, String url, String message, String defaultValue,

                                      final JsPromptResult result) {

                AlertDialog.Builder builder = new AlertDialog.Builder(JSActivity.this);

                builder.setTitle("提示:看到这个,说明Java成功重写了Js的Prompt方法");

                builder.setMessage(message);//这个message就是alert传递过来的值

                //添加一个EditText

                final EditText editText = new EditText(JSActivity.this);

                editText.setText(defaultValue);//这个就是prompt 输入框的默认值

                //添加到对话框

                builder.setView(editText);

                builder.setPositiveButton("确定", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //获取edittext的新输入的值

                        String newValue = editText.getText().toString().trim();

                        //处理确定按钮了,且过jsresult传递,告诉js点击的是确定按钮(参数就是输入框新输入的值,我们需要回传到js中)

                        result.confirm(newValue);

                    }

                });

                builder.setNegativeButton("取消", new OnClickListener() {

                    @Override

                    public void onClick(DialogInterface dialog, int which) {

                        //处理取消按钮,且过jsresult传递,告诉js点击的是取消按钮

                        result.cancel();

                    }

                });

                builder.setOnCancelListener(new DialogInterface.OnCancelListener() {

                    @Override

                    public void onCancel(DialogInterface dialog) {

                        //防止用户点击对话框外围,再次点击按钮页面无反应

                        result.cancel();

                    }

                });

                builder.show();

                //自己处理

                return true;

            }

        });

        //java与js回调,自定义方法

        //1.java调用js

        //2.js调用java

        //首先java暴露接口,供js调用

        /**

        * obj:暴露的要调用的对象

        * interfaceName:对象的映射名称 ,object的对象名,在js中可以直接调用

        * 在html的js中:JSTest.showToast(msg)

        * 可以直接访问JSTest,这是因为JSTest挂载到js的window对象下了

        */

        webView.addJavascriptInterface(new Object() {

            //定义要调用的方法

            //msg由js调用的时候传递

            @JavascriptInterface

            public void showToast(String msg) {

                Toast.makeText(getApplicationContext(),

                        msg, Toast.LENGTH_SHORT).show();

            }

        }, "JSTest");

    }

    @TargetApi(Build.VERSION_CODES.KITKAT)

    public void onSum(View view){

        webView.evaluateJavascript("sum(1,2)", new ValueCallback<String>() {

            @Override

            public void onReceiveValue(String value) {

                Toast.makeText(getApplicationContext(),

                        "相加结果:"+value, Toast.LENGTH_SHORT).show();

            }

        });

    }

    public void onDoing(View view){

        String msg = "测试";

        webView.loadUrl("javascript:showInfoFromJava('"+msg+"')");

    }

    @Override

    public void onBackPressed() {

        if (webView.canGoBack()) {

            //返回上一个页

            webView.goBack();

            return ;

        }

        super.onBackPressed();

    }

}

布局文件activity_js.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools"

    android:layout_width="match_parent"

    android:layout_height="match_parent"

    android:orientation="vertical"

    tools:context=".JSoupHtmlActivity">

    <Button

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:onClick="onSum"

        android:text="Java调用JS的Sum方法"

        tools:ignore="OnClick" />

    <Button

        android:layout_width="match_parent"

        android:layout_height="wrap_content"

        android:onClick="onDoing"

        android:text="Java调用JS的  showInfoFromJava 方法"

        tools:ignore="OnClick" />

    <WebView

        android:id="@+id/webView"

        android:layout_width="fill_parent"

        android:layout_height="fill_parent" />

</LinearLayout>

代码过程描述的废话我就不多说了,注释写的算是比较仔细了,另外再强调两点需要注意的地方:

1、不要忘记通过setJavaScriptEnabled(true)设置webview支持JS代码

2、在使用addJavascriptInterface方法添加挂载对象时,要注意在Android4.2之后需要给对象方法加上@JavascriptInterface注解。

3、重绘alert、confirm和prompt的弹出效果之后,在对话框结束之后一定要调用result.confirm()或者result.cancel()两个方法中的一个,否则会出现后续再次点击html页面按钮,页面无响应的情况

---------------------

作者:流船

来源:CSDN

原文:https://blog.csdn.net/qq_35229022/article/details/79739048

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容