背景

org 直接导出的 html 文件,需要转化为 pdf,才能方便大家浏览。

解决方法

结果网上搜索之后,找到 wkhtmltopdf 这个工具,经过尝试之后,使用的时候,是有一些注意点的。

ubuntu

ubuntu 下面可以直接使用 apt 安装,但是安装后的,只能使用默认选项,其他的很多参数是无效的,提示 qt 的问题。

win

  • windows 下面直接下载 7z 格式的,安装后,使用报错。
  • 去 github 上面下载 0.13.2 的版本,这个版本支持 目录等很多功能,新版反而不支持了。 https://github.com/wkhtmltopdf/wkhtmltopdf/releases/0.12.3.2/

增加专门的目录页

使用 toc --toc-header-text "目录",这样就可以生成专门的目录页。 例如:

.\wkhtmltopdf.exe toc --toc-header-text "目录" C:\Users\Administrator\Downloads\before_tooling_with_production_department.html be.pdf

org 增加标题和中文目录,去除页尾的签名

#+TITLE: 标题
#+LANGUAGE: zh-CN
#+OPTIONS: html-postamble:nil

如果不说明是中文,那么目录栏显示的就是 “Table of Contents”

参考:

Org-mode 导出 html 模版设置
https://bnlt.org/2019/Org-Mode%E5%AF%BC%E5%87%BAhtml%E6%A8%A1%E7%89%88%E8%AE%BE%E7%BD%AE/

使用 wkhtmltopdf 将网页生成 带有 封面 、 目录、水印的pdf
https://www.jianshu.com/p/1a0fb28e6bec

html超链接点不了_HTML 转 PDF 之 wkhtmltopdf 工具精讲
https://blog.csdn.net/weixin_31098573/article/details/112516783

wkhtmltopdf
https://wkhtmltopdf.org/downloads.html

背景

需要在 fragment 进行 replace 然后 pop 回来之后,一些选择项还是原来的选项。

方法

  1. 查阅了很多资料,发现如果是 view 类型那么可以被自动保存,非 view 的就需要我们自己来保存了。
  2. 按照 fragment 的生命周期来说,需要在 onDestroyView 中进行保存,在 onCreateView 中进行读取。
  3. 如果是比较简单的,那么直接使用本身生成的 fragement 类型中的 setArgumentsgetArguments 来得到 bundle 来操作即可。
  4. 如果是比较复杂的,最好自己来实现一个保存和读取的,方便控制。然后在 onDestroyView, onCreateView 里面进行调用即可。
    public Bundle setSaveBundle(Bundle bundle) {
        if (bundle == null) {
            bundle = new Bundle();
        }
        bundle.putInt("type_index", analysisTypeLastSelected);

        DLog.d(TAG, "setSaveBundle: " + "type_index: " + analysisTypeLastSelected);

        return bundle;
    }

    public void getSaveBundle(Bundle bundle) {
        if (bundle == null) {
            DLog.d(TAG, "getSaveBundle: " + "null!");
            return;
        }

        analysisTypeLastSelected = bundle.getInt("type_index");

        DLog.d(TAG, "getSaveBundle: " + "type_index: " + analysisTypeLastSelected);
    }

参考:

[译] 保存/恢复 Activity 和 Fragment 状态的最佳实践
https://segmentfault.com/a/1190000006691830

切换Fragment时实现数据保持
https://www.cnblogs.com/zhujiabin/p/4192613.html

Pro Android学习笔记(四一):Fragment(6):数据保留
https://blog.csdn.net/flowingflying/article/details/12749403

Android Fragment 生命周期onCreatView、onViewCreated
https://blog.csdn.net/asdf717/article/details/51383750

Android中Fragment数据保存和恢复
https://www.jianshu.com/p/015c79bedb41

onSaveInstanceState()和onRestoreInstanceState()使用详解
https://www.jianshu.com/p/27181e2e32d2

Android 使用onSaveInstanceState保存数据
https://blog.csdn.net/qq_34694875/article/details/109530012

android onSaveInstanceState方法
https://www.jianshu.com/p/5dde143a143f

Android Fragment 在返回栈popBackStack() ,返回时onResume问题
https://blog.csdn.net/qq_34983989/article/details/78126362

Android Fragment回退栈管理(popBackStack)
https://blog.csdn.net/qq_35988274/article/details/100518346

转自: https://blog.csdn.net/nonamepao/article/details/120943245

项目介绍

本次项目在上一次类微信界面的设计上,增加了用recyclerview实现的模拟微信聊天联系人列表,其中从列表跳转到联系内容的部分使用了activity的跳转。

设计

列表界面

用一个xml来设计列表的界面,这里采用了最简单的单个textview作为标题模式。

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_margin="8dp"
        android:orientation="vertical">

        <TextView
            android:id="@+id/textView1"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_centerVertical="true"
            android:layout_marginLeft="8dp"
            android:gravity="center"
            android:text="TextView" />

    </LinearLayout>

联系内容界面

显示联系内容的界面新建两个activity来实现,在activity2.xml文件作为被跳转的布局文件用来显示联系内容。

activity_1.xml

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:gravity="center"
        android:orientation="vertical"
        tools:context=".Activity1">

        <TextView
            android:id="@+id/textView7"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:gravity="center"
            android:text="这是第一个activity"
            android:textSize="40sp" />

    </LinearLayout>

activity_2.xml

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:context=".Activity2">

        <TextView
            android:id="@+id/textView5"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:gravity="center"
            android:text="联系内容"
            android:textSize="40dp" />
    </LinearLayout>

列表的展示

修改之前的weixin.java文件,调用布局文件来实现列表的展示

    package com.example.wechatlike;

    import android.content.Context;
    import android.os.Bundle;
    import android.view.LayoutInflater;
    import android.view.View;
    import android.view.ViewGroup;
    import android.app.Fragment;

    import androidx.recyclerview.widget.LinearLayoutManager;
    import androidx.recyclerview.widget.RecyclerView;

    import java.util.ArrayList;
    import java.util.List;

    public class weixin extends Fragment {

        private RecyclerView recyclerView;
        private MyAdapter myadapter;
        private List<String> list = new ArrayList<>();
        private Context context;

        public weixin() {

        }

        private void initData(){
            for(int i=0;i<10;i++) {
                list.add("联系人" + i );
            }
        }

        private void initView3(){
            context=this.getActivity();
            myadapter=new MyAdapter(list,context);

            LinearLayoutManager manager3=new LinearLayoutManager(context);
            manager3.setOrientation(LinearLayoutManager.VERTICAL);

            recyclerView.setAdapter(myadapter);
            recyclerView.setLayoutManager(manager3);
        }
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container,
                                 Bundle savedInstanceState) {

            View view = inflater.inflate(R.layout.fragment_weixin, container, false);
            recyclerView=view.findViewById(R.id.recycleview);
            initData();
            initView3();
            return view;
        }

    }

adpter适配器

adpter对前后接口进行适配,在常规recyclerview适配上增加OnItemClickListener 监听事件。

    package com.example.wechatlike;

    import android.content.Context;
    import android.content.Intent;
    import android.view.LayoutInflater;
    import android.view.View;
    import android.view.ViewGroup;
    import android.widget.TextView;

    import androidx.annotation.NonNull;
    import androidx.recyclerview.widget.RecyclerView;

    import java.util.List;

    public class MyAdapter extends RecyclerView.Adapter <MyAdapter.MyViewHolder>{
        private View itemView;
        private Context context;
        private MyAdapter myadapter;
        private List<String> list;
        private OnItemClickListener onItemClickListener;
        public MyAdapter(List<String> list, Context context){
            this.context=context;
            this.list =list;
        }

        @NonNull
        @Override
        public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
            itemView= LayoutInflater.from(context).inflate(R.layout.item,parent,false);
            MyViewHolder myViewHolder=new MyViewHolder(itemView);
            return myViewHolder;
        }

        @Override
        public void onBindViewHolder(@NonNull MyViewHolder holder, int position) {
            holder.textView1.setText(list.get(position));
            holder.itemView.setOnClickListener(new View.OnClickListener() {

                @Override
                public void onClick(View view) {
                    Intent intent = new Intent(context,Activity2.class);
                    context.startActivity(intent);
                }
            });

        }

        @Override
        public int getItemCount() {
            return list.size();
        }

        public class MyViewHolder extends RecyclerView.ViewHolder{
            TextView textView1;
            public MyViewHolder(@NonNull View itemView) {
                super(itemView);
                textView1=itemView.findViewById(R.id.textView1);

            }
        }

        public interface OnItemClickListener{
            void onItemClick(int position);
        }

        public void setOnItemClickListener(OnItemClickListener onItemClickListener){
            this.onItemClickListener = onItemClickListener;
        }
    }

结果展示

代码仓库:张润宇/WeChatlike https://gitee.com/noname-pao/we-chatlike

转自: https://blog.csdn.net/weixin_46093316/article/details/106138435

总结目录

视图篇
    如何理解非主线程可以更新UI
    dialogFragment 全屏时左右留空的解决方案
    dialogFragment 全屏时状态栏出现黑色布局的解决方案
    多个fragment 切换重叠的解决方案
    多个fragment 保存状态时可能出现 TransactionTooLargeException 的解决方案
    recyclerview 调用 notifyItemRemoved 方法移除某个 Item 之后因为引用 position 引起 crash 的原因
    recyclerview 局部刷新Item时会因为默认动画导致闪烁的解决方案
    recyclerview 中的 item 出现莫名的偏移滚动
    recyclerview 内容超过一屏时,findFistCompletelyVisibleItemPosition 会返回 -1 的原因
    textview 中富文本点击事件拦截了长按事件的解决方案
    如何禁止 ViewPager 的滑动
    如何仿蘑菇街/马蜂窝 Viewpager 装载图片之后切换时动态变更高度
    如何控制 appbarLayout 随时定位到某个位置
    如何禁止 appbarLayout 滚动
    edittext 未能响应 onClickListener 事件的解决方案
    使用 listview 或gridview 的处理 item 的 state_selected 事件是无效的解决方案
    解决5.0以上Button自带阴影效果的方案
    针对 onSingleTapUp 和 onSIngleTapConfirmed 的使用区别
    如何使用layer-list画三角形
    关于属性动画中旋转 View 时部分机型出现 View 闪烁的解决方案
    关于 ConstraintLayout 的代码布局下的注意事项
    TextView 在 6.0 版本下设置单行尾部缩略的坑
服务篇
    后台手动清理应用之后,service中启动的notifications并没有消失的解决方案
    全局的Context使用更为优雅的获取方案
线程篇
    建议新起线程不要随便调用网络请求,一般的newThread没有looper队列,参考handlerThread
网络篇
    Retorfit get 请求参数出现错误的解决方案
数据篇
    如何优雅处理 sqlite 多线程读写问题
    如何理解 Intent 传递数据出现 TransactionTooLargeException 的问题
机型系统适配篇
    如何解决 NatigationBar 被 PopupWindow 遮挡的问题
    如何解决MIUI系统后台无法 toast 的问题
    如何解决关闭通知栏权限无法弹出 toast 的问题
    如何适配 vivo 等双面屏幕
    如何解决华为设备产生太多 broadcast 导致crash的问题
    各种通知栏的适配方案
    解决针对魅族推送内容限制的问题
    解决从系统安装起安装应用后启动,Home 隐藏后 Launcher 重复启动的问题
    针对有launcher做为Activity的应用,在完全没有启动下收到第三方推送(小米,华为,魅族)/分享拉起的注意事项
    针对 App 多场景拉起场景下的场景判断分析
    8.0 部分 ROM 出现 Only fullscreen opaque activities can request orientation 的解决方案
    9.0 android 支持明文连接(Http)
编译构建篇
    travis-ci 高版本androidO编译遇到 license 没通过编译失败的解决方案
    Dalvik 支持的 android 版本下进行分包执行会有一些限制
    Dalvik 分包构建每一个 dex 文件时可能出现 java.lang.NoClassDefFoundError
    Java 8 methods of java.lang.Long and java.lang.Character are not desugared by D8
    databinding 中 findBinding vs getBinding 的场景区别
    版本构建出现 Gradle sync failed: Cannot choose between the following configurations of project
    gradle 配置本地离线包
    解决kvm/jvm 编译时 -classpath 遇到的分割及空格的问题
    databinding NoSuchMethodError with buildTool 3.4.0
    AS连接真机调试出现 debug info can be unavailabe 的解决方法
版本控制篇
    git 修改 commit 记录
    解决git ignore 文件不生效的问题
其他
    ExoPlayer在接听电话之后会导致原来设置的 Source 中静音状态消失了导致可能返回 app 续播的时候视频突然有声音
    AndroidStudio 提示 Please select Android SDK
    关于 Java 中字符与字节的编码关系认识
    关于 emoji 编码的长度计算问题

<<视图篇>>

如何理解非主线程可以更新UI

谷歌在 viewRootImpl 中检查更新ui的线程

void checkThread() {  
    if (mThread != Thread.currentThread()) {  
        throw new CalledFromWrongThreadException(  
                "Only the original thread that created a view hierarchy can touch its views.");  
    }  
} 

在执行onCreate的时候这个判断并没有执行到

dialogFragment 全屏时左右留空的解决方案

在 fragment#onResume 中重新调整 window 布局

android.view.WindowManager.LayoutParams lp = window.getAttributes();
lp.width = WindowManager.LayoutParams.MATCH_PARENT;
lp.height = WindowManager.LayoutParams.WRAP_CONTENT;
window.setAttributes(lp);

dialogFragment 全屏时状态栏出现黑色布局的解决方案

在主题中设置

<item name="android:windowIsFloating">true</item>

此时 window 为 wrap_content,如果出现左右空白,则考虑使用上个问题的方案。

当应用退回后台一段时间重返后,Fragment 切换重叠的解决方案

在线上项目中我们遇到一个场景:当应用按下 Home 退回后台,然后过一段时间之后从后台拉起我们的项目。极少数机型在主页进行多个 fragment 的切换时出现了 fragment 的重叠。经过定位之后发现,这些机型的运存偏小,性能偏差,出现这种现象的原因是由于内存的压力的原因,系统并不知后台的程序哪一个才需要保持运行,就会尝试回收内存占用较大的页面,当我们的页面被系统销毁时,fragmentActivity#onSaveInstanceState 被执行并保存了一些瞬态信息,比如界面 fragment 的视图信息。当我们再次拉起应用的时候,会让原来的 fragmentActivity 重建并重新构建了一个新的 fragment ,此时会叠加到已经被恢复的 fragment 之上导致重叠。

比较暴力的做法是不让 activity 保存状态,比如

 @Override
public void onSaveInstanceState(Bundle outState) {
     //直接不调用 super.onSaveInstanceState(outState);
     //或者直接传递空数据 
    super.onSaveInstanceState(new Bundle());
}

比较优雅的做法是,比如

 @Override
public void onSaveInstanceState(Bundle outState) {
     getSupportFragmentManager().putFragment(outState, you_key, CusFragment);
    super.onSaveInstanceState(outState);
}

//在onCreate的时候判断是否已经存在保存的信息
 @Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    if (savedInstanceState != null) {
        CusFragment fragment =  (CusFragment)getSupportFragmentManager().getFragment(savedInstanceState, you_key);
    } else {
          //init CusFragment
    }
}

多个fragment 保存状态时可能出现 TransactionTooLargeException 的解决方案

出现 TransactionTooLargeException 异常时,因为线上我们使用了 FragmentStatePagerAdapter 作为 fragment 适配器为了尽可能过缓存下浏览过的 fragment 以获得更好的体验,承载多个 FragmentStatePagerAdapter#saveState 会被调用并对每一个 fragment 的 bundle 数据进行保存。由于我们的 bundle 较大,并且保存下来的 bundle 并不会因为 fragment 被销毁而销毁,所以需要保存的 bundle 数据会一直增长,直到出现TransactionTooLargeException 异常. 我们参考stackoverflow相关问题 直接重载 saveState 丢弃 states 内容。

public Parcelable saveState() {
    Bundle bundle = (Bundle) super.saveState();
    bundle.putParcelableArray("states", null); // Never maintain any states from the base class, just null it out
    return bundle;
}

另外推荐 toolargetool 工具可以在开发中实时观测页面内存变化。

recyclerview 调用 notifyItemRemoved 方法移除某个 Item 之后因为引用 position 引起 crash 的原因

notifyItemRemoved方法并不会移除列表的数据源的数据项导致数据源中的数据与列表Item数目不一致,需要同步刷新数据源。

recyclerview 局部刷新Item时会因为默认动画导致闪烁的解决方案

因为recyclerview存在ItemAnimator,且在删除/更新/插入Item时会触发,可设置不支持该动画即可。

((SimpleItemAnimator)recyclerView.getItemAnimator()).setSupportsChangeAnimations(false);

recyclerview 中的 item 出现莫名的偏移滚动

这个问题经过定位存在于 viewholder 中的某个 view 可能提前获取到焦点。 同时在。alibaba-vlayout 库中也发现有人反馈改问题。 issues-255 解决的方法是在 Recyclerview中外层父布局中添加 android:descendantFocusability="blocksDescendants" 用于父布局覆盖 Recyclerview 优先抢占焦点。

recyclerview 内容超过一屏时,findFistCompletelyVisibleItemPosition 会返回 -1 的原因

原因是在 findOneVisibleChild 计算出来的 start 和 end 已经超过了 reclclerview 的 start 和 end.经过研究源码得到以下。

findFirstCompletelyVisibleItemPosition -> -1
findLastCompletelyVisibleItemPosition -> -1
findFirstVisibleItemPosition -> 正常
findLast

textview 中富文本点击事件拦截了长按事件的解决方案

这个问题常见于消息列表中,某条消息使用了ClickableSpan用于处理富媒体的点击事件,同时这个消息又需要支持长按复制。由于LinkMovementMethod方法在onTouchEvent一直返回true,可以通过自定义View.onTouchListener来替换setMovenmentMethod达到效果。

public class ClickMovementMethod implements View.OnTouchListener {
private LongClickCallback longClickCallback;

public static ClickMovementMethod newInstance() {
    return new ClickMovementMethod();
}

@Override
public boolean onTouch(final View v, MotionEvent event) {
    if (longClickCallback == null) {
        longClickCallback = new LongClickCallback(v);
    }

    TextView widget = (TextView) v;
    // MovementMethod设为空,防止消费长按事件
    widget.setMovementMethod(null);
    CharSequence text = widget.getText();
    Spannable spannable = Spannable.Factory.getInstance().newSpannable(text);
    int action = event.getAction();
    if (action == MotionEvent.ACTION_DOWN || action == MotionEvent.ACTION_UP) {
        int x = (int) event.getX();
        int y = (int) event.getY();
        x -= widget.getTotalPaddingLeft();
        y -= widget.getTotalPaddingTop();
        x += widget.getScrollX();
        y += widget.getScrollY();
        Layout layout = widget.getLayout();
        int line = layout.getLineForVertical(y);
        int off = layout.getOffsetForHorizontal(line, x);
        ClickableSpan[] link = spannable.getSpans(off, off, ClickableSpan.class);
        if (link.length != 0) {
            if (action == MotionEvent.ACTION_DOWN) {
                v.postDelayed(longClickCallback, ViewConfiguration.getLongPressTimeout());
            } else {
                v.removeCallbacks(longClickCallback);
                link[0].onClick(widget);
            }
            return true;
        }
    } else if (action == MotionEvent.ACTION_CANCEL) {
        v.removeCallbacks(longClickCallback);
    }

    return false;
}

private static class LongClickCallback implements Runnable {
    private View view;

    LongClickCallback(View view) {
        this.view = view;
    }

    @Override
    public void run() {
        // 找到能够消费长按事件的View
        View v = view;
        boolean consumed = v.performLongClick();
        while (!consumed) {
            v = (View) v.getParent();
            if (v == null) {
                break;
            }
            consumed = v.performLongClick();
        }
    }
}
}

textView.setOnTouchListener(ClickMovementMethod.newInstance());

如何禁止 ViewPager 的滑动

重写ViewPager onTouchEvent 和 onInterceptTouchEvent 并返回false,不处理任何滑动事件

@Override
public boolean onTouchEvent(MotionEvent arg0) {
    return false;
}

@Override
public boolean onInterceptTouchEvent(MotionEvent arg0) {
    return false;
}

如何仿蘑菇街/马蜂窝 Viewpager 装载图片之后切换时动态变更高度

imageViewPager 为普通的 Viewpager 对象

imageListInfo为存放图片信息的list,imageShowHeight为业务需要显示高度,通过切换时动态计算调整

        imageViewPager.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                imageViewPager.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                //根据viewpager的高度,拉伸显示图片的宽度调整高度。
                ViewGroup.LayoutParams layoutParams = imageViewPager.getLayoutParams();
                layoutParams.height = imageListInfo.imageShowHeight[0];
                imageViewPager.setLayoutParams(layoutParams);
            }
        });
        imageViewPager.setAdapter(imagePagerAdapter);
        imageViewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                if (position == imageListInfo.getImageListSize() - 1) {
                    return;
                }
                int height = (int) (imageListInfo.imageShowHeight[position] * (1 - positionOffset) + imageListInfo.imageShowHeight[position + 1] * positionOffset);
                ViewGroup.LayoutParams params = imageViewPager.getLayoutParams();
                params.height = height;
                imageViewPager.setLayoutParams(params);
            }

            @Override
            public void onPageSelected(int position) {
                if (!clickListBySelf) {
                    toSelectIndex(imageListInfo.selected, position);
                }
            }

            @Override
            public void onPageScrollStateChanged(int state) {

            }
        });

如何控制 appbarLayout 随时定位到某个位置

CoordinatorLayout.Behavior behavior =((CoordinatorLayout.LayoutParams)mAppBarLayout.getLayoutParams()).getBehavior();
if (behavior instanceof AppBarLayout.Behavior) {
      AppBarLayout.Behavior appBarLayoutBehavior = (AppBarLayout.Behavior) behavior;
      int topAndBottomOffset = appBarLayoutBehavior.getTopAndBottomOffset();
      if (topAndBottomOffset != 0) {
             appBarLayoutBehavior.setTopAndBottomOffset(0);
      }

如何禁止 appbarLayout 滚动

CoordinatorLayout.LayoutParams params = (CoordinatorLayout.LayoutParams) appBarLayout.getLayoutParams();
AppBarLayout.Behavior behavior = (AppBarLayout.Behavior) params.getBehavior();
behavior.setDragCallback(new AppBarLayout.Behavior.DragCallback() {
    @Override
    public boolean canDrag(@NonNull AppBarLayout appBarLayout) {
        return false;
    }
});

edittext 未能响应 onClickListener 事件的解决方案

Edittext监听未获取焦点的Edittext的点击事件,第一次点击触发OnFocusChangeListener,在获取焦点的情况下才能响应onClickListener

使用 listview 或gridview 的处理 item 的 state_selected 事件是无效的解决方案

在xml布局中对listview或gridview设置Android:choiceMode=“singleChoice”,并使用state_activated状态来代替state_selected状态。(2016.12.10)

解决5.0以上Button自带阴影效果的方案

在xml定义的Button中,添加以下样式定义

style="?android:attr/borderlessButtonStyle"

针对 onSingleTapUp 和 onSIngleTapConfirmed 的使用区别

前者在按下并抬起时发生,后者有一个附加条件时Android会确保点击之后在短时间内没有再次点击才会触发。常用于如果需要监听单击和双击事件。

如何使用layer-list画三角形

<layer-list xmlns:android="http://schemas.android.com/apk/res/android" >
//左
<item>
    <rotate
        android:fromDegrees="45"
        android:pivotX="85%"
        android:pivotY="135%">
        <shape android:shape="rectangle">
            <size
                android:width="16dp"
                android:height="16dp" />
            <solid android:color="#7d72ff" />
        </shape>
    </rotate>

</item>

//右
<item>
    <rotate
        android:fromDegrees="45"
        android:pivotX="15%"
        android:pivotY="-35%">
        <shape android:shape="rectangle">
            <size
                android:width="16dp"
                android:height="16dp" />
            <solid android:color="#7d72ff" />
        </shape>
    </rotate>

</item>

//上/正
<item>
    <rotate
        android:fromDegrees="45"
        android:pivotX="-40%"
        android:pivotY="80%">
        <shape android:shape="rectangle">
            <size
                android:width="16dp"
                android:height="16dp"/>
            <solid android:color="#7d72ff"/>
        </shape>
    </rotate>
</item>

//下
<item>
    <rotate
        android:fromDegrees="45"
        android:pivotX="135%"
        android:pivotY="15%">
        <shape android:shape="rectangle">
            <size
                android:width="16dp"
                android:height="16dp"/>
            <solid android:color="#7d72ff"/>
        </shape>
    </rotate>
</item>

关于属性动画中旋转 View 时部分机型出现 View 闪烁的解决方案

在大神 app 信息流快捷评论模块中,在交付快捷评论动画的时候发现,使用属性动画实现的抖动效果在部分机型上出现闪烁。而我们的实现抖动效果是通过 View.ROTATION 来实现的。经过研究,部分机型因为硬件加速的原因导致的。为动画 view 进行以下设置

view.setLayerType(View.LAYER_TYPE_HARDWARE,null);

关于 ConstraintLayout 的代码布局下的注意事项

不同于其他 ViewGroup 控制子 View 的排版,ConstraintLayout 需要构建 ConstraintSet 对象来粘合。 在手动添加子 View 的场景下,可以通过 ConstraintSet#clone(ConstraintLayout constraintLayout) 来克隆当前已有 ConstraintLayout 的排版信息,然后最后调用 ConstraintSet#applyTo(ConstraintLayout constraintLayout) 确认最终的排版信息。

TextView 在 6.0 版本下设置单行尾部缩略的坑

在大神信息流中,有一些卡片信息需要设置单行缩略。在 MTL 兼容测试过程中发现有一些机型显示异常,经过归纳及校验,这部分机型的版本都是 < 6.0。 通过在 stackoverflow 也找到了相同的问题场景 text ellipsize behavior in android version < 6.0 . 针对这部分版本的手机,我们需要在设置单行的时候把 android:maxLines="1" 改成 android:singleLine="true"。即使 IDE 提示该 API 已经过期了!

<<服务篇>>

后台手动清理应用之后,service中启动的notifications并没有消失的解决方案

从 How to remove all notifications when an android app (activity or service) is killed? 的诸多讨论中学习到, Service#onTaskRemoved 是我们的App被清理之后Service的回调。尝试过一下方法并不能达到清除的效果。
@Override
public void onTaskRemoved(Intent rootIntent) {
    super.onTaskRemoved(rootIntent);
    NotificationManager nManager = ((NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE));
    nManager.cancelAll();
}

在线上应用中,由于我们的通知类似于将军令这种有定时更新的功能,需要彻底干掉所有serivce承载的功能,下面方法可行

 @Override
public void onTaskRemoved(Intent rootIntent) {
    super.onTaskRemoved(rootIntent);
    stopSelf();
    stopForeground(true);
}

全局的Context使用更为优雅的获取方案

由于我们在优化 Application 启动时间时,打算移除 applciation 所有有关静态申明的变量,其中就包含全局 context 这个变量。我们参考的是 leakCanary 库的做法,使用 ContentProvider 来承载全局 context 的获取,原因是在 ActivityThread 的初始化流程中,ContentProvider#onCreate() 是在 Application#attachBaseContext(Context) 和 Application#onCreate() 之间的。所以获取的 context 是有效的

class ContextProvider : ContentProvider() {

    companion object {
        private lateinit var mContext: Context
        private lateinit var mApplication: Application
        fun getGlobalContext(): Context = mContext
        fun getGlobalApplication(): Application = mApplication
    }
    override fun onCreate(): Boolean {
        mContext = context!!
        mContext = context!!.applicationContext as Application
        return false
    }

    override fun insert(uri: Uri, values: ContentValues?): Uri? = null
    override fun query(uri: Uri, projection: Array<out String>?, selection: String?, selectionArgs: Array<out String>?, sortOrder: String?): Cursor? = null
    override fun update(uri: Uri, values: ContentValues?, selection: String?, selectionArgs: Array<out String>?): Int = -1
    override fun delete(uri: Uri, selection: String?, selectionArgs: Array<out String>?): Int = -1
    override fun getType(uri: Uri): String? = null
}

//manifest申明
<!-- Context提供者 -->
<provider
        android:name=".ContextProvider"
        android:authorities="${your_application_id}.contextprovider"
        android:exported="false" />

<<线程篇>>

建议新起线程不要随便调用网络请求,一般的newThread没有looper队列,参考handlerThread

<<网络篇>>

Retorfit get 请求参数出现错误的解决方案

@GET( BASE_URL + "index/login" )
Observable< LoginResult > requestLogin( @QueryMap(encoded = true) Map< String, String > params );

final Map< String, String > paramsMap = new ParamsProvider.Builder().
        append( "username", account ).
        append( "password",URLEncoded(password) ).
比如登陆,encode = true 表示未对特殊字符进行url编码,默认是false。

<<数据篇>>

如何优雅处理 sqlite 多线程读写问题

  • 一个helper实例对应一个 db connectton,这个连接能提供读连接和写连接,如果只是调用 read-only,则默认也会有写连接。
  • 一个helper实例可以在多个线程中使用,java层会使用锁机制保证线程同步,哪怕有100个线程,对数据度的调用也会被序列化
  • 如果尝试从不同 connection 同时对数据库进行写操作,则有一个会失败。并不会按照第一个写完再轮到第二个写,有一些sqlite版本甚至不会有错误提示。

    一般而言,如果要在多线程环境下使用数据库,则确保多个线程中使用的是同一个SQLiteDataBase对象,该对象对应一个db文件。

    特殊情况,如果多个 SQLiteDataBase 打开同一个 db 文件,同时使用不同线程同时写(insert,update,exexSQL)会导致在 SQLiteStatement.native_execute 方法时可能导致异常。这个异常来自本地方法里面,仅仅在Java对有对 SQLiteDataBase 进行同步锁保护。但是多线程读(query)返回的事 SQLiteCursor保存查询条件并没有立刻执行查询,仅仅在需要时加载部分数据,可以多线程不同 SQLiteDataBase 进行读。

    如果要处理上述问题,可以使用 “一个线程写,多个线程同时读,每个线程都用各自SQLiteOpenHelper。”

    在android 3.0版本以上 打开 enableWriteAheadLogging。当打开时,它允许一个写线程与多个读线程同时在一个SQLiteDatabase上起作用。实现原理是写操作其实是在一个单独的文件,不是原数据库文件。所以写在执行时,不会影响读操作,读操作读的是原数据文件,是写操作开始之前的内容。在写操作执行成功后,会把修改合并会原数据库文件。此时读操作才能读到修改后的内容。但是这样将花费更多的内存。

如何理解 Intent 传递数据出现 TransactionTooLargeException的问题

Intent 传输数据的机制中,用到了 Binder。Intent 中的数据,会作为 Parcel 被存储在 Binder 的事务缓冲区(Binder transaction buffer)中的对象进行传输.而这个 Binder 事务缓冲区具有一个有限的固定大小,当前为 1MB。你可别以为传递 1MB 以下的数据就安全了,这里的 1MB 空间并不是当前操作独享的,而是由当前进程所共享。也就是说 Intent 在 Activity 间传输数据,本身也不适合传递太大的数据.

参考阿里 《Android 开发者手册》 对于Activity间数据通讯数据较大,避免使用Intent+Parcelable的方式,可以考虑使用EventBus等代替方案,避免 TransactionTooLargeException。EventBus使用黏性事件来解决,但是针对Activity重建又能拿到Intent而EventBus则不可以,所以需要根据业务来调整。

<<机型系统适配篇>>

如何解决 NatigationBar 被 PopupWindow 遮挡的问题

popupWindow.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);

如何解决MIUI系统后台无法 toast 的问题

参考 github.com/zhitaocai/T… 项目,但是在小米3或者小米Note(4.4.4)手机上

mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);

mContext 需要使用 ApplicationContext 才能生效。

如何解决关闭通知栏权限无法弹出 toast 的问题

由于谷歌把 Toast 设置为系统消息权限,可以参考 Android关闭通知消息权限无法弹出Toast的解决方案 维护自己的消息队列.

如何适配 vivo 等双面屏幕

在 AndroidManifest.xml 中声明一下 meta-data

<meta-data
android:name="andriod.max_aspect" android:value="ratio_float"/>

或者使用 android:maxAspectRatio="ratio_float"(API LEVEL 26) ratio_float 一般为屏幕分辨率高宽比。 其他比如凹槽区域,圆角切割等问题可以参考市面上最新的vivo机型对应的 vivo开放平台 文档。

如何解决华为设备产生太多 broadcast 导致crash的问题

由于部分华为中,如果app注册超过500个BroadcastReceiver就会抛出 “ Register too many Broadcast Receivers ” 异常。通过分析发现其内部有一个白名单,自己可以通过创建一个新的app,使用微信包名进行测试,发现并没有这个限制。通过反射 LoadedApk 类拿到 mReceiverResource 中的 mWhiteList 对象添加我们的包名就可以了。 可以参考 github.com/llew2011/Hu… 这个项目。

各种通知栏的适配方案

参考 网易考拉实现的适配方法

解决针对魅族推送内容限制的问题

今天收到魅族渠道的警报称“推送内容可能过长”。IM功能针对离线设备走设备商的推送,魅族推送限制了title标题1-32字符,content内容1-100字符。如果频繁推送超过限制的通知,魅族推送服务器可能不会下发推送到魅族设备。故服务端限制发送到魅族服务器的消息标题和内容长度解决。

解决从系统安装起安装应用后启动,Home 隐藏后 Launcher 重复启动的问题

判断启动页面是否是根节点(推荐)

if(!isTaskRoot()){
    finish();
    return 
}

或者判断Activity是否多了 FLAG_ACTIVITY_BROUGHT_TO_FRONT ,这个tag是该场景导致的

if ((getIntent().getFlags() & Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT) != 0) {
        finish();
        return;
}

针对有launcher做为Activity的应用,在完全没有启动下收到第三方推送(小米,华为,魅族)/分享拉起的注意事项

由于我们的应用LauncherActivity用于分发不同场景的入口,A逻辑进入特殊场景页面A,B逻辑进入主页面B。

  • onCreate中优先拦截 intent 判断拉起参数,如果有拉起参数则直接进入主页面B,intent交付给主页面B处理
  • 部分场景下,比如第三方消息推送,华为和小米拉起闪屏或 launcher intent无法区分,针对该做法是,限定进入闪屏/launcher的逻辑,剩余场景统一进入主页面B
if (后端控制是否需要进入特殊场景页面) {
    boolean goToA = false;
    if (getIntent() != null) {
        String action = getIntent().getAction();
        Set<String> category = getIntent().getCategories();
        if (TextUtils.equals(action, "android.intent.action.MAIN")) {
            if (category != null && category.contains("android.intent.category.LAUNCHER")) {
                Intent intent = new Intent(this, 页面A.class);
                intent.setData(getIntent().getData());
                startActivity(intent);
                goToA = true;
            }
        }
    }
    if (! goToA) {
        goToMainActivity();
    }
} else {
    goToMainActivity();
}

针对 App 多场景拉起场景下的场景判断分析

可参考我另一篇文章 对线上项目拉起应用场景的思考总结

8.0 部分 ROM 出现 Only fullscreen opaque activities can request orientation 的解决方案

由于我们项目需要处理沉浸式,所以针对 android:windowIsTranslucent 的属性默认打开的。但是线上发现部分 8.0设备出现诡异的 crash,原因是我们对于页面的 orientation 申明都统一为 portrait 。查阅 android 源码的更新发现在 8.0 源码的逻辑里面这两个逻辑竟然不兼容,随后在 8.0 版本后谷歌进行了修复。但是国内部分 ROM 看起来并没有修复这个问题。后面同事提供了一个比较取巧的方案,通过为页面指定 android:screenOrientation=“behind” 来避免 8.0 版本的问题同时兼容所有 android 版本。

9.0 android 支持明文连接(Http)

Android 9(API级别28)开始,默认情况下禁用明文支持

<?xml version="1.0" encoding="utf-8"?>
<manifest ...>
    <uses-permission android:name="android.permission.INTERNET" />
    <application
        ...
        android:usesCleartextTraffic="true"
        ...>
        ...
    </application>
</manifest>

<<编译构建篇>>

travis-ci 高版本androidO编译遇到 license 没通过编译失败的解决方案

参考 CI 讨论区 添加 dist: precise 及 before_install 项中新增 sdkmanager指令。具体可参考我的 开源项目配置

Dalvik 支持的 android 版本下进行分包执行会有一些限制

  • 冷启动时需要安装dex文件,如果dex文件太大则可能导致处理时间太长导致 ANR
  • 即使使用 multiDex 方案在低于 4.0 系统上可能会出现 Dalvik linearAlloc 的bug,这是因为该方案需要申请一个很大的内存,运行时可能的导致程序崩溃。这个限制在 4.0 上虽然有所改善了,但是还是可能在低于 5.0 的机器上触发。

Dalvik 分包构建每一个 dex 文件时可能出现 java.lang.NoClassDefFoundError

这个问题的原因是构建工具绘制行比较复杂决策来确定主 dex 文件中需要的类以便应用能够正常的启动。如果启动期间需要的任何类在主 dex 中未能找到,则会抛出上述异常。所有必须要 multiDexKeepFile 或 multiDexKeepProguard 属性中声明他们,手动将这些类指定为主 dex 文件中的必需项。

创建 multidex-new.txt文件,写入以下新增的类

com/example/Main2.class
com/example/Main3.class

创建 meltidex-new.pro,写入以下 keep 住的类

-keep class com.example.Main2
-keep class com.example.Main3

然后在gradle multiDexKeepFile属性 和 multiDexKeepProguard属性声明上述文件

android {
buildTypes {
    release {
        multiDexKeepFile file 'multidex-new.txt'
        multiDexKeepProguard 'multidex-new.pro'
        ...
    }
}
}

Java 8 methods of java.lang.Long and java.lang.Character are not desugared by D8

这个问题出现在使用 Kotlin 编译时,从 Kotlin1.3.30 版本开始 ndroid.compileOptions中的Java版本推断出JVM目标,如果同时设置了sourceCompatibility和targetCompatibility,则选择“1.8”到那个或更高. 可以通过指定 JavaVersion 1.6 来解决这个问题。

sourceCompatibility JavaVersion.VERSION_1_6
targetCompatibility JavaVersion.VERSION_1_6

Issue 中表示,AGP(Android Gradle Plugin)3.4 已解决脱糖问题,可尝试升级解决。

databinding 中 findBinding vs getBinding 的场景区别

不同之处在于,findBinding将遍历父节点,而如果使用getBinding时当view不是跟节点会返回null。

版本构建出现 Gradle sync failed: Cannot choose between the following configurations of project

参考 issues 的回答

If you’re using Android plugin for Gradle 3.0.0 or higher, the plugin automatically matches each variant of your app with corresponding variants of its local library module dependencies for you. That is, you should no longer target specific variants of local module dependencies, show as below

 dependencies {
// Adds the 'debug' varaint of the library to the debug varaint of the app
debugCompile project(path: ':my-library-module', configuration: 'debug')

// Adds the 'release' varaint of the library to the release varaint of the app
releaseCompile project(path: ':my-library-module', configuration: 'release')
}   

gradle 配置本地离线包

  1. 离线下载 gradle 离线包保存在 url 中
  2. 修改 gradle/wrapper/gradle-wrapper.properties 调整 distributionUrl 目录指向 url
  3. 修改 build.gradle classpath 版本映射 gradle 版本

解决kvm/jvm 编译时 -classpath 遇到的分割及空格的问题

linux/mac OS 上使用 “:” 分割多个classpath路径,window使用 “;” 分割。

如果linux/mac OS 路径存在空格,暂时避免,使用多种方式尝试未果=。=。

databinding NoSuchMethodError with buildTool 3.4.0

项目从 gradle 3.1 升级到 3.4.0 并使用了 androidx 之后,发现编译失败了。来项目就是使用 databinding,编译出现了

java.lang.NoSuchMethodError: No direct method <init>
(Landroidx/databinding/DataBindingComponent;Landroid/view/View;I)V in 
class Landroidx/databinding/ViewDataBinding; or its super classes
(declaration of 'androidx.databinding.ViewDataBinding'

原因我们使用的 aar 库中使用了旧版本 gradle 编译,新版本主端 gradle 升级了,导致旧的 ViewDataBinding 构造器签名匹配不上新版 androidx.databinding.ViewDataBinding 的签名。

//旧版本
protected ViewDataBinding(DataBindingComponent bindingComponent, View root, int localFieldCount)
//新版本
protected ViewDataBinding(Object bindingComponent, View root, int localFieldCount)
幸运的是,3.4.1已经修复了。更改 3.4.0 -> 3.4.1 就可以了。

AS连接真机调试出现 debug info can be unavailabe 的解决方法

在使用 AS 连接华为真机调试的时候,IDE 一直出现 “Warning: debug info can be unavailable. Please close other application using ADB: Restart ADB integration and try again” 的错误提示。 重启 ADB 无数遍和关闭除 IDE 意外可能连接 ADB 的软件都无效,最终重启真机解决。原因是ADB连接的问题,因为有时ADB会在真实/虚拟设备上缓存一个无效的连接,并且由于该连接繁忙导致也无法连接到该设备。

<<版本控制篇>>

git 修改 commit 记录

git reset --soft HEAD^

撤销当前的commit,如果只是修改提示,则使用
git commit --amend

解决git ignore 文件不生效的问题

git rm -r --cached .
git add .
git commit -m 'update .gitignore'

<<其他>>

ExoPlayer在接听电话之后会导致原来设置的 Source 中静音状态消失了导致可能返回 app 续播的时候视频突然有声音

原因是多媒体焦点被通话抢夺之后播放音量被充值,解决方法可参考 github.com/google/ExoP…

AndroidStudio 提示 Please select Android SDK

解决手段: File->Project Structure中修改Build tools version

关于 Java 中字符与字节的编码关系认识

    // 1
    Log.d("编码测试-字符","a".length.toString())
    // 1
    Log.d("编码测试-字符","测".length.toString())
    // 5
    Log.d("编码测试-字符","测试abc".length.toString())

    // 1
    Log.d("编码测试-UTF_8","a".toByteArray(Charsets.UTF_8).size.toString())
    // 3
    Log.d("编码测试-UTF_8","测".toByteArray(Charsets.UTF_8).size.toString())
    // 9 ,UTF_8 支持使用 1,2,3,4个字节进行编码,一个中文占3个字节,一个英文占1个字节
    Log.d("编码测试-UTF_8","测试abc".toByteArray(Charsets.UTF_8).size.toString())

    // 1
    Log.d("编码测试-US_ASCII","a".toByteArray(Charsets.US_ASCII).size.toString())
    // 1
    Log.d("编码测试-US_ASCII","测".toByteArray(Charsets.US_ASCII).size.toString())
    // 5,一个中文占1个字节,一个英文占1个字节
    Log.d("编码测试-US_ASCII","测试abc".toByteArray(Charsets.US_ASCII).size.toString())

    // 1
    Log.d("编码测试-ISO_8859_1","a".toByteArray(Charsets.ISO_8859_1).size.toString())
    // 1
    Log.d("编码测试-ISO_8859_1","测".toByteArray(Charsets.ISO_8859_1).size.toString())
    // 5,一个中文占1个字节,一个英文占1个字节
    Log.d("编码测试-ISO_8859_1","测试abc".toByteArray(Charsets.ISO_8859_1).size.toString())

    // 4,存在代理对 +2个字节
    Log.d("编码测试-UTF_16","a".toByteArray(Charsets.UTF_16).size.toString())
    // 4,存在代理对 +2个字节
    Log.d("编码测试-UTF_16","测".toByteArray(Charsets.UTF_16).size.toString())
    // 12,UTF_16只支持2或者4个字节编码,一个中文占2个字节,一个英文占2个字节
    Log.d("编码测试-UTF_16","测试abc".toByteArray(Charsets.UTF_16).size.toString())

    // 2
    Log.d("编码测试-UTF_16BE","a".toByteArray(Charsets.UTF_16BE).size.toString())
    // 2
    Log.d("编码测试-UTF_16BE","测".toByteArray(Charsets.UTF_16BE).size.toString())
    // 10,一个中文占2个字节,一个英文占2个字节
    Log.d("编码测试-UTF_16BE","测试abc".toByteArray(Charsets.UTF_16BE).size.toString())

    // 2
    Log.d("编码测试-UTF_16LE","a".toByteArray(Charsets.UTF_16LE).size.toString())
    // 2
    Log.d("编码测试-UTF_16LE","测".toByteArray(Charsets.UTF_16LE).size.toString())
    // 10,一个中文占2个字节,一个英文占2个字节
    Log.d("编码测试-UTF_16LE","测试abc".toByteArray(Charsets.UTF_16LE).size.toString())

    // 8
    Log.d("编码测试-UTF_32","a".toByteArray(Charsets.UTF_32).size.toString())
    // 8
    Log.d("编码测试-UTF_32","测".toByteArray(Charsets.UTF_32).size.toString())
    // 24 utf-32支持 4个字节编码,同 utf-16 原理,一个中文占4个字节,一个英文占4个字节
    Log.d("编码测试-UTF_32","测试abc".toByteArray(Charsets.UTF_32).size.toString())

    // 4
    Log.d("编码测试-UTF_32LE","a".toByteArray(Charsets.UTF_32LE).size.toString())
    // 4
    Log.d("编码测试-UTF_32LE","测".toByteArray(Charsets.UTF_32LE).size.toString())
    //20,一个中文占4个字节,一个英文占4个字节
    Log.d("编码测试-UTF_32LE","测试abc".toByteArray(Charsets.UTF_32LE).size.toString())

    // 4
    Log.d("编码测试-UTF_32BE","a".toByteArray(Charsets.UTF_32BE).size.toString())
    // 4
    Log.d("编码测试-UTF_32BE","测".toByteArray(Charsets.UTF_32BE).size.toString())
    // 20,一个中文占4个字节,一个英文占4个字节
    Log.d("编码测试-UTF_32BE","测试abc".toByteArray(Charsets.UTF_32BE).size.toString())

关于 emoji 编码的长度计算问题

重点熟悉下 Unicode 编码标识的 emoji 下针对多平面 emoji 的拆分逻辑。

作者:yummyLau 链接:https://juejin.im/post/5eb65d7be51d4525602d393e

背景

最近在手机上面安装 orgzly 来查看 org 版本的文档,考虑同步问题,所以需要找一个 android 版本的 git 软件。

解决

找到一个比较适合的,就是 mgit,这个里面可以生成手机端的 ssh key,把这个公钥放到 服务器端,就可以进行版本同步了。

参考

用 Git 在 Android 和 Windows 间同步 Obsidian 数据库
https://sspai.com/post/68989#!

问题

手动设置了 ip 地址后,直接 ping baidu 是不通的,但是 ping baidu 对应的 ip 地址是可以的,说明问题出在 dns 上面。 但是自动设置的时候,是可以正常工作的。

解决方法

ubuntu 14 这个版本,可以通过设置里面的 网络图标,进去可以看到当前 自动模式对应的 dns 服务器,把相应的服务器手动填写到手动模式对应的文本框里面,即可。

重启网络,sudo /etc/init.d/networking restart 提示失败,可以 sudo NetworkManager restart 来获取对应的 pid,然后 kill 掉,再 sudo NetworkManager restart 一下即可。

参考:

ubuntu提示错误stop: Job failed while stopping
https://blog.csdn.net/qq_23926575/article/details/76862566

ubuntu修改配置IP地址和DNS的方法总结(4种)
https://blog.csdn.net/davidhzq/article/details/102991577

问题

org-mode 中 _ 在导出时会被认为是 下标, ^ 在导出时会被认为是上标。

解决方法

  • (setq org-use-sub-superscripts "{}") 上下标必须使用 {} 来包裹才能生效
  • #+OPTIONS: ^:nil,上下标无效, #+OPTIONS: ^:{} 上下标需要 {} 来包裹。

参考:

Org 文字有下划线时,导出来很混乱
https://emacs-china.org/t/org/936

orgmode中文粗体/斜体/下划线等等的显示
https://emacs-china.org/t/orgmode/9740/11

科研人员请进,你们平时都用emacs做些什么?
https://emacs-china.org/t/topic/5216?u=kinono

在Emacs组织模式导出中禁用下划线到下标
https://cloud.tencent.com/developer/ask/sof/118655

我的Emacs之Org-Mode
https://blog.51cto.com/lavenliu/1626566

安装

安装完成后,会自动打开 freeplaneFunctions.mm

使用

因为默认的 standard-1.6.mm 这个里面的默认字体不能显示中文,并且字体太小,所以可以直接修改这个文件。 确保 font NAME 这一行,字体为 SansSerif,大小为 24。 即可。

2

2.6 plain lists

  • 描述性的列表,可以使用 :: 作为中间分隔符
  • m-s-ret 插入一个新的条目,并且带有选择框
  • c-c c-c 如果时选择框,那么就改变状态
  • c-c - 切换列表的开头,-,+,*,1.,1)

3 tables

  • 正常的表可以在下面这种情况按 tab 来生成。
|name|phone|
|-
  • 如果想要更快的生成,可以在 |name|phone 时使用 c-c ret 来生成
  • c-c | 把区域数据转换为表格,但是不太推荐,格式不太好
  • tab, s-tab 下一个或上一个
  • ret 下一行对应的列
  • m-left 等四个方向就是移动行和列。
  • m-s-left 这样的四个方向,就是删除或增加行或者列
  • c-c - 也是向下增加一行
  • c-c ret 下乡增加一行

4 hyperlinks

连接的种类非常多,从文件中内容,到图片,文档,邮件,聊天等等。

  • c-c l 保存文件中连接
  • c-c c-l 增加连接。或修改连接
  • c-c c-o 打开连接
  • c-c & 从连接跳转回来。

5. todo

5.1 todo 基本

  • c-c c-t 循环 todo 的状态。
  • s-left 左右方向循环状态。
  • c-c / t 查找 todo
  • c-c a t 全部 todo 表单
  • s-m-ret 下一行新增一个 todo

5.2 todo 扩展

  • 所有 org 文件都支持的全局扩展
(setq org-todo-keywords
      '((sequence "TODO" "FEEDBACK" "VERIFY" "|" "DONE" "DELEGATED")))
(setq org-todo-keywords
      '((sequence "TODO(t)" "|" "DONE(d)")
      (sequence "REPORT(r)" "BUG(b)" "KNOWNCAUSE(k)" "|" "FIXED(f)")))
  • 只在本 org 文件中增加的扩展,需要注意,开头 #+ 这样的,需要在当前行使用 c-c c-c 来刷新,让功能立即生效。
#+TODO: TODO(t) | DONE(d)
#+TODO: REPORT(r) BUG(b) KNOWNCAUSE(k) | FIXED(f)
#+TODO: | CANCELED(c)

5.3 时间戳

  • 如果需要所有 org 都追踪 DONE,那么 (setq org-log-done 'time)
  • 如果还需要添加相应的文本,那么 (setq org-log-done 'note) 这两种只需要一个就行了。
  • 如果只是本文件需要追踪,那么 #+STARTUP: logdone,需要添加文本的话,就 #+STARTUP: logenotedone 这两种只需要一个就行了。
  • 如果需要更加详细的,那些只记录时间,那些需要填写 note,可以 #+TODO: TODO(t) WAIT(w@/!) | DONE(d!) CANCELED(c@)! 表示时间, @ 表示文本。

5.4 优先级

  • c-c , todo 可以选择优先级,A最高,C最低,空格移除优先级。
  • s-up 上下两个方向键可以调整优先级。

5.5 子 todo

  • 可以通过正常的 * 的个数来决定 todo 的任务层级
  • 在需要进度的任务主任务后面添加 [/] 或者 [%] 可以在子任务有完成的时候,自动更新。 [/] 类似于 [1/2][%] 类似于 [50%]

*** 5.6 选择框

  • 当父进度 包含一些简单步骤的时候,就可以使用子选择框。
  • 父选择框和子选择框的区别就是 缩进不同,后面也可以通过 [/] 来表示进度
  • 当子进度有部分完成的时候,父进度会变为 [-]
  • c-c c-c 用来改变复选框是否勾选
  • m-s-ret 用来生成下一行新的复选框。

转自: https://www.cnblogs.com/zhmlzhml/p/12345580.html

一只小白看了大佬的讲解视频结合其他大佬给整理的笔记再结合emacs 官方英文手册,觉得自己的知识好像增加了.

当时自己在全网搜索的时候没有看到那种纯小白的手册,本小白就写了这个,就当奉献社会了.若有不当指出,希望大佬批评指正~

需要声明一个问题,emacs 中有很多命令行操作.本小白在短暂的emacs 生涯中,确实感觉到了命令行操作的高效.可以说,打开了emacs,基本上你就不需要碰鼠标了.(多么美好的事情!!)  当然我也是逼自己用习惯了才感觉出来的.emacs 命令行在之前的博客园也写过,大家可以参考一下~~

(刚刚发现一个问题!我下文中的左右键,全都是键盘左右键!!emacs 基本不用鼠标!!!)

注意:C- 是指按住ctrl再按另一个键,M-是指按住alt再按另一个键.

打开emacs ,(本机为linux debian,windows机应该还需要别的配置...)C-x C-f 后输入一个文件名,注意,后缀是 .org

回车打开它.

基本操作:建立任务(注意注意,这玩意叫任务)

* (星号后加一个空格)是第一极

** (两个星号加空格) 是第二级

...

一直可以到十个*

按住alt敲回车.底下会自动出来同级的* 复制代码

* go home
** do my homework
** watch TV//这个就是alt 回车 之后自动变行
** drink some milk
*** add some sugar
*** 200ml
* go to school

一.显示

其实这样看起来挺难受的

那还有一个可以实现一下缩进

在最开头加

#+STARTUP: indent

(注意冒号后面有个空格)

加上之后,按C-c C-c (就是按两次C-c)

就有缩进了

把这个说完

在开头加这些东西,就会有不同的打开视图.(当然不要一下写好几个)

1 #+STARTUP: overview
2 #+STARTUP: content 
3 #+STARTUP: showall
4 #+STARTUP: showeverything

从上到下依次显示完全

(第一个是默认的,他只能显示一个星的那级,其他的被省略号代替了,当然你要是想打开,按tab,合上就是tab shift一起按.)

注意一下3,4的区别.

4可以把包括抽屉在内的全部显示.3不显示抽屉.

抽屉的代码:

1 : PROPERTIES
2 这是抽屉//两个东西之间的就是抽屉
3 : END

二.时间

如何在任务管理中插入时间呢.

时间分为激活时间和未激活时间

1.激活时间命令

C-c .(英文状态的句号)

然后他会给你一个日历,你可以手动选哪天,也可以直接写上  +1  这样的表示往后推一天.

如果需要具体时间,就自己打上就行.

激活时间是这个样子的<2001-06-15  五>,后面是星期几,如果忘记了命令,也可以手动打上这个样子,他也会变成应该的颜色.

如果打日期不知道星期几,那就把括号先补全就像这个样子<2001-06-15> 然后光标定在他身上,按C-c C-c,他自己就出来了.

并且,按住shift 再按键盘左右键,他可以变前后日期.

如果鼠标点他,他会出来另一个窗格(emacs 的多窗格特别强大),如果你不需要他,按 C-x 再按1 就行了.

2.非激活

C-c !(英文状态下的感叹号)

显示为[2001-06-15 五]

3.三种时间状态

schedule

C-c C-s

他会让选一个时间,选完了之后,和激活状态不同的是他前面会有SCHEDULE:字样(刚才试了试,自己打也可以,就是有时候会落前后的空格导致失效)

deadline

C-c C-d

closed

这个比较特殊,需要在开头加上

#+STARTUP: updone

(冒号后面有空格不要吃亏)

这个closed就是在你把完成状态定为done 之后,他的时间便不会i再显示.(下一个就讲完成状态!!!!)

三.完成状态

默认完成状态有两种 TODO DONE(颜色分别是红色和绿色)

光标放在在一个任务中,你可以规定他的状态,左边的shift配合键盘左右键,多按几下就看出来规律了.

当然,你可能不满意只有两种状态,那就自定义啊.

在开头写上

#+TODO   TODO(t)  DOWN(d)  WAIT(w)  ARRANGE(a)

(这种的弄完了以后一定要按C-c C-c,光标别忘定在那一排上)

TODO(t)  DOWN(d)  WAIT(w)  ARRANGE(a)这些完全是自己写,规格就是大写(一个字母)那一个字母是快捷键

可以用快捷键换这个状态,当然我会选择直接shift 加左右键

当然,如果你自己定义的里面没有写原来的TODO DONE那底下的这两个也会自动失去他们的颜色(就是失效)

四.标签

光标定在所需要行,C-c C-c 召唤标签,可以看出在下面出现 tags: 输入后回车就可以插入标签了.

(所以在上面的操作中,如果你不把光标定在#+那一排,你召唤的是标签)

自定义标签

开头写

#+TAGS:  学校(s)     家(h)     school (a)   home(b)

C-c C-c 之后,

再找你需要标签的那行 C-c C-c,他会给你显示你刚刚最上面输的那几个选项.你按小括号里的那个字母就可以定上他所代表的标签.

注意:可以插入多个标签,插入结束后回车.

五.agenda

这样就算是写完了,C -x C-s 保存之后,C-c [ (英文模式的左中括号)将他设置成agenda 格式

然后M-x 输入 org-agenda 他会询问你显示什么,你根据他的询问按他前面的那个字母就行了.

  

总之,这是一个我觉得挺漂亮还很强大的一个编辑器.

多说一句,用了emacs 以后很容易发现Ctrl键是多么的脆弱,每天都在不停地被戳.

希望这篇能对大家有帮助吧!

为了自己,和那些爱你的人