/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * See LICENSE.txt included in this distribution for the specific * language governing permissions and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at LICENSE.txt. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved. * Portions Copyright (c) 2017, Chris Fraire . */ package org.opengrok.indexer.analysis.swift; import java.io.IOException; import org.opengrok.indexer.analysis.JFlexSymbolMatcher; import org.opengrok.indexer.analysis.ScopeAction; import org.opengrok.indexer.util.StringUtils; import org.opengrok.indexer.web.HtmlConsts; %% %public %class SwiftXref %extends JFlexSymbolMatcher %unicode %int %char %include ../CommonLexer.lexh %include ../CommonXref.lexh %{ /* Must match {WhiteSpace} regex */ private final static String WHITE_SPACE = "[ \\t\\f]+"; private int nestedComment; @Override public void reset() { super.reset(); nestedComment = 0; } @Override public void yypop() throws IOException { onDisjointSpanChanged(null, yychar); super.yypop(); } protected void chkLOC() { switch (yystate()) { case COMMENT: case SCOMMENT: case SDOC: break; default: phLOC(); break; } } %} File = [a-zA-Z]{FNameChar}* "." ([Jj][Aa][Vv][Aa] | [Pp][Rr][Oo][Pp][Ee][Rr][Tt][Ii][Ee][Ss] | [Pp][Rr][Oo][Pp][Ss] | [Xx][Mm][Ll] | [Cc][Oo][Nn][Ff] | [Tt][Xx][Tt] | [Hh][Tt][Mm][Ll]? | [Ii][Nn][Ii] | [Jj][Nn][Ll][Pp] | [Jj][Aa][Dd] | [Dd][Ii][Ff][Ff] | [Pp][Aa][Tt][Cc][Hh]) /* TODO support markdown in comments SdocWithClassArg = "@throws" | "@exception" SdocWithParamNameArg = "@param" ClassName = "class"{WhiteSpace}({Identifier} ".")* {Identifier} ParamName = {Identifier} | "<" {Identifier} ">" */ %state STRING COMMENT SCOMMENT SDOC TSTRING %include ../Common.lexh %include ../CommonURI.lexh %include ../CommonPath.lexh %include Swift.lexh %% { \{ { chkLOC(); onScopeChanged(ScopeAction.INC, yytext(), yychar); } \} { chkLOC(); onScopeChanged(ScopeAction.DEC, yytext(), yychar); } \; { chkLOC(); onScopeChanged(ScopeAction.END, yytext(), yychar); } {Identifier} { chkLOC(); String id = yytext(); onFilteredSymbolMatched(id, yychar, Consts.kwd); } [`] {Identifier} [`] { chkLOC(); String capture = yytext(); String id = capture.substring(1, capture.length() - 1); onNonSymbolMatched("`", yychar); onFilteredSymbolMatched(id, yychar, null); onNonSymbolMatched("`", yychar); } {ImplicitIdentifier} { chkLOC(); onKeywordMatched(yytext(), yychar); } "<" ({File}|{FPath}) ">" { chkLOC(); onNonSymbolMatched("<", yychar); String path = yytext(); path = path.substring(1, path.length() - 1); onFilelikeMatched(path, yychar + 1); onNonSymbolMatched(">", yychar + 1 + path.length()); } /*{Hier} { onPathlikeMatched(yytext(), '.', false, yychar); } */ {Number} { chkLOC(); onDisjointSpanChanged(HtmlConsts.NUMBER_CLASS, yychar); onNonSymbolMatched(yytext(), yychar); onDisjointSpanChanged(null, yychar); } \" { chkLOC(); yypush(STRING); onDisjointSpanChanged(HtmlConsts.STRING_CLASS, yychar); onNonSymbolMatched(yytext(), yychar); } \"\"\" { chkLOC(); yypush(TSTRING); onDisjointSpanChanged(HtmlConsts.STRING_CLASS, yychar); onNonSymbolMatched(yytext(), yychar); } "/**" / [^/] { if (nestedComment++ == 0) { yypush(SDOC); onDisjointSpanChanged(HtmlConsts.COMMENT_CLASS, yychar); } onNonSymbolMatched(yytext(), yychar); } "//" { yypush(SCOMMENT); onDisjointSpanChanged(HtmlConsts.COMMENT_CLASS, yychar); onNonSymbolMatched(yytext(), yychar); } } { \\[\"\\] { chkLOC(); onNonSymbolMatched(yytext(), yychar); } \" { chkLOC(); onNonSymbolMatched(yytext(), yychar); yypop(); } } { \"\"\" { chkLOC(); onNonSymbolMatched(yytext(), yychar); yypop(); } } { {WhspChar}*{EOL} { onDisjointSpanChanged(null, yychar); onEndOfLineMatched(yytext(), yychar); onDisjointSpanChanged(HtmlConsts.STRING_CLASS, yychar); } } { "/*" { if (nestedComment++ == 0) { yypush(COMMENT); onDisjointSpanChanged(HtmlConsts.COMMENT_CLASS, yychar); } onNonSymbolMatched(yytext(), yychar); } } { "*/" { onNonSymbolMatched(yytext(), yychar); if (--nestedComment == 0) { yypop(); } } {WhspChar}*{EOL} { onDisjointSpanChanged(null, yychar); onEndOfLineMatched(yytext(), yychar); onDisjointSpanChanged(HtmlConsts.COMMENT_CLASS, yychar); } } /* TODO: support markdown in comments { {SdocWithParamNameArg} {WhiteSpace} {ParamName} | {SdocWithClassArg} {WhiteSpace} {ClassName} { String text = yytext(); String[] tokens = text.split(WHITE_SPACE, 2); onNonSymbolMatched(tokens[0], EmphasisHint.STRONG, yychar); onNonSymbolMatched(text.substring(tokens[0].length(), text.length() - tokens[1].length()), yychar); onNonSymbolMatched(tokens[1], EmphasisHint.EM, yychar); } "@" {Identifier} { onNonSymbolMatched(yytext(), EmphasisHint.STRONG, yychar); } } */ { {WhspChar}*{EOL} { yypop(); onEndOfLineMatched(yytext(), yychar); } } { {WhspChar}*{EOL} { onEndOfLineMatched(yytext(), yychar); } [[\s]--[\n]] { onNonSymbolMatched(yytext(), yychar); } [^\n] { chkLOC(); onNonSymbolMatched(yytext(), yychar); } } { {FPath} { chkLOC(); onPathlikeMatched(yytext(), '/', false, yychar); } {File} { chkLOC(); String path = yytext(); onFilelikeMatched(path, yychar); } {FNameChar}+ "@" {FNameChar}+ "." {FNameChar}+ { chkLOC(); onEmailAddressMatched(yytext(), yychar); } } { {BrowseableURI} { chkLOC(); onUriMatched(yytext(), yychar); } } { {BrowseableURI} { onUriMatched(yytext(), yychar, StringUtils.END_C_COMMENT); } }